/* * Copyright 2019-2021 Haiku, Inc. All rights reserved. * Distributed under the terms of the MIT License. * * Authors: * Alexander von Gluck IV */ #include #include #include #include #include extern "C" { #include } #include "dtb.h" const struct supported_interrupt_controllers { const char* dtb_compat; const char* kind; } kSupportedInterruptControllers[] = { { "arm,cortex-a9-gic", INTC_KIND_GICV1 }, { "arm,cortex-a15-gic", INTC_KIND_GICV2 }, { "arm,gic-400", INTC_KIND_GICV2 }, { "ti,omap3-intc", INTC_KIND_OMAP3 }, { "marvell,pxa-intc", INTC_KIND_PXA }, { "allwinner,sun4i-a10-ic", INTC_KIND_SUN4I }, }; void arch_handle_fdt(const void* fdt, int node) { const char* deviceType = (const char*)fdt_getprop(fdt, node, "device_type", NULL); if (deviceType != NULL) { if (strcmp(deviceType, "cpu") == 0) { platform_cpu_info* info; arch_smp_register_cpu(&info); if (info == NULL) return; info->id = fdt32_to_cpu(*(uint32*)fdt_getprop(fdt, node, "reg", NULL)); dprintf("cpu\n"); dprintf(" id: %" B_PRIu32 "\n", info->id); } } int compatibleLen; const char* compatible = (const char*)fdt_getprop(fdt, node, "compatible", &compatibleLen); if (compatible == NULL) return; intc_info &interrupt_controller = gKernelArgs.arch_args.interrupt_controller; if (interrupt_controller.kind[0] == 0) { for (uint32 i = 0; i < B_COUNT_OF(kSupportedInterruptControllers); i++) { if (dtb_has_fdt_string(compatible, compatibleLen, kSupportedInterruptControllers[i].dtb_compat)) { memcpy(interrupt_controller.kind, kSupportedInterruptControllers[i].kind, sizeof(interrupt_controller.kind)); dtb_get_reg(fdt, node, 0, interrupt_controller.regs1); dtb_get_reg(fdt, node, 1, interrupt_controller.regs2); } } } } void arch_dtb_set_kernel_args(void) { 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); } }