/* * * Copyright (c) 2020, Hewlett Packard Enterprise Development LP. All rights reserved.
* * SPDX-License-Identifier: BSD-2-Clause * * Copyright (c) 2019 Western Digital Corporation or its affiliates. * * Authors: * Atish Patra */ #include #include #include #include #include #include #include #include #include #include #include #include #define U540_HART_COUNT FixedPcdGet32(PcdHartCount) #define U540_BOOTABLE_HART_COUNT FixedPcdGet32(PcdBootableHartNumber) #define U540_HART_STACK_SIZE FixedPcdGet32(PcdOpenSbiStackSize) #define U540_BOOT_HART_ID FixedPcdGet32(PcdBootHartId) #define U540_SYS_CLK FixedPcdGet32(PcdU5PlatformSystemClock) #define U540_PLIC_ADDR 0xc000000 #define U540_PLIC_NUM_SOURCES 0x35 #define U540_PLIC_NUM_PRIORITIES 7 #define U540_UART_ADDR FixedPcdGet32(PcdU5UartBase) #define U540_UART_BAUDRATE 115200 /* PRCI clock related macros */ //TODO: Do we need a separate driver for this ? #define U540_PRCI_BASE_ADDR 0x10000000 #define U540_PRCI_CLKMUXSTATUSREG 0x002C #define U540_PRCI_CLKMUX_STATUS_TLCLKSEL (0x1 << 1) /* Full tlb flush always */ #define U540_TLB_RANGE_FLUSH_LIMIT 0 unsigned long log2roundup(unsigned long x); static struct plic_data plic = { .addr = U540_PLIC_ADDR, .num_src = U540_PLIC_NUM_SOURCES, }; static struct clint_data clint = { .addr = CLINT_REG_BASE_ADDR, .first_hartid = 0, .hart_count = U540_HART_COUNT, .has_64bit_mmio = TRUE, }; static void U540_modify_dt(void *fdt) { fdt_cpu_fixup(fdt); fdt_fixups(fdt); /* * SiFive Freedom U540 has an erratum that prevents S-mode software * to access a PMP protected region using 1GB page table mapping, so * always add the no-map attribute on this platform. */ fdt_reserved_memory_nomap_fixup(fdt); } static int U540_final_init(bool cold_boot) { void *fdt; struct sbi_scratch *ThisScratch; if (!cold_boot) return 0; fdt = sbi_scratch_thishart_arg1_ptr(); U540_modify_dt(fdt); // // Set PMP of firmware regions to R and X. We will lock this in the end of PEI. // This region only protects SEC, PEI and Scratch buffer. // ThisScratch = sbi_scratch_thishart_ptr (); pmp_set(0, PMP_R | PMP_X | PMP_W, ThisScratch->fw_start, log2roundup (ThisScratch->fw_size)); return 0; } static u32 U540_pmp_region_count(u32 hartid) { return 1; } static int U540_pmp_region_info(u32 hartid, u32 index, ulong *prot, ulong *addr, ulong *log2size) { int ret = 0; switch (index) { case 0: *prot = PMP_R | PMP_W | PMP_X; *addr = 0; *log2size = __riscv_xlen; break; default: ret = -1; break; }; return ret; } static int U540_console_init(void) { unsigned long peri_in_freq; peri_in_freq = U540_SYS_CLK/2; return sifive_uart_init(U540_UART_ADDR, peri_in_freq, U540_UART_BAUDRATE); } static int U540_irqchip_init(bool cold_boot) { int rc; u32 hartid = current_hartid(); if (cold_boot) { rc = plic_cold_irqchip_init(&plic); if (rc) return rc; } return plic_warm_irqchip_init(&plic, (hartid) ? (2 * hartid - 1) : 0, (hartid) ? (2 * hartid) : -1); } static int U540_ipi_init(bool cold_boot) { int rc; if (cold_boot) { rc = clint_cold_ipi_init(&clint); if (rc) return rc; } return clint_warm_ipi_init(); } static u64 U540_get_tlbr_flush_limit(void) { return U540_TLB_RANGE_FLUSH_LIMIT; } static int U540_timer_init(bool cold_boot) { int rc; if (cold_boot) { rc = clint_cold_timer_init(&clint, NULL); if (rc) return rc; } return clint_warm_timer_init(); } /** * The U540 SoC has 5 HARTs, Boot HART ID is determined by * PcdBootHartId. */ static u32 U540_hart_index2id[U540_BOOTABLE_HART_COUNT] = {1, 2, 3, 4}; static int U540_system_reset(u32 type) { /* For now nothing to do. */ return 0; } const struct sbi_platform_operations platform_ops = { .pmp_region_count = U540_pmp_region_count, .pmp_region_info = U540_pmp_region_info, .final_init = U540_final_init, .console_putc = sifive_uart_putc, .console_getc = sifive_uart_getc, .console_init = U540_console_init, .irqchip_init = U540_irqchip_init, .ipi_send = clint_ipi_send, .ipi_clear = clint_ipi_clear, .ipi_init = U540_ipi_init, .get_tlbr_flush_limit = U540_get_tlbr_flush_limit, .timer_value = clint_timer_value, .timer_event_stop = clint_timer_event_stop, .timer_event_start = clint_timer_event_start, .timer_init = U540_timer_init, .system_reset = U540_system_reset }; const struct sbi_platform platform = { .opensbi_version = OPENSBI_VERSION, // The OpenSBI version this platform table is built bassed on. .platform_version = SBI_PLATFORM_VERSION(0x0001, 0x0000), // SBI Platform version 1.0 .name = "SiFive Freedom U540", .features = SBI_PLATFORM_DEFAULT_FEATURES, .hart_count = U540_BOOTABLE_HART_COUNT, .hart_index2id = U540_hart_index2id, .hart_stack_size = U540_HART_STACK_SIZE, .platform_ops_addr = (unsigned long)&platform_ops };