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