mirror of
https://github.com/RGBCube/serenity
synced 2025-07-27 09:37:34 +00:00
Kernel: Add driver for interrupt controller on the Raspberry Pi
This implements the simpler IRQController class and adds a pending_interrupts() function to the class.
This commit is contained in:
parent
f085903f62
commit
5eac2b9f33
4 changed files with 111 additions and 0 deletions
|
@ -22,6 +22,8 @@ public:
|
||||||
|
|
||||||
virtual void eoi(GenericInterruptHandler const&) const = 0;
|
virtual void eoi(GenericInterruptHandler const&) const = 0;
|
||||||
|
|
||||||
|
virtual u64 pending_interrupts() const = 0;
|
||||||
|
|
||||||
virtual StringView model() const = 0;
|
virtual StringView model() const = 0;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
69
Kernel/Arch/aarch64/RPi/InterruptController.cpp
Normal file
69
Kernel/Arch/aarch64/RPi/InterruptController.cpp
Normal file
|
@ -0,0 +1,69 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022, Timon Kruiper <timonkruiper@gmail.com>
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <AK/Format.h>
|
||||||
|
#include <Kernel/Arch/aarch64/RPi/InterruptController.h>
|
||||||
|
#include <Kernel/Arch/aarch64/RPi/MMIO.h>
|
||||||
|
#include <Kernel/Interrupts/GenericInterruptHandler.h>
|
||||||
|
|
||||||
|
namespace Kernel::RPi {
|
||||||
|
|
||||||
|
// "7.5 Interrupts Registers"
|
||||||
|
// https://github.com/raspberrypi/documentation/files/1888662/BCM2837-ARM-Peripherals.-.Revised.-.V2-1.pdf
|
||||||
|
struct InterruptControllerRegisters {
|
||||||
|
u32 irq_basic_pending;
|
||||||
|
u32 irq_pending_1;
|
||||||
|
u32 irq_pending_2;
|
||||||
|
u32 fiq_control;
|
||||||
|
|
||||||
|
u32 enable_irqs_1;
|
||||||
|
u32 enable_irqs_2;
|
||||||
|
u32 enable_basic_irqs;
|
||||||
|
|
||||||
|
u32 disable_irqs_1;
|
||||||
|
u32 disable_irqs_2;
|
||||||
|
u32 disable_basic_irqs;
|
||||||
|
};
|
||||||
|
|
||||||
|
InterruptController::InterruptController()
|
||||||
|
: m_registers(MMIO::the().peripheral<InterruptControllerRegisters>(0xB200))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void InterruptController::enable(GenericInterruptHandler const& handler)
|
||||||
|
{
|
||||||
|
u8 interrupt_number = handler.interrupt_number();
|
||||||
|
VERIFY(interrupt_number < 64);
|
||||||
|
|
||||||
|
if (interrupt_number < 32)
|
||||||
|
m_registers->enable_irqs_1 = m_registers->enable_irqs_1 | (1 << interrupt_number);
|
||||||
|
else
|
||||||
|
m_registers->enable_irqs_2 = m_registers->enable_irqs_2 | (1 << (interrupt_number - 32));
|
||||||
|
}
|
||||||
|
|
||||||
|
void InterruptController::disable(GenericInterruptHandler const& handler)
|
||||||
|
{
|
||||||
|
u8 interrupt_number = handler.interrupt_number();
|
||||||
|
VERIFY(interrupt_number < 64);
|
||||||
|
|
||||||
|
if (interrupt_number < 32)
|
||||||
|
m_registers->disable_irqs_1 = m_registers->disable_irqs_1 | (1 << interrupt_number);
|
||||||
|
else
|
||||||
|
m_registers->disable_irqs_2 = m_registers->disable_irqs_2 | (1 << (interrupt_number - 32));
|
||||||
|
}
|
||||||
|
|
||||||
|
void InterruptController::eoi(GenericInterruptHandler const&) const
|
||||||
|
{
|
||||||
|
// NOTE: The interrupt controller cannot clear the interrupt, since it is basically just a big multiplexer.
|
||||||
|
// The interrupt should be cleared by the corresponding device driver, such as a timer or uart.
|
||||||
|
}
|
||||||
|
|
||||||
|
u64 InterruptController::pending_interrupts() const
|
||||||
|
{
|
||||||
|
return ((u64)m_registers->irq_pending_2 << 32) | (u64)m_registers->irq_pending_1;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
39
Kernel/Arch/aarch64/RPi/InterruptController.h
Normal file
39
Kernel/Arch/aarch64/RPi/InterruptController.h
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022, Timon Kruiper <timonkruiper@gmail.com>
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <AK/Types.h>
|
||||||
|
#include <Kernel/Arch/aarch64/IRQController.h>
|
||||||
|
|
||||||
|
namespace Kernel::RPi {
|
||||||
|
|
||||||
|
struct InterruptControllerRegisters;
|
||||||
|
|
||||||
|
// This class implements the simple Interrupt Controller found in the BCM2837. (RPi3)
|
||||||
|
// A description of this device can be found at chapter 7 (Interrupts) of the manual:
|
||||||
|
// https://github.com/raspberrypi/documentation/files/1888662/BCM2837-ARM-Peripherals.-.Revised.-.V2-1.pdf (RPi3)
|
||||||
|
class InterruptController : public IRQController {
|
||||||
|
public:
|
||||||
|
InterruptController();
|
||||||
|
|
||||||
|
private:
|
||||||
|
virtual void enable(GenericInterruptHandler const&) override;
|
||||||
|
virtual void disable(GenericInterruptHandler const&) override;
|
||||||
|
|
||||||
|
virtual void eoi(GenericInterruptHandler const&) const override;
|
||||||
|
|
||||||
|
virtual u64 pending_interrupts() const override;
|
||||||
|
|
||||||
|
virtual StringView model() const override
|
||||||
|
{
|
||||||
|
return "Raspberry Pi Interrupt Controller";
|
||||||
|
}
|
||||||
|
|
||||||
|
InterruptControllerRegisters volatile* m_registers;
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
|
@ -401,6 +401,7 @@ else()
|
||||||
set(RPI_SOURCES
|
set(RPI_SOURCES
|
||||||
Arch/aarch64/RPi/Framebuffer.cpp
|
Arch/aarch64/RPi/Framebuffer.cpp
|
||||||
Arch/aarch64/RPi/GPIO.cpp
|
Arch/aarch64/RPi/GPIO.cpp
|
||||||
|
Arch/aarch64/RPi/InterruptController.cpp
|
||||||
Arch/aarch64/RPi/Mailbox.cpp
|
Arch/aarch64/RPi/Mailbox.cpp
|
||||||
Arch/aarch64/RPi/MMIO.cpp
|
Arch/aarch64/RPi/MMIO.cpp
|
||||||
Arch/aarch64/RPi/Timer.cpp
|
Arch/aarch64/RPi/Timer.cpp
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue