/* * SPDX-License-Identifier: BSD-2-Clause * * Copyright (c) 2020 Western Digital Corporation or its affiliates. * * Authors: * Anup Patel */ #include #include #include #include /* List of FDT ipi drivers generated at compile time */ extern struct fdt_ipi *fdt_ipi_drivers[]; extern unsigned long fdt_ipi_drivers_size; static struct fdt_ipi *current_driver = NULL; void fdt_ipi_exit(void) { if (current_driver && current_driver->exit) current_driver->exit(); } static int fdt_ipi_warm_init(void) { if (current_driver && current_driver->warm_init) return current_driver->warm_init(); return 0; } static int fdt_ipi_cold_init(void) { int pos, noff, rc; struct fdt_ipi *drv; const struct fdt_match *match; const void *fdt = fdt_get_address(); for (pos = 0; pos < fdt_ipi_drivers_size; pos++) { drv = fdt_ipi_drivers[pos]; noff = -1; while ((noff = fdt_find_match(fdt, noff, drv->match_table, &match)) >= 0) { if (!fdt_node_is_enabled(fdt, noff)) continue; /* drv->cold_init must not be NULL */ if (drv->cold_init == NULL) return SBI_EFAIL; rc = drv->cold_init(fdt, noff, match); if (rc == SBI_ENODEV) continue; if (rc) return rc; current_driver = drv; /* * We will have multiple IPI devices on multi-die or * multi-socket systems so we cannot break here. */ } } /* * On some single-hart system there is no need for ipi, * so we cannot return a failure here */ return 0; } int fdt_ipi_init(bool cold_boot) { int rc; if (cold_boot) { rc = fdt_ipi_cold_init(); if (rc) return rc; } return fdt_ipi_warm_init(); }