18eef2ab6SToomas Soome /*
2199767f8SToomas Soome  * Copyright (c) 1998 Michael Smith <msmith@freebsd.org>
3199767f8SToomas Soome  * All rights reserved.
4199767f8SToomas Soome  *
5199767f8SToomas Soome  * Redistribution and use in source and binary forms, with or without
6199767f8SToomas Soome  * modification, are permitted provided that the following conditions
7199767f8SToomas Soome  * are met:
8199767f8SToomas Soome  * 1. Redistributions of source code must retain the above copyright
9199767f8SToomas Soome  *    notice, this list of conditions and the following disclaimer.
10199767f8SToomas Soome  * 2. Redistributions in binary form must reproduce the above copyright
11199767f8SToomas Soome  *    notice, this list of conditions and the following disclaimer in the
12199767f8SToomas Soome  *    documentation and/or other materials provided with the distribution.
13199767f8SToomas Soome  *
14199767f8SToomas Soome  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15199767f8SToomas Soome  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16199767f8SToomas Soome  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17199767f8SToomas Soome  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18199767f8SToomas Soome  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19199767f8SToomas Soome  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20199767f8SToomas Soome  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21199767f8SToomas Soome  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22199767f8SToomas Soome  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23199767f8SToomas Soome  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24199767f8SToomas Soome  * SUCH DAMAGE.
25199767f8SToomas Soome  */
26199767f8SToomas Soome 
27199767f8SToomas Soome #include <sys/cdefs.h>
28199767f8SToomas Soome 
29199767f8SToomas Soome /*
30199767f8SToomas Soome  * Obtain memory configuration information from the BIOS
31199767f8SToomas Soome  */
32199767f8SToomas Soome #include <stand.h>
33199767f8SToomas Soome #include <sys/param.h>
34199767f8SToomas Soome #include <sys/linker.h>
35199767f8SToomas Soome #include <sys/queue.h>
36199767f8SToomas Soome #include <sys/stddef.h>
37199767f8SToomas Soome #include <machine/metadata.h>
38199767f8SToomas Soome #include <machine/pc/bios.h>
39199767f8SToomas Soome #include "bootstrap.h"
40199767f8SToomas Soome #include "libi386.h"
41199767f8SToomas Soome #include "btxv86.h"
42199767f8SToomas Soome 
43199767f8SToomas Soome struct smap_buf {
44199767f8SToomas Soome 	struct bios_smap	smap;
45199767f8SToomas Soome 	uint32_t		xattr;	/* Extended attribute from ACPI 3.0 */
46199767f8SToomas Soome 	STAILQ_ENTRY(smap_buf)	bufs;
47199767f8SToomas Soome };
48199767f8SToomas Soome 
49199767f8SToomas Soome #define	SMAP_BUFSIZE		offsetof(struct smap_buf, bufs)
50199767f8SToomas Soome 
51199767f8SToomas Soome static struct bios_smap		*smapbase;
52199767f8SToomas Soome static uint32_t			*smapattr;
53199767f8SToomas Soome static u_int			smaplen;
54199767f8SToomas Soome 
55199767f8SToomas Soome void
bios_getsmap(void)56199767f8SToomas Soome bios_getsmap(void)
57199767f8SToomas Soome {
58199767f8SToomas Soome 	struct smap_buf		buf;
59199767f8SToomas Soome 	STAILQ_HEAD(smap_head, smap_buf) head =
60199767f8SToomas Soome 	    STAILQ_HEAD_INITIALIZER(head);
61199767f8SToomas Soome 	struct smap_buf		*cur, *next;
62199767f8SToomas Soome 	u_int			n, x;
63199767f8SToomas Soome 
64199767f8SToomas Soome 	STAILQ_INIT(&head);
65199767f8SToomas Soome 	n = 0;
66199767f8SToomas Soome 	x = 0;
67199767f8SToomas Soome 	v86.ebx = 0;
68199767f8SToomas Soome 	do {
69199767f8SToomas Soome 		v86.ctl = V86_FLAGS;
70199767f8SToomas Soome 		v86.addr = 0x15;
71199767f8SToomas Soome 		v86.eax = 0xe820;	/* int 0x15 function 0xe820 */
72199767f8SToomas Soome 		v86.ecx = SMAP_BUFSIZE;
73199767f8SToomas Soome 		v86.edx = SMAP_SIG;
74199767f8SToomas Soome 		v86.es = VTOPSEG(&buf);
75199767f8SToomas Soome 		v86.edi = VTOPOFF(&buf);
76199767f8SToomas Soome 		v86int();
77199767f8SToomas Soome 		if (V86_CY(v86.efl) || v86.eax != SMAP_SIG ||
78199767f8SToomas Soome 		    v86.ecx < sizeof(buf.smap) || v86.ecx > SMAP_BUFSIZE)
79199767f8SToomas Soome 			break;
80199767f8SToomas Soome 
81199767f8SToomas Soome 		next = malloc(sizeof(*next));
82199767f8SToomas Soome 		if (next == NULL)
83199767f8SToomas Soome 			break;
84199767f8SToomas Soome 		next->smap = buf.smap;
85199767f8SToomas Soome 		if (v86.ecx == SMAP_BUFSIZE) {
86199767f8SToomas Soome 			next->xattr = buf.xattr;
87199767f8SToomas Soome 			x++;
88199767f8SToomas Soome 		}
89199767f8SToomas Soome 		STAILQ_INSERT_TAIL(&head, next, bufs);
90199767f8SToomas Soome 		n++;
91199767f8SToomas Soome 	} while (v86.ebx != 0);
92199767f8SToomas Soome 	smaplen = n;
93199767f8SToomas Soome 
94199767f8SToomas Soome 	if (smaplen > 0) {
95199767f8SToomas Soome 		smapbase = malloc(smaplen * sizeof(*smapbase));
96199767f8SToomas Soome 		if (smapbase != NULL) {
97199767f8SToomas Soome 			n = 0;
98199767f8SToomas Soome 			STAILQ_FOREACH(cur, &head, bufs)
99199767f8SToomas Soome 				smapbase[n++] = cur->smap;
100199767f8SToomas Soome 		}
101199767f8SToomas Soome 		if (smaplen == x) {
102199767f8SToomas Soome 			smapattr = malloc(smaplen * sizeof(*smapattr));
103199767f8SToomas Soome 			if (smapattr != NULL) {
104199767f8SToomas Soome 				n = 0;
105199767f8SToomas Soome 				STAILQ_FOREACH(cur, &head, bufs)
106199767f8SToomas Soome 					smapattr[n++] = cur->xattr &
107199767f8SToomas Soome 					    SMAP_XATTR_MASK;
108199767f8SToomas Soome 			}
109199767f8SToomas Soome 		} else
110199767f8SToomas Soome 			smapattr = NULL;
111199767f8SToomas Soome 		cur = STAILQ_FIRST(&head);
112199767f8SToomas Soome 		while (cur != NULL) {
113199767f8SToomas Soome 			next = STAILQ_NEXT(cur, bufs);
114199767f8SToomas Soome 			free(cur);
115199767f8SToomas Soome 			cur = next;
116199767f8SToomas Soome 		}
117199767f8SToomas Soome 	}
118199767f8SToomas Soome }
119199767f8SToomas Soome 
120199767f8SToomas Soome void
bios_addsmapdata(struct preloaded_file * kfp)121199767f8SToomas Soome bios_addsmapdata(struct preloaded_file *kfp)
122199767f8SToomas Soome {
123199767f8SToomas Soome 	size_t			size;
124199767f8SToomas Soome 
125199767f8SToomas Soome 	if (smapbase == NULL || smaplen == 0)
126199767f8SToomas Soome 		return;
127199767f8SToomas Soome 	size = smaplen * sizeof(*smapbase);
128199767f8SToomas Soome 	file_addmetadata(kfp, MODINFOMD_SMAP, size, smapbase);
129199767f8SToomas Soome 	if (smapattr != NULL) {
130199767f8SToomas Soome 		size = smaplen * sizeof(*smapattr);
131199767f8SToomas Soome 		file_addmetadata(kfp, MODINFOMD_SMAP_XATTR, size, smapattr);
132199767f8SToomas Soome 	}
133199767f8SToomas Soome }
134199767f8SToomas Soome 
135199767f8SToomas Soome COMMAND_SET(smap, "smap", "show BIOS SMAP", command_smap);
136199767f8SToomas Soome 
137199767f8SToomas Soome static int
command_smap(int argc __unused,char * argv[]__unused)1388eef2ab6SToomas Soome command_smap(int argc __unused, char *argv[] __unused)
139199767f8SToomas Soome {
140199767f8SToomas Soome 	u_int			i;
141199767f8SToomas Soome 
142199767f8SToomas Soome 	if (smapbase == NULL || smaplen == 0)
143199767f8SToomas Soome 		return (CMD_ERROR);
144199767f8SToomas Soome 	if (smapattr != NULL)
145199767f8SToomas Soome 		for (i = 0; i < smaplen; i++)
146199767f8SToomas Soome 			printf("SMAP type=%02x base=%016llx len=%016llx attr=%02x\n",
147199767f8SToomas Soome 			    (unsigned int)smapbase[i].type,
148199767f8SToomas Soome 			    (unsigned long long)smapbase[i].base,
149199767f8SToomas Soome 			    (unsigned long long)smapbase[i].length,
150199767f8SToomas Soome 			    (unsigned int)smapattr[i]);
151199767f8SToomas Soome 	else
152199767f8SToomas Soome 		for (i = 0; i < smaplen; i++)
153199767f8SToomas Soome 			printf("SMAP type=%02x base=%016llx len=%016llx\n",
154199767f8SToomas Soome 			    (unsigned int)smapbase[i].type,
155199767f8SToomas Soome 			    (unsigned long long)smapbase[i].base,
156199767f8SToomas Soome 			    (unsigned long long)smapbase[i].length);
157199767f8SToomas Soome 	return (CMD_OK);
158199767f8SToomas Soome }
159