xref: /minix3/sys/arch/i386/stand/lib/isapnp.c (revision 58a2b0008e28f606a7f7f5faaeaba4faac57a1ea)
1*58a2b000SEvgeniy Ivanov /*	$NetBSD: isapnp.c,v 1.5 2008/12/14 17:03:43 christos Exp $	 */
2*58a2b000SEvgeniy Ivanov 
3*58a2b000SEvgeniy Ivanov /*
4*58a2b000SEvgeniy Ivanov  * Copyright (c) 1997
5*58a2b000SEvgeniy Ivanov  *	Matthias Drochner.  All rights reserved.
6*58a2b000SEvgeniy Ivanov  *
7*58a2b000SEvgeniy Ivanov  * Redistribution and use in source and binary forms, with or without
8*58a2b000SEvgeniy Ivanov  * modification, are permitted provided that the following conditions
9*58a2b000SEvgeniy Ivanov  * are met:
10*58a2b000SEvgeniy Ivanov  * 1. Redistributions of source code must retain the above copyright
11*58a2b000SEvgeniy Ivanov  *    notice, this list of conditions and the following disclaimer.
12*58a2b000SEvgeniy Ivanov  * 2. Redistributions in binary form must reproduce the above copyright
13*58a2b000SEvgeniy Ivanov  *    notice, this list of conditions and the following disclaimer in the
14*58a2b000SEvgeniy Ivanov  *    documentation and/or other materials provided with the distribution.
15*58a2b000SEvgeniy Ivanov  *
16*58a2b000SEvgeniy Ivanov  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17*58a2b000SEvgeniy Ivanov  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18*58a2b000SEvgeniy Ivanov  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19*58a2b000SEvgeniy Ivanov  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20*58a2b000SEvgeniy Ivanov  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21*58a2b000SEvgeniy Ivanov  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22*58a2b000SEvgeniy Ivanov  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23*58a2b000SEvgeniy Ivanov  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24*58a2b000SEvgeniy Ivanov  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25*58a2b000SEvgeniy Ivanov  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26*58a2b000SEvgeniy Ivanov  *
27*58a2b000SEvgeniy Ivanov  */
28*58a2b000SEvgeniy Ivanov 
29*58a2b000SEvgeniy Ivanov /*
30*58a2b000SEvgeniy Ivanov  * minimal ISA PnP implementation: find adapter, return settings (1 IO and 1
31*58a2b000SEvgeniy Ivanov  * DMA only for now)
32*58a2b000SEvgeniy Ivanov  */
33*58a2b000SEvgeniy Ivanov 
34*58a2b000SEvgeniy Ivanov #include <sys/types.h>
35*58a2b000SEvgeniy Ivanov #include <machine/pio.h>
36*58a2b000SEvgeniy Ivanov 
37*58a2b000SEvgeniy Ivanov #include <lib/libsa/stand.h>
38*58a2b000SEvgeniy Ivanov 
39*58a2b000SEvgeniy Ivanov #include <libi386.h>
40*58a2b000SEvgeniy Ivanov 
41*58a2b000SEvgeniy Ivanov #include "isapnpvar.h"
42*58a2b000SEvgeniy Ivanov 
43*58a2b000SEvgeniy Ivanov #define PNPADDR 0x0279
44*58a2b000SEvgeniy Ivanov #define PNPWDATA 0x0a79
45*58a2b000SEvgeniy Ivanov #define PNPRDATAMIN 0x0203
46*58a2b000SEvgeniy Ivanov #define PNPRDATAMAX 0x03ff
47*58a2b000SEvgeniy Ivanov 
48*58a2b000SEvgeniy Ivanov enum {
49*58a2b000SEvgeniy Ivanov 	DATAPORT,
50*58a2b000SEvgeniy Ivanov 	ISOL,
51*58a2b000SEvgeniy Ivanov 	CONTROL,
52*58a2b000SEvgeniy Ivanov 	WAKE,
53*58a2b000SEvgeniy Ivanov 	RESDATA,
54*58a2b000SEvgeniy Ivanov 	RESSTAT,
55*58a2b000SEvgeniy Ivanov 	SETCSN,
56*58a2b000SEvgeniy Ivanov 	SETLDEV
57*58a2b000SEvgeniy Ivanov };
58*58a2b000SEvgeniy Ivanov 
59*58a2b000SEvgeniy Ivanov #define MEMBASE 0x40
60*58a2b000SEvgeniy Ivanov #define IOBASE 0x60
61*58a2b000SEvgeniy Ivanov #define INTBASE 0x70
62*58a2b000SEvgeniy Ivanov #define DMABASE 0x74
63*58a2b000SEvgeniy Ivanov 
64*58a2b000SEvgeniy Ivanov static int      pnpdataport;
65*58a2b000SEvgeniy Ivanov 
66*58a2b000SEvgeniy Ivanov static int
getiobase(int nr)67*58a2b000SEvgeniy Ivanov getiobase(int nr)
68*58a2b000SEvgeniy Ivanov {
69*58a2b000SEvgeniy Ivanov 	unsigned short  iobase;
70*58a2b000SEvgeniy Ivanov 
71*58a2b000SEvgeniy Ivanov 	outb(PNPADDR, SETLDEV);
72*58a2b000SEvgeniy Ivanov 	outb(PNPWDATA, 0);	/* subdev 0 */
73*58a2b000SEvgeniy Ivanov 
74*58a2b000SEvgeniy Ivanov 	outb(PNPADDR, IOBASE + nr * 2);
75*58a2b000SEvgeniy Ivanov 	iobase = (inb(pnpdataport) << 8);
76*58a2b000SEvgeniy Ivanov 	outb(PNPADDR, IOBASE + nr * 2 + 1);
77*58a2b000SEvgeniy Ivanov 	iobase |= inb(pnpdataport);
78*58a2b000SEvgeniy Ivanov 
79*58a2b000SEvgeniy Ivanov 	return iobase;
80*58a2b000SEvgeniy Ivanov }
81*58a2b000SEvgeniy Ivanov 
82*58a2b000SEvgeniy Ivanov static int
getdmachan(int nr)83*58a2b000SEvgeniy Ivanov getdmachan(int nr)
84*58a2b000SEvgeniy Ivanov {
85*58a2b000SEvgeniy Ivanov 	unsigned char   dmachannel;
86*58a2b000SEvgeniy Ivanov 
87*58a2b000SEvgeniy Ivanov 	outb(PNPADDR, SETLDEV);
88*58a2b000SEvgeniy Ivanov 	outb(PNPWDATA, 0);	/* subdev 0 */
89*58a2b000SEvgeniy Ivanov 
90*58a2b000SEvgeniy Ivanov 	outb(PNPADDR, DMABASE + nr);
91*58a2b000SEvgeniy Ivanov 	dmachannel = inb(pnpdataport) & 0x07;
92*58a2b000SEvgeniy Ivanov 
93*58a2b000SEvgeniy Ivanov 	return dmachannel;
94*58a2b000SEvgeniy Ivanov }
95*58a2b000SEvgeniy Ivanov 
96*58a2b000SEvgeniy Ivanov struct cardid {
97*58a2b000SEvgeniy Ivanov 	unsigned char   eisaid[4];
98*58a2b000SEvgeniy Ivanov 	unsigned int    serial;
99*58a2b000SEvgeniy Ivanov 	unsigned char   crc;
100*58a2b000SEvgeniy Ivanov };
101*58a2b000SEvgeniy Ivanov 
102*58a2b000SEvgeniy Ivanov /*
103*58a2b000SEvgeniy Ivanov  do isolation, call pnpscanresc() in board config state
104*58a2b000SEvgeniy Ivanov  */
105*58a2b000SEvgeniy Ivanov static int
pnpisol(int csn)106*58a2b000SEvgeniy Ivanov pnpisol(int csn)
107*58a2b000SEvgeniy Ivanov {
108*58a2b000SEvgeniy Ivanov 	unsigned char   buf[9];
109*58a2b000SEvgeniy Ivanov 	int             i, j;
110*58a2b000SEvgeniy Ivanov 	struct cardid  *id;
111*58a2b000SEvgeniy Ivanov 	unsigned char   crc = 0x6a;
112*58a2b000SEvgeniy Ivanov 
113*58a2b000SEvgeniy Ivanov 	/*
114*58a2b000SEvgeniy Ivanov 	 * do 72 pairs of reads from ISOL register all but 1 go to sleep
115*58a2b000SEvgeniy Ivanov 	 * state (ch. 3.3)
116*58a2b000SEvgeniy Ivanov 	 */
117*58a2b000SEvgeniy Ivanov 	outb(PNPADDR, ISOL);
118*58a2b000SEvgeniy Ivanov 	delay(1000);
119*58a2b000SEvgeniy Ivanov 
120*58a2b000SEvgeniy Ivanov 	for (i = 0; i < 9; i++) {
121*58a2b000SEvgeniy Ivanov 		for (j = 0; j < 8; j++) {
122*58a2b000SEvgeniy Ivanov 			unsigned char   a, b;
123*58a2b000SEvgeniy Ivanov 			int             bitset;
124*58a2b000SEvgeniy Ivanov 
125*58a2b000SEvgeniy Ivanov 			a = inb(pnpdataport);
126*58a2b000SEvgeniy Ivanov 			b = inb(pnpdataport);
127*58a2b000SEvgeniy Ivanov 			if ((a == 0x55) && (b == 0xaa))
128*58a2b000SEvgeniy Ivanov 				bitset = 1;
129*58a2b000SEvgeniy Ivanov 			else if ((a == 0xff) && (b == 0xff))
130*58a2b000SEvgeniy Ivanov 				bitset = 0;
131*58a2b000SEvgeniy Ivanov 			else
132*58a2b000SEvgeniy Ivanov 				return -1;	/* data port conflict */
133*58a2b000SEvgeniy Ivanov 
134*58a2b000SEvgeniy Ivanov 			buf[i] = (buf[i] >> 1) | (bitset << 7);
135*58a2b000SEvgeniy Ivanov 
136*58a2b000SEvgeniy Ivanov 			if (i < 8)	/* calc crc for first 8 bytes (app.
137*58a2b000SEvgeniy Ivanov 					 * B.2) */
138*58a2b000SEvgeniy Ivanov 				crc = (crc >> 1) |
139*58a2b000SEvgeniy Ivanov 				  ((bitset != ((crc & 1) == !(crc & 2))) << 7);
140*58a2b000SEvgeniy Ivanov 
141*58a2b000SEvgeniy Ivanov 			delay(250);
142*58a2b000SEvgeniy Ivanov 		}
143*58a2b000SEvgeniy Ivanov 	}
144*58a2b000SEvgeniy Ivanov 	id = (struct cardid *) buf;
145*58a2b000SEvgeniy Ivanov 
146*58a2b000SEvgeniy Ivanov 	if (id->crc != crc)
147*58a2b000SEvgeniy Ivanov 		return 0;	/* normal end */
148*58a2b000SEvgeniy Ivanov 
149*58a2b000SEvgeniy Ivanov 	outb(PNPADDR, SETCSN);
150*58a2b000SEvgeniy Ivanov 	outb(PNPWDATA, csn);	/* set csn for winning card and put it to
151*58a2b000SEvgeniy Ivanov 				 * config state */
152*58a2b000SEvgeniy Ivanov 
153*58a2b000SEvgeniy Ivanov 	return (id->eisaid[0] << 24) | (id->eisaid[1] << 16)
154*58a2b000SEvgeniy Ivanov 		| (id->eisaid[2] << 8) | (id->eisaid[3]);
155*58a2b000SEvgeniy Ivanov }
156*58a2b000SEvgeniy Ivanov 
157*58a2b000SEvgeniy Ivanov static void
pnpisolreset(void)158*58a2b000SEvgeniy Ivanov pnpisolreset(void)
159*58a2b000SEvgeniy Ivanov {
160*58a2b000SEvgeniy Ivanov 	outb(PNPADDR, WAKE);
161*58a2b000SEvgeniy Ivanov 	outb(PNPWDATA, 0);	/* put all remaining cards to isolation state */
162*58a2b000SEvgeniy Ivanov }
163*58a2b000SEvgeniy Ivanov 
164*58a2b000SEvgeniy Ivanov /*
165*58a2b000SEvgeniy Ivanov  send initiation sequence (app. B.1)
166*58a2b000SEvgeniy Ivanov  */
167*58a2b000SEvgeniy Ivanov static void
pnpinit(void)168*58a2b000SEvgeniy Ivanov pnpinit(void)
169*58a2b000SEvgeniy Ivanov {
170*58a2b000SEvgeniy Ivanov 	int             i;
171*58a2b000SEvgeniy Ivanov 	unsigned char   key = 0x6a;
172*58a2b000SEvgeniy Ivanov 
173*58a2b000SEvgeniy Ivanov 	outb(PNPADDR, 0);
174*58a2b000SEvgeniy Ivanov 	outb(PNPADDR, 0);
175*58a2b000SEvgeniy Ivanov 
176*58a2b000SEvgeniy Ivanov 	for (i = 0; i < 32; i++) {
177*58a2b000SEvgeniy Ivanov 		outb(PNPADDR, key);
178*58a2b000SEvgeniy Ivanov 		key = (key >> 1) |
179*58a2b000SEvgeniy Ivanov 			(((key & 1) == !(key & 2)) << 7);
180*58a2b000SEvgeniy Ivanov 	}
181*58a2b000SEvgeniy Ivanov }
182*58a2b000SEvgeniy Ivanov 
183*58a2b000SEvgeniy Ivanov int
isapnp_finddev(int id,int * iobase,int * dmachan)184*58a2b000SEvgeniy Ivanov isapnp_finddev(int id, int *iobase, int *dmachan)
185*58a2b000SEvgeniy Ivanov {
186*58a2b000SEvgeniy Ivanov 	int             csn;
187*58a2b000SEvgeniy Ivanov 
188*58a2b000SEvgeniy Ivanov 	outb(PNPADDR, CONTROL);
189*58a2b000SEvgeniy Ivanov 	outb(PNPWDATA, 2);	/* XXX force wait for key */
190*58a2b000SEvgeniy Ivanov 
191*58a2b000SEvgeniy Ivanov 	/* scan all allowed data ports (ch. 3.1) */
192*58a2b000SEvgeniy Ivanov 	for (pnpdataport = PNPRDATAMIN; pnpdataport <= PNPRDATAMAX;
193*58a2b000SEvgeniy Ivanov 	     pnpdataport += 4) {
194*58a2b000SEvgeniy Ivanov 		int             res, found = 0;
195*58a2b000SEvgeniy Ivanov 
196*58a2b000SEvgeniy Ivanov 		pnpinit();	/* initiation sequence */
197*58a2b000SEvgeniy Ivanov 
198*58a2b000SEvgeniy Ivanov 		outb(PNPADDR, CONTROL);
199*58a2b000SEvgeniy Ivanov 		outb(PNPWDATA, 4);	/* CSN=0 - only these respond to
200*58a2b000SEvgeniy Ivanov 					 * WAKE[0] */
201*58a2b000SEvgeniy Ivanov 
202*58a2b000SEvgeniy Ivanov 		outb(PNPADDR, WAKE);
203*58a2b000SEvgeniy Ivanov 		outb(PNPWDATA, 0);	/* put into isolation state */
204*58a2b000SEvgeniy Ivanov 
205*58a2b000SEvgeniy Ivanov 		outb(PNPADDR, DATAPORT);
206*58a2b000SEvgeniy Ivanov 		outb(PNPWDATA, pnpdataport >> 2);	/* set READ_DATA port */
207*58a2b000SEvgeniy Ivanov 
208*58a2b000SEvgeniy Ivanov 		csn = 0;
209*58a2b000SEvgeniy Ivanov 		do {
210*58a2b000SEvgeniy Ivanov 			res = pnpisol(++csn);
211*58a2b000SEvgeniy Ivanov 
212*58a2b000SEvgeniy Ivanov 			if ((res) == id) {
213*58a2b000SEvgeniy Ivanov 				if (iobase)
214*58a2b000SEvgeniy Ivanov 					*iobase = getiobase(0);
215*58a2b000SEvgeniy Ivanov 				if (dmachan)
216*58a2b000SEvgeniy Ivanov 					*dmachan = getdmachan(0);
217*58a2b000SEvgeniy Ivanov 				found = 1;
218*58a2b000SEvgeniy Ivanov 			}
219*58a2b000SEvgeniy Ivanov 			pnpisolreset();
220*58a2b000SEvgeniy Ivanov 		} while ((res != 0) && (res != -1));
221*58a2b000SEvgeniy Ivanov 
222*58a2b000SEvgeniy Ivanov 		outb(PNPADDR, CONTROL);
223*58a2b000SEvgeniy Ivanov 		outb(PNPWDATA, 2);	/* return to wait for key */
224*58a2b000SEvgeniy Ivanov 
225*58a2b000SEvgeniy Ivanov 		if (csn > 1)	/* at least 1 board found */
226*58a2b000SEvgeniy Ivanov 			return !found;
227*58a2b000SEvgeniy Ivanov 
228*58a2b000SEvgeniy Ivanov 		/* if no board found, try next dataport */
229*58a2b000SEvgeniy Ivanov 	}
230*58a2b000SEvgeniy Ivanov 	return -1;		/* nothing found */
231*58a2b000SEvgeniy Ivanov }
232