2 * Copyright (c) 2009 The DragonFly Project. All rights reserved.
4 * This code is derived from software contributed to The DragonFly Project
5 * by Sepherosa Ziehau <sepherosa@gmail.com>
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in
15 * the documentation and/or other materials provided with the
17 * 3. Neither the name of The DragonFly Project nor the names of its
18 * contributors may be used to endorse or promote products derived
19 * from this software without specific, prior written permission.
21 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
25 * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26 * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
27 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
29 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
30 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
31 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
35 #include <sys/param.h>
37 #include <sys/kernel.h>
38 #include <sys/systm.h>
40 #include <machine/pmap.h>
41 #include <machine/smp.h>
42 #include <machine/md_var.h>
43 #include <machine/specialreg.h>
44 #include <machine_base/apic/mpapic.h>
47 #include "acpi_sdt_var.h"
49 #define SDT_VPRINTF(fmt, arg...) \
52 kprintf("ACPI SDT: " fmt , ##arg); \
55 #define ACPI_RSDP_EBDA_MAPSZ 1024
56 #define ACPI_RSDP_BIOS_MAPSZ 0x20000
57 #define ACPI_RSDP_BIOS_MAPADDR 0xe0000
59 #define ACPI_RSDP_ALIGN 16
61 #define ACPI_RSDP_SIGLEN 8
62 #define ACPI_RSDP_SIG "RSD PTR "
64 /* Root System Description Pointer */
66 uint8_t rsdp_sig[ACPI_RSDP_SIGLEN];
68 uint8_t rsdp_oem_id[6];
73 uint8_t rsdp_ext_cksum;
77 /* Extended System Description Table */
79 struct acpi_sdth xsdt_hdr;
80 uint64_t xsdt_ents[1];
83 /* Root System Description Table */
85 struct acpi_sdth rsdt_hdr;
86 uint32_t rsdt_ents[1];
89 typedef vm_paddr_t (*sdt_search_t)(vm_paddr_t, const uint8_t *);
91 static const struct acpi_rsdp *sdt_rsdp_search(const uint8_t *, int);
92 static vm_paddr_t sdt_search_xsdt(vm_paddr_t, const uint8_t *);
93 static vm_paddr_t sdt_search_rsdt(vm_paddr_t, const uint8_t *);
95 extern u_long ebda_addr;
97 static sdt_search_t sdt_search_func;
98 static vm_paddr_t sdt_search_paddr;
103 const struct acpi_rsdp *rsdp;
107 if (ebda_addr != 0) {
108 mapsz = ACPI_RSDP_EBDA_MAPSZ;
109 ptr = pmap_mapdev(ebda_addr, mapsz);
111 rsdp = sdt_rsdp_search(ptr, mapsz);
113 SDT_VPRINTF("RSDP not in EBDA\n");
114 pmap_unmapdev((vm_offset_t)ptr, mapsz);
119 SDT_VPRINTF("RSDP in EBDA\n");
124 mapsz = ACPI_RSDP_BIOS_MAPSZ;
125 ptr = pmap_mapdev(ACPI_RSDP_BIOS_MAPADDR, mapsz);
127 rsdp = sdt_rsdp_search(ptr, mapsz);
129 kprintf("sdt_probe: no RSDP\n");
130 pmap_unmapdev((vm_offset_t)ptr, mapsz);
133 SDT_VPRINTF("RSDP in BIOS mem\n");
137 if (rsdp->rsdp_rev != 2) {
138 sdt_search_func = sdt_search_rsdt;
139 sdt_search_paddr = rsdp->rsdp_rsdt;
141 sdt_search_func = sdt_search_xsdt;
142 sdt_search_paddr = rsdp->rsdp_xsdt;
144 pmap_unmapdev((vm_offset_t)ptr, mapsz);
146 SYSINIT(sdt_probe, SI_BOOT2_PRESMP, SI_ORDER_FIRST, sdt_probe, 0);
148 static const struct acpi_rsdp *
149 sdt_rsdp_search(const uint8_t *target, int size)
151 const struct acpi_rsdp *rsdp;
154 KKASSERT(size > sizeof(*rsdp));
156 for (i = 0; i < size - sizeof(*rsdp); i += ACPI_RSDP_ALIGN) {
157 rsdp = (const struct acpi_rsdp *)&target[i];
158 if (memcmp(rsdp->rsdp_sig, ACPI_RSDP_SIG,
159 ACPI_RSDP_SIGLEN) == 0)
166 sdt_sdth_map(vm_paddr_t paddr)
168 struct acpi_sdth *sdth;
171 sdth = pmap_mapdev(paddr, sizeof(*sdth));
172 mapsz = sdth->sdth_len;
173 pmap_unmapdev((vm_offset_t)sdth, sizeof(*sdth));
175 if (mapsz < sizeof(*sdth))
178 return pmap_mapdev(paddr, mapsz);
182 sdt_sdth_unmap(struct acpi_sdth *sdth)
184 pmap_unmapdev((vm_offset_t)sdth, sdth->sdth_len);
188 sdt_search_xsdt(vm_paddr_t xsdt_paddr, const uint8_t *sig)
190 struct acpi_xsdt *xsdt;
191 vm_paddr_t sdt_paddr = 0;
194 if (xsdt_paddr == 0) {
195 kprintf("sdt_search_xsdt: XSDT paddr == 0\n");
199 xsdt = sdt_sdth_map(xsdt_paddr);
201 kprintf("sdt_search_xsdt: can't map XSDT\n");
205 if (memcmp(xsdt->xsdt_hdr.sdth_sig, ACPI_XSDT_SIG,
206 ACPI_SDTH_SIGLEN) != 0) {
207 kprintf("sdt_search_xsdt: not XSDT\n");
211 if (xsdt->xsdt_hdr.sdth_rev != 1) {
212 kprintf("sdt_search_xsdt: unsupported XSDT revision %d\n",
213 xsdt->xsdt_hdr.sdth_rev);
217 nent = (xsdt->xsdt_hdr.sdth_len - sizeof(xsdt->xsdt_hdr)) /
218 sizeof(xsdt->xsdt_ents[0]);
219 for (i = 0; i < nent; ++i) {
220 struct acpi_sdth *sdth;
222 if (xsdt->xsdt_ents[i] == 0)
225 sdth = sdt_sdth_map(xsdt->xsdt_ents[i]);
229 ret = memcmp(sdth->sdth_sig, sig, ACPI_SDTH_SIGLEN);
230 sdt_sdth_unmap(sdth);
233 sdt_paddr = xsdt->xsdt_ents[i];
239 sdt_sdth_unmap(&xsdt->xsdt_hdr);
244 sdt_search_rsdt(vm_paddr_t rsdt_paddr, const uint8_t *sig)
246 struct acpi_rsdt *rsdt;
247 vm_paddr_t sdt_paddr = 0;
250 if (rsdt_paddr == 0) {
251 kprintf("sdt_search_rsdt: RSDT paddr == 0\n");
255 rsdt = sdt_sdth_map(rsdt_paddr);
257 kprintf("sdt_search_rsdt: can't map RSDT\n");
261 if (memcmp(rsdt->rsdt_hdr.sdth_sig, ACPI_RSDT_SIG,
262 ACPI_SDTH_SIGLEN) != 0) {
263 kprintf("sdt_search_rsdt: not RSDT\n");
267 if (rsdt->rsdt_hdr.sdth_rev != 1) {
268 kprintf("sdt_search_rsdt: unsupported RSDT revision %d\n",
269 rsdt->rsdt_hdr.sdth_rev);
273 nent = (rsdt->rsdt_hdr.sdth_len - sizeof(rsdt->rsdt_hdr)) /
274 sizeof(rsdt->rsdt_ents[0]);
275 for (i = 0; i < nent; ++i) {
276 struct acpi_sdth *sdth;
278 if (rsdt->rsdt_ents[i] == 0)
281 sdth = sdt_sdth_map(rsdt->rsdt_ents[i]);
285 ret = memcmp(sdth->sdth_sig, sig, ACPI_SDTH_SIGLEN);
286 sdt_sdth_unmap(sdth);
289 sdt_paddr = rsdt->rsdt_ents[i];
295 sdt_sdth_unmap(&rsdt->rsdt_hdr);
300 sdt_search(const uint8_t *sig)
302 if (sdt_search_func == NULL)
304 return sdt_search_func(sdt_search_paddr, sig);