xref: /freebsd/sys/arm64/acpica/acpi_machdep.c (revision fdafd315)
1617994efSAndrew Turner /*-
2617994efSAndrew Turner  * Copyright (c) 2001 Mitsuru IWASAKI
3617994efSAndrew Turner  * Copyright (c) 2015 The FreeBSD Foundation
4617994efSAndrew Turner  * All rights reserved.
5617994efSAndrew Turner  *
6617994efSAndrew Turner  * This software was developed by Andrew Turner under
7617994efSAndrew Turner  * sponsorship from the FreeBSD Foundation.
8617994efSAndrew Turner  *
9617994efSAndrew Turner  * Redistribution and use in source and binary forms, with or without
10617994efSAndrew Turner  * modification, are permitted provided that the following conditions
11617994efSAndrew Turner  * are met:
12617994efSAndrew Turner  * 1. Redistributions of source code must retain the above copyright
13617994efSAndrew Turner  *    notice, this list of conditions and the following disclaimer.
14617994efSAndrew Turner  * 2. Redistributions in binary form must reproduce the above copyright
15617994efSAndrew Turner  *    notice, this list of conditions and the following disclaimer in the
16617994efSAndrew Turner  *    documentation and/or other materials provided with the distribution.
17617994efSAndrew Turner  *
18617994efSAndrew Turner  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
19617994efSAndrew Turner  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20617994efSAndrew Turner  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21617994efSAndrew Turner  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
22617994efSAndrew Turner  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23617994efSAndrew Turner  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24617994efSAndrew Turner  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25617994efSAndrew Turner  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26617994efSAndrew Turner  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27617994efSAndrew Turner  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28617994efSAndrew Turner  * SUCH DAMAGE.
29617994efSAndrew Turner  */
30617994efSAndrew Turner 
31617994efSAndrew Turner #include <sys/param.h>
32617994efSAndrew Turner #include <sys/bus.h>
33617994efSAndrew Turner #include <sys/kernel.h>
34617994efSAndrew Turner 
35617994efSAndrew Turner #include <vm/vm.h>
36617994efSAndrew Turner #include <vm/pmap.h>
37617994efSAndrew Turner 
3812425888SAndrew Turner #include <machine/machdep.h>
3912425888SAndrew Turner 
40617994efSAndrew Turner #include <contrib/dev/acpica/include/acpi.h>
41617994efSAndrew Turner #include <contrib/dev/acpica/include/accommon.h>
42617994efSAndrew Turner #include <contrib/dev/acpica/include/actables.h>
43617994efSAndrew Turner 
44617994efSAndrew Turner #include <dev/acpica/acpivar.h>
45617994efSAndrew Turner 
46eba1a249SAndrew Turner extern struct bus_space memmap_bus;
47eba1a249SAndrew Turner 
48617994efSAndrew Turner int
acpi_machdep_init(device_t dev)49617994efSAndrew Turner acpi_machdep_init(device_t dev)
50617994efSAndrew Turner {
51617994efSAndrew Turner 
52617994efSAndrew Turner 	return (0);
53617994efSAndrew Turner }
54617994efSAndrew Turner 
55617994efSAndrew Turner int
acpi_machdep_quirks(int * quirks)56617994efSAndrew Turner acpi_machdep_quirks(int *quirks)
57617994efSAndrew Turner {
58617994efSAndrew Turner 
59617994efSAndrew Turner 	return (0);
60617994efSAndrew Turner }
61617994efSAndrew Turner 
62617994efSAndrew Turner static void *
map_table(vm_paddr_t pa,const char * sig)63ee4df38dSMark Johnston map_table(vm_paddr_t pa, const char *sig)
64617994efSAndrew Turner {
65617994efSAndrew Turner 	ACPI_TABLE_HEADER *header;
6652fba9a9SJessica Clarke 	vm_size_t length;
67617994efSAndrew Turner 	void *table;
68617994efSAndrew Turner 
69617994efSAndrew Turner 	header = pmap_mapbios(pa, sizeof(ACPI_TABLE_HEADER));
70278f0de6SJung-uk Kim 	if (strncmp(header->Signature, sig, ACPI_NAMESEG_SIZE) != 0) {
717ae99f80SJohn Baldwin 		pmap_unmapbios(header, sizeof(ACPI_TABLE_HEADER));
72617994efSAndrew Turner 		return (NULL);
73617994efSAndrew Turner 	}
74617994efSAndrew Turner 	length = header->Length;
757ae99f80SJohn Baldwin 	pmap_unmapbios(header, sizeof(ACPI_TABLE_HEADER));
76617994efSAndrew Turner 
77617994efSAndrew Turner 	table = pmap_mapbios(pa, length);
7819ee8335SJung-uk Kim 	if (ACPI_FAILURE(AcpiUtChecksum(table, length))) {
79617994efSAndrew Turner 		if (bootverbose)
80617994efSAndrew Turner 			printf("ACPI: Failed checksum for table %s\n", sig);
81617994efSAndrew Turner #if (ACPI_CHECKSUM_ABORT)
82617994efSAndrew Turner 		pmap_unmapbios(table, length);
83617994efSAndrew Turner 		return (NULL);
84617994efSAndrew Turner #endif
85617994efSAndrew Turner 	}
86617994efSAndrew Turner 	return (table);
87617994efSAndrew Turner }
88617994efSAndrew Turner 
89617994efSAndrew Turner /*
90617994efSAndrew Turner  * See if a given ACPI table is the requested table.  Returns the
9143909748SJessica Clarke  * length of the table if it matches or zero on failure.
92617994efSAndrew Turner  */
93617994efSAndrew Turner static int
probe_table(vm_paddr_t address,const char * sig)94617994efSAndrew Turner probe_table(vm_paddr_t address, const char *sig)
95617994efSAndrew Turner {
96617994efSAndrew Turner 	ACPI_TABLE_HEADER *table;
97617994efSAndrew Turner 
98617994efSAndrew Turner 	table = pmap_mapbios(address, sizeof(ACPI_TABLE_HEADER));
99617994efSAndrew Turner 	if (table == NULL) {
100617994efSAndrew Turner 		if (bootverbose)
101617994efSAndrew Turner 			printf("ACPI: Failed to map table at 0x%jx\n",
102617994efSAndrew Turner 			    (uintmax_t)address);
103617994efSAndrew Turner 		return (0);
104617994efSAndrew Turner 	}
105617994efSAndrew Turner 
106278f0de6SJung-uk Kim 	if (strncmp(table->Signature, sig, ACPI_NAMESEG_SIZE) != 0) {
1077ae99f80SJohn Baldwin 		pmap_unmapbios(table, sizeof(ACPI_TABLE_HEADER));
108617994efSAndrew Turner 		return (0);
109617994efSAndrew Turner 	}
1107ae99f80SJohn Baldwin 	pmap_unmapbios(table, sizeof(ACPI_TABLE_HEADER));
111617994efSAndrew Turner 	return (1);
112617994efSAndrew Turner }
113617994efSAndrew Turner 
114617994efSAndrew Turner /* Unmap a table previously mapped via acpi_map_table(). */
115617994efSAndrew Turner void
acpi_unmap_table(void * table)116617994efSAndrew Turner acpi_unmap_table(void *table)
117617994efSAndrew Turner {
118617994efSAndrew Turner 	ACPI_TABLE_HEADER *header;
119617994efSAndrew Turner 
120617994efSAndrew Turner 	header = (ACPI_TABLE_HEADER *)table;
1217ae99f80SJohn Baldwin 	pmap_unmapbios(table, header->Length);
122617994efSAndrew Turner }
123617994efSAndrew Turner 
124617994efSAndrew Turner /*
125617994efSAndrew Turner  * Try to map a table at a given physical address previously returned
126617994efSAndrew Turner  * by acpi_find_table().
127617994efSAndrew Turner  */
128617994efSAndrew Turner void *
acpi_map_table(vm_paddr_t pa,const char * sig)129617994efSAndrew Turner acpi_map_table(vm_paddr_t pa, const char *sig)
130617994efSAndrew Turner {
131617994efSAndrew Turner 
132ee4df38dSMark Johnston 	return (map_table(pa, sig));
133617994efSAndrew Turner }
134617994efSAndrew Turner 
135617994efSAndrew Turner /*
136617994efSAndrew Turner  * Return the physical address of the requested table or zero if one
137617994efSAndrew Turner  * is not found.
138617994efSAndrew Turner  */
139617994efSAndrew Turner vm_paddr_t
acpi_find_table(const char * sig)140617994efSAndrew Turner acpi_find_table(const char *sig)
141617994efSAndrew Turner {
142617994efSAndrew Turner 	ACPI_PHYSICAL_ADDRESS rsdp_ptr;
143617994efSAndrew Turner 	ACPI_TABLE_RSDP *rsdp;
144617994efSAndrew Turner 	ACPI_TABLE_XSDT *xsdt;
145617994efSAndrew Turner 	ACPI_TABLE_HEADER *table;
146617994efSAndrew Turner 	vm_paddr_t addr;
147617994efSAndrew Turner 	int i, count;
148617994efSAndrew Turner 
149617994efSAndrew Turner 	if (resource_disabled("acpi", 0))
150617994efSAndrew Turner 		return (0);
151617994efSAndrew Turner 
152617994efSAndrew Turner 	/*
153617994efSAndrew Turner 	 * Map in the RSDP.  Since ACPI uses AcpiOsMapMemory() which in turn
154617994efSAndrew Turner 	 * calls pmap_mapbios() to find the RSDP, we assume that we can use
155617994efSAndrew Turner 	 * pmap_mapbios() to map the RSDP.
156617994efSAndrew Turner 	 */
157617994efSAndrew Turner 	if ((rsdp_ptr = AcpiOsGetRootPointer()) == 0)
158617994efSAndrew Turner 		return (0);
159617994efSAndrew Turner 	rsdp = pmap_mapbios(rsdp_ptr, sizeof(ACPI_TABLE_RSDP));
160617994efSAndrew Turner 	if (rsdp == NULL) {
161617994efSAndrew Turner 		printf("ACPI: Failed to map RSDP\n");
162617994efSAndrew Turner 		return (0);
163617994efSAndrew Turner 	}
164617994efSAndrew Turner 
165617994efSAndrew Turner 	addr = 0;
166617994efSAndrew Turner 	if (rsdp->Revision >= 2 && rsdp->XsdtPhysicalAddress != 0) {
167617994efSAndrew Turner 		/*
168617994efSAndrew Turner 		 * AcpiOsGetRootPointer only verifies the checksum for
169617994efSAndrew Turner 		 * the version 1.0 portion of the RSDP.  Version 2.0 has
170617994efSAndrew Turner 		 * an additional checksum that we verify first.
171617994efSAndrew Turner 		 */
17219ee8335SJung-uk Kim 		if (AcpiUtChecksum((UINT8 *)rsdp, ACPI_RSDP_XCHECKSUM_LENGTH)) {
173617994efSAndrew Turner 			printf("ACPI: RSDP failed extended checksum\n");
1747ae99f80SJohn Baldwin 			pmap_unmapbios(rsdp, sizeof(ACPI_TABLE_RSDP));
175617994efSAndrew Turner 			return (0);
176617994efSAndrew Turner 		}
177ee4df38dSMark Johnston 		xsdt = map_table(rsdp->XsdtPhysicalAddress, ACPI_SIG_XSDT);
178617994efSAndrew Turner 		if (xsdt == NULL) {
179617994efSAndrew Turner 			printf("ACPI: Failed to map XSDT\n");
1807ae99f80SJohn Baldwin 			pmap_unmapbios(rsdp, sizeof(ACPI_TABLE_RSDP));
181617994efSAndrew Turner 			return (0);
182617994efSAndrew Turner 		}
183617994efSAndrew Turner 		count = (xsdt->Header.Length - sizeof(ACPI_TABLE_HEADER)) /
184617994efSAndrew Turner 		    sizeof(UINT64);
185617994efSAndrew Turner 		for (i = 0; i < count; i++)
186617994efSAndrew Turner 			if (probe_table(xsdt->TableOffsetEntry[i], sig)) {
187617994efSAndrew Turner 				addr = xsdt->TableOffsetEntry[i];
188617994efSAndrew Turner 				break;
189617994efSAndrew Turner 			}
190617994efSAndrew Turner 		acpi_unmap_table(xsdt);
191195f7943SWarner Losh 	} else {
192281a6ff2SWarner Losh 		printf("ACPI: Unsupported RSDP version %d and XSDT %#lx\n",
193195f7943SWarner Losh 		    rsdp->Revision, rsdp->XsdtPhysicalAddress);
194617994efSAndrew Turner 	}
1957ae99f80SJohn Baldwin 	pmap_unmapbios(rsdp, sizeof(ACPI_TABLE_RSDP));
196617994efSAndrew Turner 
1978f6355b5SAlexander Motin 	if (addr == 0)
198617994efSAndrew Turner 		return (0);
199617994efSAndrew Turner 
200617994efSAndrew Turner 	/*
201617994efSAndrew Turner 	 * Verify that we can map the full table and that its checksum is
202617994efSAndrew Turner 	 * correct, etc.
203617994efSAndrew Turner 	 */
204ee4df38dSMark Johnston 	table = map_table(addr, sig);
205617994efSAndrew Turner 	if (table == NULL)
206617994efSAndrew Turner 		return (0);
207617994efSAndrew Turner 	acpi_unmap_table(table);
208617994efSAndrew Turner 
209617994efSAndrew Turner 	return (addr);
210617994efSAndrew Turner }
211eba1a249SAndrew Turner 
212eba1a249SAndrew Turner int
acpi_map_addr(struct acpi_generic_address * addr,bus_space_tag_t * tag,bus_space_handle_t * handle,bus_size_t size)213eba1a249SAndrew Turner acpi_map_addr(struct acpi_generic_address *addr, bus_space_tag_t *tag,
214eba1a249SAndrew Turner     bus_space_handle_t *handle, bus_size_t size)
215eba1a249SAndrew Turner {
216eba1a249SAndrew Turner 	bus_addr_t phys;
217eba1a249SAndrew Turner 
218eba1a249SAndrew Turner 	/* Check if the device is Memory mapped */
219eba1a249SAndrew Turner 	if (addr->SpaceId != 0)
220eba1a249SAndrew Turner 		return (ENXIO);
221eba1a249SAndrew Turner 
222eba1a249SAndrew Turner 	phys = addr->Address;
223eba1a249SAndrew Turner 	*tag = &memmap_bus;
224eba1a249SAndrew Turner 
225eba1a249SAndrew Turner 	return (bus_space_map(*tag, phys, size, 0, handle));
226eba1a249SAndrew Turner }
227ec630664SJayachandran C. 
228ec630664SJayachandran C. #if MAXMEMDOM > 1
229ec630664SJayachandran C. static void
parse_pxm_tables(void * dummy)230ec630664SJayachandran C. parse_pxm_tables(void *dummy)
231ec630664SJayachandran C. {
232dd6fd1d4SD Scott Phillips 	uint64_t mmfr0, parange;
233ec630664SJayachandran C. 
23412425888SAndrew Turner 	/* Only parse ACPI tables when booting via ACPI */
23512425888SAndrew Turner 	if (arm64_bus_method != ARM64_BUS_ACPI)
23612425888SAndrew Turner 		return;
23712425888SAndrew Turner 
238dd6fd1d4SD Scott Phillips 	if (!get_kernel_reg(ID_AA64MMFR0_EL1, &mmfr0)) {
239dd6fd1d4SD Scott Phillips 		/* chosen arbitrarily */
240dd6fd1d4SD Scott Phillips 		mmfr0 = ID_AA64MMFR0_PARange_1T;
241dd6fd1d4SD Scott Phillips 	}
242dd6fd1d4SD Scott Phillips 
243dd6fd1d4SD Scott Phillips 	switch (ID_AA64MMFR0_PARange_VAL(mmfr0)) {
244dd6fd1d4SD Scott Phillips 	case ID_AA64MMFR0_PARange_4G:
245dd6fd1d4SD Scott Phillips 		parange = (vm_paddr_t)4 << 30 /* GiB */;
246dd6fd1d4SD Scott Phillips 		break;
247dd6fd1d4SD Scott Phillips 	case ID_AA64MMFR0_PARange_64G:
248dd6fd1d4SD Scott Phillips 		parange = (vm_paddr_t)64 << 30 /* GiB */;
249dd6fd1d4SD Scott Phillips 		break;
250dd6fd1d4SD Scott Phillips 	case ID_AA64MMFR0_PARange_1T:
251dd6fd1d4SD Scott Phillips 		parange = (vm_paddr_t)1 << 40 /* TiB */;
252dd6fd1d4SD Scott Phillips 		break;
253dd6fd1d4SD Scott Phillips 	case ID_AA64MMFR0_PARange_4T:
254dd6fd1d4SD Scott Phillips 		parange = (vm_paddr_t)4 << 40 /* TiB */;
255dd6fd1d4SD Scott Phillips 		break;
256dd6fd1d4SD Scott Phillips 	case ID_AA64MMFR0_PARange_16T:
257dd6fd1d4SD Scott Phillips 		parange = (vm_paddr_t)16 << 40 /* TiB */;
258dd6fd1d4SD Scott Phillips 		break;
259dd6fd1d4SD Scott Phillips 	case ID_AA64MMFR0_PARange_256T:
260dd6fd1d4SD Scott Phillips 		parange = (vm_paddr_t)256 << 40 /* TiB */;
261dd6fd1d4SD Scott Phillips 		break;
262dd6fd1d4SD Scott Phillips 	case ID_AA64MMFR0_PARange_4P:
263dd6fd1d4SD Scott Phillips 		parange = (vm_paddr_t)4 << 50 /* PiB */;
264dd6fd1d4SD Scott Phillips 		break;
265dd6fd1d4SD Scott Phillips 	default:
266dd6fd1d4SD Scott Phillips 		/* chosen arbitrarily */
267dd6fd1d4SD Scott Phillips 		parange = (vm_paddr_t)1 << 40 /* TiB */;
268dd6fd1d4SD Scott Phillips 		printf("Unknown value for PARange in mmfr0 (%#lx)\n", mmfr0);
269dd6fd1d4SD Scott Phillips 		break;
270dd6fd1d4SD Scott Phillips 	}
271dd6fd1d4SD Scott Phillips 
272dd6fd1d4SD Scott Phillips 	acpi_pxm_init(MAXCPU, parange);
273ec630664SJayachandran C. 	acpi_pxm_parse_tables();
274ec630664SJayachandran C. 	acpi_pxm_set_mem_locality();
275ec630664SJayachandran C. }
276ec630664SJayachandran C. SYSINIT(parse_pxm_tables, SI_SUB_VM - 1, SI_ORDER_FIRST, parse_pxm_tables,
277ec630664SJayachandran C.     NULL);
278ec630664SJayachandran C. #endif
279