blob: f4d5c1e25d02022e8a0a333c66638a47083ba724 [file] [log] [blame]
Simon Glass7ae69762023-05-04 16:54:58 -06001// SPDX-License-Identifier: GPL-2.0
2/*
3 * Utility functions for ACPI
4 *
5 * Copyright 2023 Google LLC
6 */
7
Simon Glass7ae69762023-05-04 16:54:58 -06008#include <mapmem.h>
9#include <acpi/acpi_table.h>
10#include <asm/global_data.h>
11
12DECLARE_GLOBAL_DATA_PTR;
13
14struct acpi_table_header *acpi_find_table(const char *sig)
15{
16 struct acpi_rsdp *rsdp;
17 struct acpi_rsdt *rsdt;
Heinrich Schuchardt23c14e12023-11-18 22:57:26 +010018 struct acpi_xsdt *xsdt;
Simon Glass7ae69762023-05-04 16:54:58 -060019 int len, i, count;
20
21 rsdp = map_sysmem(gd_acpi_start(), 0);
22 if (!rsdp)
23 return NULL;
Heinrich Schuchardt23c14e12023-11-18 22:57:26 +010024 if (rsdp->xsdt_address) {
Simon Glass919f8352023-12-31 08:25:54 -070025 xsdt = nomap_sysmem(rsdp->xsdt_address, 0);
Heinrich Schuchardt23c14e12023-11-18 22:57:26 +010026 len = xsdt->header.length - sizeof(xsdt->header);
27 count = len / sizeof(u64);
28 } else {
29 if (!rsdp->rsdt_address)
30 return NULL;
Simon Glass919f8352023-12-31 08:25:54 -070031 rsdt = nomap_sysmem(rsdp->rsdt_address, 0);
Heinrich Schuchardt23c14e12023-11-18 22:57:26 +010032 len = rsdt->header.length - sizeof(rsdt->header);
33 count = len / sizeof(u32);
34 }
Simon Glass7ae69762023-05-04 16:54:58 -060035 for (i = 0; i < count; i++) {
36 struct acpi_table_header *hdr;
37
Heinrich Schuchardt23c14e12023-11-18 22:57:26 +010038 if (rsdp->xsdt_address)
Simon Glass919f8352023-12-31 08:25:54 -070039 hdr = nomap_sysmem(xsdt->entry[i], 0);
Heinrich Schuchardt23c14e12023-11-18 22:57:26 +010040 else
Simon Glass919f8352023-12-31 08:25:54 -070041 hdr = nomap_sysmem(rsdt->entry[i], 0);
Simon Glass7ae69762023-05-04 16:54:58 -060042 if (!memcmp(hdr->signature, sig, ACPI_NAME_LEN))
43 return hdr;
44 if (!memcmp(hdr->signature, "FACP", ACPI_NAME_LEN)) {
45 struct acpi_fadt *fadt = (struct acpi_fadt *)hdr;
46
Heinrich Schuchardtfc7a0ee2023-12-16 09:12:00 +010047 if (!memcmp(sig, "DSDT", ACPI_NAME_LEN)) {
48 void *dsdt;
49
50 if (fadt->header.revision >= 3 && fadt->x_dsdt)
51 dsdt = nomap_sysmem(fadt->x_dsdt, 0);
52 else if (fadt->dsdt)
53 dsdt = nomap_sysmem(fadt->dsdt, 0);
54 else
55 dsdt = NULL;
56 return dsdt;
57 }
58
59 if (!memcmp(sig, "FACS", ACPI_NAME_LEN)) {
60 void *facs;
61
62 if (fadt->header.revision >= 3 &&
63 fadt->x_firmware_ctrl)
64 facs = nomap_sysmem(fadt->x_firmware_ctrl, 0);
65 else if (fadt->firmware_ctrl)
66 facs = nomap_sysmem(fadt->firmware_ctrl, 0);
67 else
68 facs = NULL;
69 return facs;
70 }
Simon Glass7ae69762023-05-04 16:54:58 -060071 }
72 }
73
74 return NULL;
75}