// SPDX-License-Identifier: GPL-2.0 /* * Utility functions for ACPI * * Copyright 2023 Google LLC */ #include #include #include DECLARE_GLOBAL_DATA_PTR; struct acpi_table_header *acpi_find_table(const char *sig) { struct acpi_rsdp *rsdp; struct acpi_rsdt *rsdt; struct acpi_xsdt *xsdt; int len, i, count; rsdp = map_sysmem(gd_acpi_start(), 0); if (!rsdp) return NULL; if (rsdp->xsdt_address) { xsdt = nomap_sysmem(rsdp->xsdt_address, 0); len = xsdt->header.length - sizeof(xsdt->header); count = len / sizeof(u64); } else { if (!rsdp->rsdt_address) return NULL; rsdt = nomap_sysmem(rsdp->rsdt_address, 0); len = rsdt->header.length - sizeof(rsdt->header); count = len / sizeof(u32); } for (i = 0; i < count; i++) { struct acpi_table_header *hdr; if (rsdp->xsdt_address) hdr = nomap_sysmem(xsdt->entry[i], 0); else hdr = nomap_sysmem(rsdt->entry[i], 0); if (!memcmp(hdr->signature, sig, ACPI_NAME_LEN)) return hdr; if (!memcmp(hdr->signature, "FACP", ACPI_NAME_LEN)) { struct acpi_fadt *fadt = (struct acpi_fadt *)hdr; if (!memcmp(sig, "DSDT", ACPI_NAME_LEN)) { void *dsdt; if (fadt->header.revision >= 3 && fadt->x_dsdt) dsdt = nomap_sysmem(fadt->x_dsdt, 0); else if (fadt->dsdt) dsdt = nomap_sysmem(fadt->dsdt, 0); else dsdt = NULL; return dsdt; } if (!memcmp(sig, "FACS", ACPI_NAME_LEN)) { void *facs; if (fadt->header.revision >= 3 && fadt->x_firmware_ctrl) facs = nomap_sysmem(fadt->x_firmware_ctrl, 0); else if (fadt->firmware_ctrl) facs = nomap_sysmem(fadt->firmware_ctrl, 0); else facs = NULL; return facs; } } } return NULL; }