2 * SMC 37C93X initialization code
5 #include <linux/kernel.h>
8 #include <linux/init.h>
9 #include <linux/delay.h>
11 #include <asm/hwrpb.h>
13 #include <asm/segment.h>
18 # define DBG_DEVS(args) printk args
20 # define DBG_DEVS(args)
27 /* device "activate" register contents */
31 /* configuration on/off keys */
32 #define CONFIG_ON_KEY 0x55
33 #define CONFIG_OFF_KEY 0xaa
35 /* configuration space device definitions */
46 /* Chip register offsets from base */
47 #define CONFIG_CONTROL 0x02
48 #define INDEX_ADDRESS 0x03
49 #define LOGICAL_DEVICE_NUMBER 0x07
50 #define DEVICE_ID 0x20
51 #define DEVICE_REV 0x21
52 #define POWER_CONTROL 0x22
53 #define POWER_MGMT 0x23
59 #define INTERRUPT_SEL 0x70
60 #define INTERRUPT_SEL_2 0x72 /* KYBD/MOUS only */
61 #define DMA_CHANNEL_SEL 0x74 /* FDC/PARP only */
63 #define FDD_MODE_REGISTER 0x90
64 #define FDD_OPTION_REGISTER 0x91
66 /* values that we read back that are expected ... */
67 #define VALID_DEVICE_ID 2
69 /* default device addresses */
70 #define KYBD_INTERRUPT 1
71 #define MOUS_INTERRUPT 12
72 #define COM2_BASE 0x2f8
73 #define COM2_INTERRUPT 3
74 #define COM1_BASE 0x3f8
75 #define COM1_INTERRUPT 4
76 #define PARP_BASE 0x3bc
77 #define PARP_INTERRUPT 7
79 static unsigned long __init
SMCConfigState(unsigned long baseAddr
)
84 unsigned long configPort
;
85 unsigned long indexPort
;
86 unsigned long dataPort
;
90 configPort
= indexPort
= baseAddr
;
91 dataPort
= configPort
+ 1;
95 for (i
= 0; i
< NUM_RETRIES
; i
++)
97 outb(CONFIG_ON_KEY
, configPort
);
98 outb(CONFIG_ON_KEY
, configPort
);
99 outb(DEVICE_ID
, indexPort
);
100 devId
= inb(dataPort
);
101 if (devId
== VALID_DEVICE_ID
) {
102 outb(DEVICE_REV
, indexPort
);
103 devRev
= inb(dataPort
);
109 return (i
!= NUM_RETRIES
) ? baseAddr
: 0L;
112 static void __init
SMCRunState(unsigned long baseAddr
)
114 outb(CONFIG_OFF_KEY
, baseAddr
);
117 static unsigned long __init
SMCDetectUltraIO(void)
119 unsigned long baseAddr
;
122 if ( ( baseAddr
= SMCConfigState( baseAddr
) ) == 0x3F0 ) {
126 if ( ( baseAddr
= SMCConfigState( baseAddr
) ) == 0x370 ) {
129 return( ( unsigned long )0 );
132 static void __init
SMCEnableDevice(unsigned long baseAddr
,
133 unsigned long device
,
134 unsigned long portaddr
,
135 unsigned long interrupt
)
137 unsigned long indexPort
;
138 unsigned long dataPort
;
140 indexPort
= baseAddr
;
141 dataPort
= baseAddr
+ 1;
143 outb(LOGICAL_DEVICE_NUMBER
, indexPort
);
144 outb(device
, dataPort
);
146 outb(ADDR_LO
, indexPort
);
147 outb(( portaddr
& 0xFF ), dataPort
);
149 outb(ADDR_HI
, indexPort
);
150 outb((portaddr
>> 8) & 0xFF, dataPort
);
152 outb(INTERRUPT_SEL
, indexPort
);
153 outb(interrupt
, dataPort
);
155 outb(ACTIVATE
, indexPort
);
156 outb(DEVICE_ON
, dataPort
);
159 static void __init
SMCEnableKYBD(unsigned long baseAddr
)
161 unsigned long indexPort
;
162 unsigned long dataPort
;
164 indexPort
= baseAddr
;
165 dataPort
= baseAddr
+ 1;
167 outb(LOGICAL_DEVICE_NUMBER
, indexPort
);
168 outb(KYBD
, dataPort
);
170 outb(INTERRUPT_SEL
, indexPort
); /* Primary interrupt select */
171 outb(KYBD_INTERRUPT
, dataPort
);
173 outb(INTERRUPT_SEL_2
, indexPort
); /* Secondary interrupt select */
174 outb(MOUS_INTERRUPT
, dataPort
);
176 outb(ACTIVATE
, indexPort
);
177 outb(DEVICE_ON
, dataPort
);
180 static void __init
SMCEnableFDC(unsigned long baseAddr
)
182 unsigned long indexPort
;
183 unsigned long dataPort
;
185 unsigned char oldValue
;
187 indexPort
= baseAddr
;
188 dataPort
= baseAddr
+ 1;
190 outb(LOGICAL_DEVICE_NUMBER
, indexPort
);
193 outb(FDD_MODE_REGISTER
, indexPort
);
194 oldValue
= inb(dataPort
);
196 oldValue
|= 0x0E; /* Enable burst mode */
197 outb(oldValue
, dataPort
);
199 outb(INTERRUPT_SEL
, indexPort
); /* Primary interrupt select */
200 outb(0x06, dataPort
);
202 outb(DMA_CHANNEL_SEL
, indexPort
); /* DMA channel select */
203 outb(0x02, dataPort
);
205 outb(ACTIVATE
, indexPort
);
206 outb(DEVICE_ON
, dataPort
);
210 static void __init
SMCReportDeviceStatus(unsigned long baseAddr
)
212 unsigned long indexPort
;
213 unsigned long dataPort
;
214 unsigned char currentControl
;
216 indexPort
= baseAddr
;
217 dataPort
= baseAddr
+ 1;
219 outb(POWER_CONTROL
, indexPort
);
220 currentControl
= inb(dataPort
);
222 printk(currentControl
& (1 << FDC
)
223 ? "\t+FDC Enabled\n" : "\t-FDC Disabled\n");
224 printk(currentControl
& (1 << IDE1
)
225 ? "\t+IDE1 Enabled\n" : "\t-IDE1 Disabled\n");
226 printk(currentControl
& (1 << IDE2
)
227 ? "\t+IDE2 Enabled\n" : "\t-IDE2 Disabled\n");
228 printk(currentControl
& (1 << PARP
)
229 ? "\t+PARP Enabled\n" : "\t-PARP Disabled\n");
230 printk(currentControl
& (1 << SER1
)
231 ? "\t+SER1 Enabled\n" : "\t-SER1 Disabled\n");
232 printk(currentControl
& (1 << SER2
)
233 ? "\t+SER2 Enabled\n" : "\t-SER2 Disabled\n");
239 int __init
SMC93x_Init(void)
241 unsigned long SMCUltraBase
;
244 local_irq_save(flags
);
245 if ((SMCUltraBase
= SMCDetectUltraIO()) != 0UL) {
247 SMCReportDeviceStatus(SMCUltraBase
);
249 SMCEnableDevice(SMCUltraBase
, SER1
, COM1_BASE
, COM1_INTERRUPT
);
250 DBG_DEVS(("SMC FDC37C93X: SER1 done\n"));
251 SMCEnableDevice(SMCUltraBase
, SER2
, COM2_BASE
, COM2_INTERRUPT
);
252 DBG_DEVS(("SMC FDC37C93X: SER2 done\n"));
253 SMCEnableDevice(SMCUltraBase
, PARP
, PARP_BASE
, PARP_INTERRUPT
);
254 DBG_DEVS(("SMC FDC37C93X: PARP done\n"));
255 /* On PC164, IDE on the SMC is not enabled;
256 CMD646 (PCI) on MB */
257 SMCEnableKYBD(SMCUltraBase
);
258 DBG_DEVS(("SMC FDC37C93X: KYB done\n"));
259 SMCEnableFDC(SMCUltraBase
);
260 DBG_DEVS(("SMC FDC37C93X: FDC done\n"));
262 SMCReportDeviceStatus(SMCUltraBase
);
264 SMCRunState(SMCUltraBase
);
265 local_irq_restore(flags
);
266 printk("SMC FDC37C93X Ultra I/O Controller found @ 0x%lx\n",
271 local_irq_restore(flags
);
272 DBG_DEVS(("No SMC FDC37C93X Ultra I/O Controller found\n"));