1 /* $Id: teleint.c,v 1.16.2.5 2004/01/19 15:31:50 keil Exp $
3 * low level stuff for TeleInt isdn cards
6 * Copyright by Karsten Keil <keil@isdn4linux.de>
8 * This software may be used and distributed according to the terms
9 * of the GNU General Public License, incorporated herein by reference.
13 #include <linux/init.h>
19 extern const char *CardType
[];
21 static const char *TeleInt_revision
= "$Revision: 1.16.2.5 $";
23 #define byteout(addr,val) outb(val,addr)
24 #define bytein(addr) inb(addr)
27 readreg(unsigned int ale
, unsigned int adr
, u_char off
)
33 ret
= HFC_BUSY
& bytein(ale
);
34 while (ret
&& --max_delay
)
35 ret
= HFC_BUSY
& bytein(ale
);
37 printk(KERN_WARNING
"TeleInt Busy not inactive\n");
45 readfifo(unsigned int ale
, unsigned int adr
, u_char off
, u_char
* data
, int size
)
48 register int max_delay
= 20000;
52 for (i
= 0; i
<size
; i
++) {
53 ret
= HFC_BUSY
& bytein(ale
);
54 while (ret
&& --max_delay
)
55 ret
= HFC_BUSY
& bytein(ale
);
57 printk(KERN_WARNING
"TeleInt Busy not inactive\n");
60 data
[i
] = bytein(adr
);
66 writereg(unsigned int ale
, unsigned int adr
, u_char off
, u_char data
)
72 ret
= HFC_BUSY
& bytein(ale
);
73 while (ret
&& --max_delay
)
74 ret
= HFC_BUSY
& bytein(ale
);
76 printk(KERN_WARNING
"TeleInt Busy not inactive\n");
83 writefifo(unsigned int ale
, unsigned int adr
, u_char off
, u_char
* data
, int size
)
86 register int max_delay
= 20000;
90 for (i
= 0; i
<size
; i
++) {
91 ret
= HFC_BUSY
& bytein(ale
);
92 while (ret
&& --max_delay
)
93 ret
= HFC_BUSY
& bytein(ale
);
95 printk(KERN_WARNING
"TeleInt Busy not inactive\n");
98 byteout(adr
, data
[i
]);
102 /* Interface functions */
105 ReadISAC(struct IsdnCardState
*cs
, u_char offset
)
107 cs
->hw
.hfc
.cip
= offset
;
108 return (readreg(cs
->hw
.hfc
.addr
| 1, cs
->hw
.hfc
.addr
, offset
));
112 WriteISAC(struct IsdnCardState
*cs
, u_char offset
, u_char value
)
114 cs
->hw
.hfc
.cip
= offset
;
115 writereg(cs
->hw
.hfc
.addr
| 1, cs
->hw
.hfc
.addr
, offset
, value
);
119 ReadISACfifo(struct IsdnCardState
*cs
, u_char
* data
, int size
)
122 readfifo(cs
->hw
.hfc
.addr
| 1, cs
->hw
.hfc
.addr
, 0, data
, size
);
126 WriteISACfifo(struct IsdnCardState
*cs
, u_char
* data
, int size
)
129 writefifo(cs
->hw
.hfc
.addr
| 1, cs
->hw
.hfc
.addr
, 0, data
, size
);
133 ReadHFC(struct IsdnCardState
*cs
, int data
, u_char reg
)
138 cs
->hw
.hfc
.cip
= reg
;
139 byteout(cs
->hw
.hfc
.addr
| 1, reg
);
140 ret
= bytein(cs
->hw
.hfc
.addr
);
141 if (cs
->debug
& L1_DEB_HSCX_FIFO
&& (data
!= 2))
142 debugl1(cs
, "hfc RD %02x %02x", reg
, ret
);
144 ret
= bytein(cs
->hw
.hfc
.addr
| 1);
149 WriteHFC(struct IsdnCardState
*cs
, int data
, u_char reg
, u_char value
)
151 byteout(cs
->hw
.hfc
.addr
| 1, reg
);
152 cs
->hw
.hfc
.cip
= reg
;
154 byteout(cs
->hw
.hfc
.addr
, value
);
155 if (cs
->debug
& L1_DEB_HSCX_FIFO
&& (data
!= 2))
156 debugl1(cs
, "hfc W%c %02x %02x", data
? 'D' : 'C', reg
, value
);
160 TeleInt_interrupt(int intno
, void *dev_id
, struct pt_regs
*regs
)
162 struct IsdnCardState
*cs
= dev_id
;
166 spin_lock_irqsave(&cs
->lock
, flags
);
167 val
= readreg(cs
->hw
.hfc
.addr
| 1, cs
->hw
.hfc
.addr
, ISAC_ISTA
);
170 isac_interrupt(cs
, val
);
171 val
= readreg(cs
->hw
.hfc
.addr
| 1, cs
->hw
.hfc
.addr
, ISAC_ISTA
);
173 if (cs
->debug
& L1_DEB_ISAC
)
174 debugl1(cs
, "ISAC IntStat after IntRoutine");
177 writereg(cs
->hw
.hfc
.addr
| 1, cs
->hw
.hfc
.addr
, ISAC_MASK
, 0xFF);
178 writereg(cs
->hw
.hfc
.addr
| 1, cs
->hw
.hfc
.addr
, ISAC_MASK
, 0x0);
179 spin_unlock_irqrestore(&cs
->lock
, flags
);
184 TeleInt_Timer(struct IsdnCardState
*cs
)
189 spin_lock_irqsave(&cs
->lock
, flags
);
190 if (cs
->bcs
[0].mode
) {
192 main_irq_hfc(&cs
->bcs
[0]);
194 if (cs
->bcs
[1].mode
) {
196 main_irq_hfc(&cs
->bcs
[1]);
198 spin_unlock_irqrestore(&cs
->lock
, flags
);
202 cs
->hw
.hfc
.timer
.expires
= jiffies
+ stat
;
203 add_timer(&cs
->hw
.hfc
.timer
);
207 release_io_TeleInt(struct IsdnCardState
*cs
)
209 del_timer(&cs
->hw
.hfc
.timer
);
212 release_region(cs
->hw
.hfc
.addr
, 2);
216 reset_TeleInt(struct IsdnCardState
*cs
)
218 printk(KERN_INFO
"TeleInt: resetting card\n");
219 cs
->hw
.hfc
.cirm
|= HFC_RESET
;
220 byteout(cs
->hw
.hfc
.addr
| 1, cs
->hw
.hfc
.cirm
); /* Reset On */
222 cs
->hw
.hfc
.cirm
&= ~HFC_RESET
;
223 byteout(cs
->hw
.hfc
.addr
| 1, cs
->hw
.hfc
.cirm
); /* Reset Off */
228 TeleInt_card_msg(struct IsdnCardState
*cs
, int mt
, void *arg
)
235 spin_lock_irqsave(&cs
->lock
, flags
);
237 spin_unlock_irqrestore(&cs
->lock
, flags
);
240 release_io_TeleInt(cs
);
243 spin_lock_irqsave(&cs
->lock
, flags
);
246 clear_pending_isac_ints(cs
);
248 /* Reenable all IRQ */
249 cs
->writeisac(cs
, ISAC_MASK
, 0);
250 cs
->writeisac(cs
, ISAC_CMDR
, 0x41);
251 spin_unlock_irqrestore(&cs
->lock
, flags
);
255 cs
->hw
.hfc
.timer
.expires
= jiffies
+ delay
;
256 add_timer(&cs
->hw
.hfc
.timer
);
265 setup_TeleInt(struct IsdnCard
*card
)
267 struct IsdnCardState
*cs
= card
->cs
;
270 strcpy(tmp
, TeleInt_revision
);
271 printk(KERN_INFO
"HiSax: TeleInt driver Rev. %s\n", HiSax_getrev(tmp
));
272 if (cs
->typ
!= ISDN_CTYPE_TELEINT
)
275 cs
->hw
.hfc
.addr
= card
->para
[1] & 0x3fe;
276 cs
->irq
= card
->para
[0];
277 cs
->hw
.hfc
.cirm
= HFC_CIRM
;
278 cs
->hw
.hfc
.isac_spcr
= 0x00;
280 cs
->hw
.hfc
.ctmt
= HFC_CTMT
| HFC_CLTIMER
;
281 cs
->bcs
[0].hw
.hfc
.send
= NULL
;
282 cs
->bcs
[1].hw
.hfc
.send
= NULL
;
283 cs
->hw
.hfc
.fifosize
= 7 * 1024 + 512;
284 cs
->hw
.hfc
.timer
.function
= (void *) TeleInt_Timer
;
285 cs
->hw
.hfc
.timer
.data
= (long) cs
;
286 init_timer(&cs
->hw
.hfc
.timer
);
287 if (!request_region(cs
->hw
.hfc
.addr
, 2, "TeleInt isdn")) {
289 "HiSax: %s config port %x-%x already in use\n",
292 cs
->hw
.hfc
.addr
+ 2);
296 byteout(cs
->hw
.hfc
.addr
, cs
->hw
.hfc
.addr
& 0xff);
297 byteout(cs
->hw
.hfc
.addr
| 1, ((cs
->hw
.hfc
.addr
& 0x300) >> 8) | 0x54);
300 cs
->hw
.hfc
.cirm
|= HFC_INTA
;
303 cs
->hw
.hfc
.cirm
|= HFC_INTB
;
306 cs
->hw
.hfc
.cirm
|= HFC_INTC
;
309 cs
->hw
.hfc
.cirm
|= HFC_INTD
;
312 cs
->hw
.hfc
.cirm
|= HFC_INTE
;
315 cs
->hw
.hfc
.cirm
|= HFC_INTF
;
318 printk(KERN_WARNING
"TeleInt: wrong IRQ\n");
319 release_io_TeleInt(cs
);
322 byteout(cs
->hw
.hfc
.addr
| 1, cs
->hw
.hfc
.cirm
);
323 byteout(cs
->hw
.hfc
.addr
| 1, cs
->hw
.hfc
.ctmt
);
325 printk(KERN_INFO
"TeleInt: defined at 0x%x IRQ %d\n",
326 cs
->hw
.hfc
.addr
, cs
->irq
);
329 cs
->readisac
= &ReadISAC
;
330 cs
->writeisac
= &WriteISAC
;
331 cs
->readisacfifo
= &ReadISACfifo
;
332 cs
->writeisacfifo
= &WriteISACfifo
;
333 cs
->BC_Read_Reg
= &ReadHFC
;
334 cs
->BC_Write_Reg
= &WriteHFC
;
335 cs
->cardmsg
= &TeleInt_card_msg
;
336 cs
->irq_func
= &TeleInt_interrupt
;
337 ISACVersion(cs
, "TeleInt:");