xref: /dflybsd-src/stand/boot/pc32/libi386/biospnp.c (revision 479ab7f0492f2a51b48e8537e4f1dc686fc6014b)
1*479ab7f0SSascha Wildner /*-
2*479ab7f0SSascha Wildner  * Copyright (c) 1998 Michael Smith <msmith@freebsd.org>
3*479ab7f0SSascha Wildner  * All rights reserved.
4*479ab7f0SSascha Wildner  *
5*479ab7f0SSascha Wildner  * Redistribution and use in source and binary forms, with or without
6*479ab7f0SSascha Wildner  * modification, are permitted provided that the following conditions
7*479ab7f0SSascha Wildner  * are met:
8*479ab7f0SSascha Wildner  * 1. Redistributions of source code must retain the above copyright
9*479ab7f0SSascha Wildner  *    notice, this list of conditions and the following disclaimer.
10*479ab7f0SSascha Wildner  * 2. Redistributions in binary form must reproduce the above copyright
11*479ab7f0SSascha Wildner  *    notice, this list of conditions and the following disclaimer in the
12*479ab7f0SSascha Wildner  *    documentation and/or other materials provided with the distribution.
13*479ab7f0SSascha Wildner  *
14*479ab7f0SSascha Wildner  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15*479ab7f0SSascha Wildner  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16*479ab7f0SSascha Wildner  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17*479ab7f0SSascha Wildner  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18*479ab7f0SSascha Wildner  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19*479ab7f0SSascha Wildner  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20*479ab7f0SSascha Wildner  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21*479ab7f0SSascha Wildner  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22*479ab7f0SSascha Wildner  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23*479ab7f0SSascha Wildner  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24*479ab7f0SSascha Wildner  * SUCH DAMAGE.
25*479ab7f0SSascha Wildner  *
26*479ab7f0SSascha Wildner  * $FreeBSD: src/sys/boot/i386/libi386/biospnp.c,v 1.9 2003/08/25 23:28:31 obrien Exp $
27*479ab7f0SSascha Wildner  * $DragonFly: src/sys/boot/pc32/libi386/biospnp.c,v 1.5 2008/10/03 19:56:10 swildner Exp $
28*479ab7f0SSascha Wildner  */
29*479ab7f0SSascha Wildner 
30*479ab7f0SSascha Wildner /*
31*479ab7f0SSascha Wildner  * PnP BIOS enumerator.
32*479ab7f0SSascha Wildner  */
33*479ab7f0SSascha Wildner 
34*479ab7f0SSascha Wildner #include <stand.h>
35*479ab7f0SSascha Wildner #include <machine/stdarg.h>
36*479ab7f0SSascha Wildner #include <bootstrap.h>
37*479ab7f0SSascha Wildner #include <isapnp.h>
38*479ab7f0SSascha Wildner #include <btxv86.h>
39*479ab7f0SSascha Wildner 
40*479ab7f0SSascha Wildner static int	biospnp_init(void);
41*479ab7f0SSascha Wildner static void	biospnp_enumerate(void);
42*479ab7f0SSascha Wildner 
43*479ab7f0SSascha Wildner struct pnphandler biospnphandler =
44*479ab7f0SSascha Wildner {
45*479ab7f0SSascha Wildner     "PnP BIOS",
46*479ab7f0SSascha Wildner     biospnp_enumerate
47*479ab7f0SSascha Wildner };
48*479ab7f0SSascha Wildner 
49*479ab7f0SSascha Wildner struct pnp_ICstructure
50*479ab7f0SSascha Wildner {
51*479ab7f0SSascha Wildner     u_int8_t	pnp_signature[4];
52*479ab7f0SSascha Wildner     u_int8_t	pnp_version;
53*479ab7f0SSascha Wildner     u_int8_t	pnp_length;
54*479ab7f0SSascha Wildner     u_int16_t	pnp_BIOScontrol;
55*479ab7f0SSascha Wildner     u_int8_t	pnp_checksum;
56*479ab7f0SSascha Wildner     u_int32_t	pnp_eventflag;
57*479ab7f0SSascha Wildner     u_int16_t	pnp_rmip;
58*479ab7f0SSascha Wildner     u_int16_t	pnp_rmcs;
59*479ab7f0SSascha Wildner     u_int16_t	pnp_pmip;
60*479ab7f0SSascha Wildner     u_int32_t	pnp_pmcs;
61*479ab7f0SSascha Wildner     u_int8_t	pnp_OEMdev[4];
62*479ab7f0SSascha Wildner     u_int16_t	pnp_rmds;
63*479ab7f0SSascha Wildner     u_int32_t	pnp_pmds;
64*479ab7f0SSascha Wildner } __packed;
65*479ab7f0SSascha Wildner 
66*479ab7f0SSascha Wildner struct pnp_devNode
67*479ab7f0SSascha Wildner {
68*479ab7f0SSascha Wildner     u_int16_t	dn_size;
69*479ab7f0SSascha Wildner     u_int8_t	dn_handle;
70*479ab7f0SSascha Wildner     u_int8_t	dn_id[4];
71*479ab7f0SSascha Wildner     u_int8_t	dn_type[3];
72*479ab7f0SSascha Wildner     u_int16_t	dn_attrib;
73*479ab7f0SSascha Wildner     u_int8_t	dn_data[1];
74*479ab7f0SSascha Wildner } __packed;
75*479ab7f0SSascha Wildner 
76*479ab7f0SSascha Wildner struct pnp_isaConfiguration
77*479ab7f0SSascha Wildner {
78*479ab7f0SSascha Wildner     u_int8_t	ic_revision;
79*479ab7f0SSascha Wildner     u_int8_t	ic_nCSN;
80*479ab7f0SSascha Wildner     u_int16_t	ic_rdport;
81*479ab7f0SSascha Wildner     u_int16_t	ic_reserved;
82*479ab7f0SSascha Wildner } __packed;
83*479ab7f0SSascha Wildner 
84*479ab7f0SSascha Wildner static struct pnp_ICstructure	*pnp_Icheck = NULL;
85*479ab7f0SSascha Wildner static u_int16_t		pnp_NumNodes;
86*479ab7f0SSascha Wildner static u_int16_t		pnp_NodeSize;
87*479ab7f0SSascha Wildner 
88*479ab7f0SSascha Wildner static void	biospnp_scanresdata(struct pnpinfo *pi, struct pnp_devNode *dn);
89*479ab7f0SSascha Wildner static int	biospnp_call(int func, const char *fmt, ...);
90*479ab7f0SSascha Wildner 
91*479ab7f0SSascha Wildner #define vsegofs(vptr)	(((u_int32_t)VTOPSEG(vptr) << 16) + VTOPOFF(vptr))
92*479ab7f0SSascha Wildner 
93*479ab7f0SSascha Wildner typedef void    v86bios_t(u_int32_t, u_int32_t, u_int32_t, u_int32_t);
94*479ab7f0SSascha Wildner v86bios_t	*v86bios = (v86bios_t *)v86int;
95*479ab7f0SSascha Wildner 
96*479ab7f0SSascha Wildner #define	biospnp_f00(NumNodes, NodeSize)			biospnp_call(0x00, "ll", NumNodes, NodeSize)
97*479ab7f0SSascha Wildner #define biospnp_f01(Node, devNodeBuffer, Control)	biospnp_call(0x01, "llw", Node, devNodeBuffer, Control)
98*479ab7f0SSascha Wildner #define biospnp_f40(Configuration)			biospnp_call(0x40, "l", Configuration)
99*479ab7f0SSascha Wildner 
100*479ab7f0SSascha Wildner /* PnP BIOS return codes */
101*479ab7f0SSascha Wildner #define PNP_SUCCESS			0x00
102*479ab7f0SSascha Wildner #define PNP_FUNCTION_NOT_SUPPORTED	0x80
103*479ab7f0SSascha Wildner 
104*479ab7f0SSascha Wildner /*
105*479ab7f0SSascha Wildner  * Initialisation: locate the PnP BIOS, test that we can call it.
106*479ab7f0SSascha Wildner  * Returns nonzero if the PnP BIOS is not usable on this system.
107*479ab7f0SSascha Wildner  */
108*479ab7f0SSascha Wildner static int
biospnp_init(void)109*479ab7f0SSascha Wildner biospnp_init(void)
110*479ab7f0SSascha Wildner {
111*479ab7f0SSascha Wildner     struct pnp_isaConfiguration	icfg;
112*479ab7f0SSascha Wildner     char			*sigptr;
113*479ab7f0SSascha Wildner     int				result;
114*479ab7f0SSascha Wildner 
115*479ab7f0SSascha Wildner     /* Search for the $PnP signature */
116*479ab7f0SSascha Wildner     pnp_Icheck = NULL;
117*479ab7f0SSascha Wildner     for (sigptr = PTOV(0xf0000); sigptr < PTOV(0xfffff); sigptr += 16)
118*479ab7f0SSascha Wildner 	if (!bcmp(sigptr, "$PnP", 4)) {
119*479ab7f0SSascha Wildner 	    pnp_Icheck = (struct pnp_ICstructure *)sigptr;
120*479ab7f0SSascha Wildner 	    break;
121*479ab7f0SSascha Wildner 	}
122*479ab7f0SSascha Wildner 
123*479ab7f0SSascha Wildner     /* No signature, no BIOS */
124*479ab7f0SSascha Wildner     if (pnp_Icheck == NULL)
125*479ab7f0SSascha Wildner 	return(1);
126*479ab7f0SSascha Wildner 
127*479ab7f0SSascha Wildner     /*
128*479ab7f0SSascha Wildner      * Fetch the system table parameters as a test of the BIOS
129*479ab7f0SSascha Wildner      */
130*479ab7f0SSascha Wildner     result = biospnp_f00(vsegofs(&pnp_NumNodes), vsegofs(&pnp_NodeSize));
131*479ab7f0SSascha Wildner     if (result != PNP_SUCCESS) {
132*479ab7f0SSascha Wildner 	return(1);
133*479ab7f0SSascha Wildner     }
134*479ab7f0SSascha Wildner 
135*479ab7f0SSascha Wildner     /*
136*479ab7f0SSascha Wildner      * Look for the PnP ISA configuration table
137*479ab7f0SSascha Wildner      */
138*479ab7f0SSascha Wildner     result = biospnp_f40(vsegofs(&icfg));
139*479ab7f0SSascha Wildner     switch (result) {
140*479ab7f0SSascha Wildner     case PNP_SUCCESS:
141*479ab7f0SSascha Wildner 	/* If the BIOS found some PnP devices, take its hint for the read port */
142*479ab7f0SSascha Wildner 	if ((icfg.ic_revision == 1) && (icfg.ic_nCSN > 0))
143*479ab7f0SSascha Wildner 	    isapnp_readport = icfg.ic_rdport;
144*479ab7f0SSascha Wildner 	break;
145*479ab7f0SSascha Wildner     case PNP_FUNCTION_NOT_SUPPORTED:
146*479ab7f0SSascha Wildner 	/* The BIOS says there is no ISA bus (should we trust that this works?) */
147*479ab7f0SSascha Wildner 	printf("PnP BIOS claims no ISA bus\n");
148*479ab7f0SSascha Wildner 	isapnp_readport = -1;
149*479ab7f0SSascha Wildner 	break;
150*479ab7f0SSascha Wildner     }
151*479ab7f0SSascha Wildner     return(0);
152*479ab7f0SSascha Wildner }
153*479ab7f0SSascha Wildner 
154*479ab7f0SSascha Wildner static void
biospnp_enumerate(void)155*479ab7f0SSascha Wildner biospnp_enumerate(void)
156*479ab7f0SSascha Wildner {
157*479ab7f0SSascha Wildner     u_int8_t		Node;
158*479ab7f0SSascha Wildner     struct pnp_devNode	*devNodeBuffer;
159*479ab7f0SSascha Wildner     int			result;
160*479ab7f0SSascha Wildner     struct pnpinfo	*pi;
161*479ab7f0SSascha Wildner     int			count;
162*479ab7f0SSascha Wildner 
163*479ab7f0SSascha Wildner     /* Init/check state */
164*479ab7f0SSascha Wildner     if (biospnp_init())
165*479ab7f0SSascha Wildner 	return;
166*479ab7f0SSascha Wildner 
167*479ab7f0SSascha Wildner     devNodeBuffer = (struct pnp_devNode *)malloc(pnp_NodeSize);
168*479ab7f0SSascha Wildner     Node = 0;
169*479ab7f0SSascha Wildner     count = 1000;
170*479ab7f0SSascha Wildner     while((Node != 0xff) && (count-- > 0)) {
171*479ab7f0SSascha Wildner 	result = biospnp_f01(vsegofs(&Node), vsegofs(devNodeBuffer), 0x1);
172*479ab7f0SSascha Wildner 	if (result != PNP_SUCCESS) {
173*479ab7f0SSascha Wildner 	    printf("PnP BIOS node %d: error 0x%x\n", Node, result);
174*479ab7f0SSascha Wildner 	} else {
175*479ab7f0SSascha Wildner 	    pi = pnp_allocinfo();
176*479ab7f0SSascha Wildner 	    pnp_addident(pi, pnp_eisaformat(devNodeBuffer->dn_id));
177*479ab7f0SSascha Wildner 	    biospnp_scanresdata(pi, devNodeBuffer);
178*479ab7f0SSascha Wildner 	    pnp_addinfo(pi);
179*479ab7f0SSascha Wildner 	}
180*479ab7f0SSascha Wildner     }
181*479ab7f0SSascha Wildner }
182*479ab7f0SSascha Wildner 
183*479ab7f0SSascha Wildner /*
184*479ab7f0SSascha Wildner  * Scan the resource data in the node's data area for compatible device IDs
185*479ab7f0SSascha Wildner  * and descriptions.
186*479ab7f0SSascha Wildner  */
187*479ab7f0SSascha Wildner static void
biospnp_scanresdata(struct pnpinfo * pi,struct pnp_devNode * dn)188*479ab7f0SSascha Wildner biospnp_scanresdata(struct pnpinfo *pi, struct pnp_devNode *dn)
189*479ab7f0SSascha Wildner {
190*479ab7f0SSascha Wildner     u_int	tag, i, rlen, dlen;
191*479ab7f0SSascha Wildner     u_int8_t	*p;
192*479ab7f0SSascha Wildner     char	*str;
193*479ab7f0SSascha Wildner 
194*479ab7f0SSascha Wildner     p = dn->dn_data;			/* point to resource data */
195*479ab7f0SSascha Wildner     dlen = dn->dn_size - (p - (u_int8_t *)dn);	/* length of resource data */
196*479ab7f0SSascha Wildner 
197*479ab7f0SSascha Wildner     for (i = 0; i < dlen; i+= rlen) {
198*479ab7f0SSascha Wildner 	tag = p[i];
199*479ab7f0SSascha Wildner 	i++;
200*479ab7f0SSascha Wildner 	if (PNP_RES_TYPE(tag) == 0) {
201*479ab7f0SSascha Wildner 	    rlen = PNP_SRES_LEN(tag);
202*479ab7f0SSascha Wildner 	    /* small resource */
203*479ab7f0SSascha Wildner 	    switch (PNP_SRES_NUM(tag)) {
204*479ab7f0SSascha Wildner 
205*479ab7f0SSascha Wildner 	    case COMP_DEVICE_ID:
206*479ab7f0SSascha Wildner 		/* got a compatible device ID */
207*479ab7f0SSascha Wildner 		pnp_addident(pi, pnp_eisaformat(p + i));
208*479ab7f0SSascha Wildner 		break;
209*479ab7f0SSascha Wildner 
210*479ab7f0SSascha Wildner 	    case END_TAG:
211*479ab7f0SSascha Wildner 		return;
212*479ab7f0SSascha Wildner 	    }
213*479ab7f0SSascha Wildner 	} else {
214*479ab7f0SSascha Wildner 	    /* large resource */
215*479ab7f0SSascha Wildner 	    rlen = *(u_int16_t *)(p + i);
216*479ab7f0SSascha Wildner 	    i += sizeof(u_int16_t);
217*479ab7f0SSascha Wildner 
218*479ab7f0SSascha Wildner 	    switch(PNP_LRES_NUM(tag)) {
219*479ab7f0SSascha Wildner 
220*479ab7f0SSascha Wildner 	    case ID_STRING_ANSI:
221*479ab7f0SSascha Wildner 		str = malloc(rlen + 1);
222*479ab7f0SSascha Wildner 		bcopy(p + i, str, rlen);
223*479ab7f0SSascha Wildner 		str[rlen] = 0;
224*479ab7f0SSascha Wildner 		if (pi->pi_desc == NULL) {
225*479ab7f0SSascha Wildner 		    pi->pi_desc = str;
226*479ab7f0SSascha Wildner 		} else {
227*479ab7f0SSascha Wildner 		    free(str);
228*479ab7f0SSascha Wildner 		}
229*479ab7f0SSascha Wildner 		break;
230*479ab7f0SSascha Wildner 	    }
231*479ab7f0SSascha Wildner 	}
232*479ab7f0SSascha Wildner     }
233*479ab7f0SSascha Wildner }
234*479ab7f0SSascha Wildner 
235*479ab7f0SSascha Wildner 
236*479ab7f0SSascha Wildner /*
237*479ab7f0SSascha Wildner  * Make a 16-bit realmode PnP BIOS call.
238*479ab7f0SSascha Wildner  *
239*479ab7f0SSascha Wildner  * The first argument passed is the function number, the last is the
240*479ab7f0SSascha Wildner  * BIOS data segment selector.  Intermediate arguments may be 16 or
241*479ab7f0SSascha Wildner  * 32 bytes in length, and are described by the format string.
242*479ab7f0SSascha Wildner  *
243*479ab7f0SSascha Wildner  * Arguments to the BIOS functions must be packed on the stack, hence
244*479ab7f0SSascha Wildner  * this evil.
245*479ab7f0SSascha Wildner  */
246*479ab7f0SSascha Wildner static int
biospnp_call(int func,const char * fmt,...)247*479ab7f0SSascha Wildner biospnp_call(int func, const char *fmt, ...)
248*479ab7f0SSascha Wildner {
249*479ab7f0SSascha Wildner     __va_list	ap;
250*479ab7f0SSascha Wildner     const char	*p;
251*479ab7f0SSascha Wildner     u_int8_t	*argp;
252*479ab7f0SSascha Wildner     u_int32_t	args[4];
253*479ab7f0SSascha Wildner     u_int32_t	i;
254*479ab7f0SSascha Wildner 
255*479ab7f0SSascha Wildner     /* function number first */
256*479ab7f0SSascha Wildner     argp = (u_int8_t *)args;
257*479ab7f0SSascha Wildner     *(u_int16_t *)argp = func;
258*479ab7f0SSascha Wildner     argp += sizeof(u_int16_t);
259*479ab7f0SSascha Wildner 
260*479ab7f0SSascha Wildner     /* take args according to format */
261*479ab7f0SSascha Wildner     __va_start(ap, fmt);
262*479ab7f0SSascha Wildner     for (p = fmt; *p != 0; p++) {
263*479ab7f0SSascha Wildner 	switch(*p) {
264*479ab7f0SSascha Wildner 
265*479ab7f0SSascha Wildner 	case 'w':
266*479ab7f0SSascha Wildner 	    i = __va_arg(ap, u_int);
267*479ab7f0SSascha Wildner 	    *(u_int16_t *)argp = i;
268*479ab7f0SSascha Wildner 	    argp += sizeof(u_int16_t);
269*479ab7f0SSascha Wildner 	    break;
270*479ab7f0SSascha Wildner 
271*479ab7f0SSascha Wildner 	case 'l':
272*479ab7f0SSascha Wildner 	    i = __va_arg(ap, u_int32_t);
273*479ab7f0SSascha Wildner 	    *(u_int32_t *)argp = i;
274*479ab7f0SSascha Wildner 	    argp += sizeof(u_int32_t);
275*479ab7f0SSascha Wildner 	    break;
276*479ab7f0SSascha Wildner 	}
277*479ab7f0SSascha Wildner     }
278*479ab7f0SSascha Wildner 
279*479ab7f0SSascha Wildner     /* BIOS segment last */
280*479ab7f0SSascha Wildner     *(u_int16_t *)argp = pnp_Icheck->pnp_rmds;
281*479ab7f0SSascha Wildner     argp += sizeof(u_int16_t);
282*479ab7f0SSascha Wildner 
283*479ab7f0SSascha Wildner     /* prepare for call */
284*479ab7f0SSascha Wildner     v86.ctl = V86_ADDR | V86_CALLF;
285*479ab7f0SSascha Wildner     v86.addr = ((u_int32_t)pnp_Icheck->pnp_rmcs << 16) + pnp_Icheck->pnp_rmip;
286*479ab7f0SSascha Wildner 
287*479ab7f0SSascha Wildner     /* call with packed stack and return */
288*479ab7f0SSascha Wildner     v86bios(args[0], args[1], args[2], args[3]);
289*479ab7f0SSascha Wildner     return(v86.eax & 0xffff);
290*479ab7f0SSascha Wildner }
291