/* $NetBSD: virt_platform.c,v 1.9 2018/10/30 16:41:52 skrll Exp $ */ /*- * Copyright (c) 2018 Jared McNeill * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ #include "opt_soc.h" #include "opt_multiprocessor.h" #include __KERNEL_RCSID(0, "$NetBSD: virt_platform.c,v 1.9 2018/10/30 16:41:52 skrll Exp $"); #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define VIRT_UART_BASE 0x09000000 void virt_platform_early_putchar(char); void virt_platform_early_putchar(char c) { volatile uint32_t *uartaddr = cpu_earlydevice_va_p() ? (volatile uint32_t *)VIRT_CORE_PTOV(VIRT_UART_BASE) : (volatile uint32_t *)VIRT_UART_BASE; while ((le32toh(uartaddr[PL01XCOM_FR / 4]) & PL01X_FR_TXFF) != 0) continue; uartaddr[PL01XCOM_DR / 4] = htole32(c); arm_dsb(); while ((le32toh(uartaddr[PL01XCOM_FR / 4]) & PL01X_FR_TXFE) == 0) continue; } static const struct pmap_devmap * virt_platform_devmap(void) { static const struct pmap_devmap devmap[] = { DEVMAP_ENTRY(VIRT_CORE_VBASE, VIRT_CORE_PBASE, VIRT_CORE_SIZE), DEVMAP_ENTRY_END }; return devmap; } static void virt_platform_init_attach_args(struct fdt_attach_args *faa) { extern struct arm32_bus_dma_tag arm_generic_dma_tag; extern struct bus_space arm_generic_bs_tag; extern struct bus_space arm_generic_a4x_bs_tag; faa->faa_bst = &arm_generic_bs_tag; faa->faa_a4x_bst = &arm_generic_a4x_bs_tag; faa->faa_dmat = &arm_generic_dma_tag; } static void virt_platform_device_register(device_t self, void *aux) { } static u_int virt_platform_uart_freq(void) { return 24000000; } static const struct arm_platform virt_platform = { .ap_devmap = virt_platform_devmap, .ap_bootstrap = arm_fdt_cpu_bootstrap, .ap_init_attach_args = virt_platform_init_attach_args, .ap_device_register = virt_platform_device_register, .ap_reset = psci_fdt_reset, .ap_delay = gtmr_delay, .ap_uart_freq = virt_platform_uart_freq, .ap_mpstart = arm_fdt_cpu_mpstart, }; ARM_PLATFORM(virt, "linux,dummy-virt", &virt_platform);