boot/efi/dtb: implement interrupt controller detection
Change-Id: I045a94c5bcb7c16297bc6fdd1fa2981e5b3f3a62 Reviewed-on: https://review.haiku-os.org/c/haiku/+/4756 Tested-by: Commit checker robot <no-reply+buildbot@haiku-os.org> Reviewed-by: Adrien Destugues <pulkomandy@gmail.com>
This commit is contained in:
parent
09a8f74d70
commit
354655e136
@ -11,6 +11,7 @@
|
|||||||
|
|
||||||
|
|
||||||
#include <util/FixedWidthPointer.h>
|
#include <util/FixedWidthPointer.h>
|
||||||
|
#include <boot/interrupt_controller.h>
|
||||||
#include <boot/uart.h>
|
#include <boot/uart.h>
|
||||||
|
|
||||||
|
|
||||||
@ -41,6 +42,7 @@ typedef struct {
|
|||||||
FixedWidthPointer<void> fdt;
|
FixedWidthPointer<void> fdt;
|
||||||
|
|
||||||
uart_info uart;
|
uart_info uart;
|
||||||
|
intc_info interrupt_controller;
|
||||||
} _PACKED arch_kernel_args;
|
} _PACKED arch_kernel_args;
|
||||||
|
|
||||||
#endif /* KERNEL_ARCH_ARM_KERNEL_ARGS_H */
|
#endif /* KERNEL_ARCH_ARM_KERNEL_ARGS_H */
|
||||||
|
26
headers/private/kernel/boot/interrupt_controller.h
Normal file
26
headers/private/kernel/boot/interrupt_controller.h
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Haiku, Inc. All rights reserved.
|
||||||
|
* Distributed under the terms of the MIT License.
|
||||||
|
*/
|
||||||
|
#ifndef KERNEL_BOOT_INTERRUPT_CONTROLLER_H
|
||||||
|
#define KERNEL_BOOT_INTERRUPT_CONTROLLER_H
|
||||||
|
|
||||||
|
|
||||||
|
#include <boot/addr_range.h>
|
||||||
|
#include <SupportDefs.h>
|
||||||
|
|
||||||
|
|
||||||
|
#define INTC_KIND_GICV1 "gicv1"
|
||||||
|
#define INTC_KIND_GICV2 "gicv2"
|
||||||
|
#define INTC_KIND_OMAP3 "omap3"
|
||||||
|
#define INTC_KIND_PXA "pxa"
|
||||||
|
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
char kind[32];
|
||||||
|
addr_range regs1;
|
||||||
|
addr_range regs2;
|
||||||
|
} __attribute__((packed)) intc_info;
|
||||||
|
|
||||||
|
|
||||||
|
#endif /* KERNEL_BOOT_INTERRUPT_CONTROLLER_H */
|
@ -73,6 +73,19 @@ const struct supported_uarts {
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
const struct supported_interrupt_controllers {
|
||||||
|
const char* dtb_compat;
|
||||||
|
const char* kind;
|
||||||
|
} kSupportedInterruptControllers[] = {
|
||||||
|
#ifdef __ARM__
|
||||||
|
{ "arm,cortex-a9-gic", INTC_KIND_GICV1 },
|
||||||
|
{ "arm,cortex-a15-gic", INTC_KIND_GICV2 },
|
||||||
|
{ "ti,omap3-intc", INTC_KIND_OMAP3 },
|
||||||
|
{ "marvell,pxa-intc", INTC_KIND_PXA },
|
||||||
|
#endif
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
static void WriteStringList(const char* prop, size_t size)
|
static void WriteStringList(const char* prop, size_t size)
|
||||||
{
|
{
|
||||||
bool first = true;
|
bool first = true;
|
||||||
@ -449,29 +462,45 @@ HandleFdt(const void* fdt, int node, uint32 addressCells, uint32 sizeCells)
|
|||||||
|
|
||||||
// TODO: We should check for the "chosen" uart and prioritize that one
|
// TODO: We should check for the "chosen" uart and prioritize that one
|
||||||
|
|
||||||
uart_info &uart = gKernelArgs.arch_args.uart;
|
|
||||||
if (uart.kind[0] != 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
// check for a uart if we don't have one
|
// check for a uart if we don't have one
|
||||||
for (uint32 i = 0; i < B_COUNT_OF(kSupportedUarts); i++) {
|
uart_info &uart = gKernelArgs.arch_args.uart;
|
||||||
if (HasFdtString(compatible, compatibleLen,
|
if (uart.kind[0] == 0) {
|
||||||
kSupportedUarts[i].dtb_compat)) {
|
for (uint32 i = 0; i < B_COUNT_OF(kSupportedUarts); i++) {
|
||||||
|
if (HasFdtString(compatible, compatibleLen,
|
||||||
|
kSupportedUarts[i].dtb_compat)) {
|
||||||
|
|
||||||
memcpy(uart.kind, kSupportedUarts[i].kind,
|
memcpy(uart.kind, kSupportedUarts[i].kind,
|
||||||
sizeof(uart.kind));
|
sizeof(uart.kind));
|
||||||
|
|
||||||
GetReg(fdt, node, addressCells, sizeCells, 0, uart.regs);
|
GetReg(fdt, node, addressCells, sizeCells, 0, uart.regs);
|
||||||
uart.irq = GetInterrupt(fdt, node);
|
uart.irq = GetInterrupt(fdt, node);
|
||||||
uart.clock = GetClockFrequency(fdt, node);
|
uart.clock = GetClockFrequency(fdt, node);
|
||||||
|
|
||||||
gUART = kSupportedUarts[i].uart_driver_init(uart.regs.start,
|
gUART = kSupportedUarts[i].uart_driver_init(uart.regs.start,
|
||||||
uart.clock);
|
uart.clock);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (gUART != NULL)
|
||||||
|
gUART->InitEarly();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (gUART != NULL)
|
intc_info &interrupt_controller = gKernelArgs.arch_args.interrupt_controller;
|
||||||
gUART->InitEarly();
|
if (interrupt_controller.kind[0] == 0) {
|
||||||
|
for (uint32 i = 0; i < B_COUNT_OF(kSupportedInterruptControllers); i++) {
|
||||||
|
if (HasFdtString(compatible, compatibleLen,
|
||||||
|
kSupportedInterruptControllers[i].dtb_compat)) {
|
||||||
|
|
||||||
|
memcpy(interrupt_controller.kind, kSupportedInterruptControllers[i].kind,
|
||||||
|
sizeof(interrupt_controller.kind));
|
||||||
|
|
||||||
|
GetReg(fdt, node, addressCells, sizeCells, 0,
|
||||||
|
interrupt_controller.regs1);
|
||||||
|
GetReg(fdt, node, addressCells, sizeCells, 1,
|
||||||
|
interrupt_controller.regs2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -558,4 +587,17 @@ dtb_set_kernel_args()
|
|||||||
dprintf(" clock: %" B_PRIu64 "\n", uart.clock);
|
dprintf(" clock: %" B_PRIu64 "\n", uart.clock);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
intc_info &interrupt_controller = gKernelArgs.arch_args.interrupt_controller;
|
||||||
|
dprintf("Chosen interrupt controller:\n");
|
||||||
|
if (interrupt_controller.kind[0] == 0) {
|
||||||
|
dprintf("kind: None!\n");
|
||||||
|
} else {
|
||||||
|
dprintf(" kind: %s\n", interrupt_controller.kind);
|
||||||
|
dprintf(" regs: %#" B_PRIx64 ", %#" B_PRIx64 "\n",
|
||||||
|
interrupt_controller.regs1.start,
|
||||||
|
interrupt_controller.regs1.size);
|
||||||
|
dprintf(" %#" B_PRIx64 ", %#" B_PRIx64 "\n",
|
||||||
|
interrupt_controller.regs2.start,
|
||||||
|
interrupt_controller.regs2.size);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user