xref: /openbmc/linux/drivers/isdn/hardware/mISDN/mISDNipac.c (revision 4f2c0a4acffbec01079c28f839422e64ddeff004)
182c29810SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
2cae86d4aSKarsten Keil /*
3cae86d4aSKarsten Keil  * isac.c   ISAC specific routines
4cae86d4aSKarsten Keil  *
5cae86d4aSKarsten Keil  * Author       Karsten Keil <keil@isdn4linux.de>
6cae86d4aSKarsten Keil  *
7cae86d4aSKarsten Keil  * Copyright 2009  by Karsten Keil <keil@isdn4linux.de>
8cae86d4aSKarsten Keil  */
9cae86d4aSKarsten Keil 
10a6b7a407SAlexey Dobriyan #include <linux/irqreturn.h>
115a0e3ad6STejun Heo #include <linux/slab.h>
12cae86d4aSKarsten Keil #include <linux/module.h>
13cae86d4aSKarsten Keil #include <linux/mISDNhw.h>
14cae86d4aSKarsten Keil #include "ipac.h"
15cae86d4aSKarsten Keil 
16cae86d4aSKarsten Keil 
17cae86d4aSKarsten Keil #define DBUSY_TIMER_VALUE	80
18cae86d4aSKarsten Keil #define ARCOFI_USE		1
19cae86d4aSKarsten Keil 
20cae86d4aSKarsten Keil #define ISAC_REV		"2.0"
21cae86d4aSKarsten Keil 
22cae86d4aSKarsten Keil MODULE_AUTHOR("Karsten Keil");
23cae86d4aSKarsten Keil MODULE_VERSION(ISAC_REV);
24cae86d4aSKarsten Keil MODULE_LICENSE("GPL v2");
25cae86d4aSKarsten Keil 
26cae86d4aSKarsten Keil #define ReadISAC(is, o)		(is->read_reg(is->dch.hw, o + is->off))
27cae86d4aSKarsten Keil #define	WriteISAC(is, o, v)	(is->write_reg(is->dch.hw, o + is->off, v))
28cae86d4aSKarsten Keil #define ReadHSCX(h, o)		(h->ip->read_reg(h->ip->hw, h->off + o))
29cae86d4aSKarsten Keil #define WriteHSCX(h, o, v)	(h->ip->write_reg(h->ip->hw, h->off + o, v))
30cae86d4aSKarsten Keil #define ReadIPAC(ip, o)		(ip->read_reg(ip->hw, o))
31cae86d4aSKarsten Keil #define WriteIPAC(ip, o, v)	(ip->write_reg(ip->hw, o, v))
32cae86d4aSKarsten Keil 
33cae86d4aSKarsten Keil static inline void
ph_command(struct isac_hw * isac,u8 command)34cae86d4aSKarsten Keil ph_command(struct isac_hw *isac, u8 command)
35cae86d4aSKarsten Keil {
36cae86d4aSKarsten Keil 	pr_debug("%s: ph_command %x\n", isac->name, command);
37cae86d4aSKarsten Keil 	if (isac->type & IPAC_TYPE_ISACX)
38cae86d4aSKarsten Keil 		WriteISAC(isac, ISACX_CIX0, (command << 4) | 0xE);
39cae86d4aSKarsten Keil 	else
40cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_CIX0, (command << 2) | 3);
41cae86d4aSKarsten Keil }
42cae86d4aSKarsten Keil 
43cae86d4aSKarsten Keil static void
isac_ph_state_change(struct isac_hw * isac)44cae86d4aSKarsten Keil isac_ph_state_change(struct isac_hw *isac)
45cae86d4aSKarsten Keil {
46cae86d4aSKarsten Keil 	switch (isac->state) {
47cae86d4aSKarsten Keil 	case (ISAC_IND_RS):
48cae86d4aSKarsten Keil 	case (ISAC_IND_EI):
49cae86d4aSKarsten Keil 		ph_command(isac, ISAC_CMD_DUI);
50cae86d4aSKarsten Keil 	}
51cae86d4aSKarsten Keil 	schedule_event(&isac->dch, FLG_PHCHANGE);
52cae86d4aSKarsten Keil }
53cae86d4aSKarsten Keil 
54cae86d4aSKarsten Keil static void
isac_ph_state_bh(struct dchannel * dch)55cae86d4aSKarsten Keil isac_ph_state_bh(struct dchannel *dch)
56cae86d4aSKarsten Keil {
57cae86d4aSKarsten Keil 	struct isac_hw *isac = container_of(dch, struct isac_hw, dch);
58cae86d4aSKarsten Keil 
59cae86d4aSKarsten Keil 	switch (isac->state) {
60cae86d4aSKarsten Keil 	case ISAC_IND_RS:
61cae86d4aSKarsten Keil 	case ISAC_IND_EI:
62cae86d4aSKarsten Keil 		dch->state = 0;
63cae86d4aSKarsten Keil 		l1_event(dch->l1, HW_RESET_IND);
64cae86d4aSKarsten Keil 		break;
65cae86d4aSKarsten Keil 	case ISAC_IND_DID:
66cae86d4aSKarsten Keil 		dch->state = 3;
67cae86d4aSKarsten Keil 		l1_event(dch->l1, HW_DEACT_CNF);
68cae86d4aSKarsten Keil 		break;
69cae86d4aSKarsten Keil 	case ISAC_IND_DR:
701e1589adSMaciej S. Szmigiero 	case ISAC_IND_DR6:
71cae86d4aSKarsten Keil 		dch->state = 3;
72cae86d4aSKarsten Keil 		l1_event(dch->l1, HW_DEACT_IND);
73cae86d4aSKarsten Keil 		break;
74cae86d4aSKarsten Keil 	case ISAC_IND_PU:
75cae86d4aSKarsten Keil 		dch->state = 4;
76cae86d4aSKarsten Keil 		l1_event(dch->l1, HW_POWERUP_IND);
77cae86d4aSKarsten Keil 		break;
78cae86d4aSKarsten Keil 	case ISAC_IND_RSY:
79cae86d4aSKarsten Keil 		if (dch->state <= 5) {
80cae86d4aSKarsten Keil 			dch->state = 5;
81cae86d4aSKarsten Keil 			l1_event(dch->l1, ANYSIGNAL);
82cae86d4aSKarsten Keil 		} else {
83cae86d4aSKarsten Keil 			dch->state = 8;
84cae86d4aSKarsten Keil 			l1_event(dch->l1, LOSTFRAMING);
85cae86d4aSKarsten Keil 		}
86cae86d4aSKarsten Keil 		break;
87cae86d4aSKarsten Keil 	case ISAC_IND_ARD:
88cae86d4aSKarsten Keil 		dch->state = 6;
89cae86d4aSKarsten Keil 		l1_event(dch->l1, INFO2);
90cae86d4aSKarsten Keil 		break;
91cae86d4aSKarsten Keil 	case ISAC_IND_AI8:
92cae86d4aSKarsten Keil 		dch->state = 7;
93cae86d4aSKarsten Keil 		l1_event(dch->l1, INFO4_P8);
94cae86d4aSKarsten Keil 		break;
95cae86d4aSKarsten Keil 	case ISAC_IND_AI10:
96cae86d4aSKarsten Keil 		dch->state = 7;
97cae86d4aSKarsten Keil 		l1_event(dch->l1, INFO4_P10);
98cae86d4aSKarsten Keil 		break;
99cae86d4aSKarsten Keil 	}
100cae86d4aSKarsten Keil 	pr_debug("%s: TE newstate %x\n", isac->name, dch->state);
101cae86d4aSKarsten Keil }
102cae86d4aSKarsten Keil 
103569e937eSBaoyou Xie static void
isac_empty_fifo(struct isac_hw * isac,int count)104cae86d4aSKarsten Keil isac_empty_fifo(struct isac_hw *isac, int count)
105cae86d4aSKarsten Keil {
106cae86d4aSKarsten Keil 	u8 *ptr;
107cae86d4aSKarsten Keil 
108cae86d4aSKarsten Keil 	pr_debug("%s: %s  %d\n", isac->name, __func__, count);
109cae86d4aSKarsten Keil 
110cae86d4aSKarsten Keil 	if (!isac->dch.rx_skb) {
111cae86d4aSKarsten Keil 		isac->dch.rx_skb = mI_alloc_skb(isac->dch.maxlen, GFP_ATOMIC);
112cae86d4aSKarsten Keil 		if (!isac->dch.rx_skb) {
113cae86d4aSKarsten Keil 			pr_info("%s: D receive out of memory\n", isac->name);
114cae86d4aSKarsten Keil 			WriteISAC(isac, ISAC_CMDR, 0x80);
115cae86d4aSKarsten Keil 			return;
116cae86d4aSKarsten Keil 		}
117cae86d4aSKarsten Keil 	}
118cae86d4aSKarsten Keil 	if ((isac->dch.rx_skb->len + count) >= isac->dch.maxlen) {
119cae86d4aSKarsten Keil 		pr_debug("%s: %s overrun %d\n", isac->name, __func__,
120cae86d4aSKarsten Keil 			 isac->dch.rx_skb->len + count);
121cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_CMDR, 0x80);
122cae86d4aSKarsten Keil 		return;
123cae86d4aSKarsten Keil 	}
124cae86d4aSKarsten Keil 	ptr = skb_put(isac->dch.rx_skb, count);
125cae86d4aSKarsten Keil 	isac->read_fifo(isac->dch.hw, isac->off, ptr, count);
126cae86d4aSKarsten Keil 	WriteISAC(isac, ISAC_CMDR, 0x80);
127cae86d4aSKarsten Keil 	if (isac->dch.debug & DEBUG_HW_DFIFO) {
128cae86d4aSKarsten Keil 		char	pfx[MISDN_MAX_IDLEN + 16];
129cae86d4aSKarsten Keil 
130cae86d4aSKarsten Keil 		snprintf(pfx, MISDN_MAX_IDLEN + 15, "D-recv %s %d ",
131cae86d4aSKarsten Keil 			 isac->name, count);
132cae86d4aSKarsten Keil 		print_hex_dump_bytes(pfx, DUMP_PREFIX_OFFSET, ptr, count);
133cae86d4aSKarsten Keil 	}
134cae86d4aSKarsten Keil }
135cae86d4aSKarsten Keil 
136cae86d4aSKarsten Keil static void
isac_fill_fifo(struct isac_hw * isac)137cae86d4aSKarsten Keil isac_fill_fifo(struct isac_hw *isac)
138cae86d4aSKarsten Keil {
139cae86d4aSKarsten Keil 	int count, more;
140cae86d4aSKarsten Keil 	u8 *ptr;
141cae86d4aSKarsten Keil 
142cae86d4aSKarsten Keil 	if (!isac->dch.tx_skb)
143cae86d4aSKarsten Keil 		return;
144cae86d4aSKarsten Keil 	count = isac->dch.tx_skb->len - isac->dch.tx_idx;
145cae86d4aSKarsten Keil 	if (count <= 0)
146cae86d4aSKarsten Keil 		return;
147cae86d4aSKarsten Keil 
148cae86d4aSKarsten Keil 	more = 0;
149cae86d4aSKarsten Keil 	if (count > 32) {
150cae86d4aSKarsten Keil 		more = !0;
151cae86d4aSKarsten Keil 		count = 32;
152cae86d4aSKarsten Keil 	}
153cae86d4aSKarsten Keil 	pr_debug("%s: %s  %d\n", isac->name, __func__, count);
154cae86d4aSKarsten Keil 	ptr = isac->dch.tx_skb->data + isac->dch.tx_idx;
155cae86d4aSKarsten Keil 	isac->dch.tx_idx += count;
156cae86d4aSKarsten Keil 	isac->write_fifo(isac->dch.hw, isac->off, ptr, count);
157cae86d4aSKarsten Keil 	WriteISAC(isac, ISAC_CMDR, more ? 0x8 : 0xa);
158cae86d4aSKarsten Keil 	if (test_and_set_bit(FLG_BUSY_TIMER, &isac->dch.Flags)) {
159cae86d4aSKarsten Keil 		pr_debug("%s: %s dbusytimer running\n", isac->name, __func__);
160cae86d4aSKarsten Keil 		del_timer(&isac->dch.timer);
161cae86d4aSKarsten Keil 	}
162cae86d4aSKarsten Keil 	isac->dch.timer.expires = jiffies + ((DBUSY_TIMER_VALUE * HZ)/1000);
163cae86d4aSKarsten Keil 	add_timer(&isac->dch.timer);
164cae86d4aSKarsten Keil 	if (isac->dch.debug & DEBUG_HW_DFIFO) {
165cae86d4aSKarsten Keil 		char	pfx[MISDN_MAX_IDLEN + 16];
166cae86d4aSKarsten Keil 
167cae86d4aSKarsten Keil 		snprintf(pfx, MISDN_MAX_IDLEN + 15, "D-send %s %d ",
168cae86d4aSKarsten Keil 			 isac->name, count);
169cae86d4aSKarsten Keil 		print_hex_dump_bytes(pfx, DUMP_PREFIX_OFFSET, ptr, count);
170cae86d4aSKarsten Keil 	}
171cae86d4aSKarsten Keil }
172cae86d4aSKarsten Keil 
173cae86d4aSKarsten Keil static void
isac_rme_irq(struct isac_hw * isac)174cae86d4aSKarsten Keil isac_rme_irq(struct isac_hw *isac)
175cae86d4aSKarsten Keil {
176cae86d4aSKarsten Keil 	u8 val, count;
177cae86d4aSKarsten Keil 
178cae86d4aSKarsten Keil 	val = ReadISAC(isac, ISAC_RSTA);
179cae86d4aSKarsten Keil 	if ((val & 0x70) != 0x20) {
180cae86d4aSKarsten Keil 		if (val & 0x40) {
181cae86d4aSKarsten Keil 			pr_debug("%s: ISAC RDO\n", isac->name);
182cae86d4aSKarsten Keil #ifdef ERROR_STATISTIC
183cae86d4aSKarsten Keil 			isac->dch.err_rx++;
184cae86d4aSKarsten Keil #endif
185cae86d4aSKarsten Keil 		}
186cae86d4aSKarsten Keil 		if (!(val & 0x20)) {
187cae86d4aSKarsten Keil 			pr_debug("%s: ISAC CRC error\n", isac->name);
188cae86d4aSKarsten Keil #ifdef ERROR_STATISTIC
189cae86d4aSKarsten Keil 			isac->dch.err_crc++;
190cae86d4aSKarsten Keil #endif
191cae86d4aSKarsten Keil 		}
192cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_CMDR, 0x80);
193cae86d4aSKarsten Keil 		dev_kfree_skb(isac->dch.rx_skb);
194cae86d4aSKarsten Keil 		isac->dch.rx_skb = NULL;
195cae86d4aSKarsten Keil 	} else {
196cae86d4aSKarsten Keil 		count = ReadISAC(isac, ISAC_RBCL) & 0x1f;
197cae86d4aSKarsten Keil 		if (count == 0)
198cae86d4aSKarsten Keil 			count = 32;
199cae86d4aSKarsten Keil 		isac_empty_fifo(isac, count);
200cae86d4aSKarsten Keil 		recv_Dchannel(&isac->dch);
201cae86d4aSKarsten Keil 	}
202cae86d4aSKarsten Keil }
203cae86d4aSKarsten Keil 
204cae86d4aSKarsten Keil static void
isac_xpr_irq(struct isac_hw * isac)205cae86d4aSKarsten Keil isac_xpr_irq(struct isac_hw *isac)
206cae86d4aSKarsten Keil {
207cae86d4aSKarsten Keil 	if (test_and_clear_bit(FLG_BUSY_TIMER, &isac->dch.Flags))
208cae86d4aSKarsten Keil 		del_timer(&isac->dch.timer);
209cae86d4aSKarsten Keil 	if (isac->dch.tx_skb && isac->dch.tx_idx < isac->dch.tx_skb->len) {
210cae86d4aSKarsten Keil 		isac_fill_fifo(isac);
211cae86d4aSKarsten Keil 	} else {
212cae86d4aSKarsten Keil 		dev_kfree_skb(isac->dch.tx_skb);
213cae86d4aSKarsten Keil 		if (get_next_dframe(&isac->dch))
214cae86d4aSKarsten Keil 			isac_fill_fifo(isac);
215cae86d4aSKarsten Keil 	}
216cae86d4aSKarsten Keil }
217cae86d4aSKarsten Keil 
218cae86d4aSKarsten Keil static void
isac_retransmit(struct isac_hw * isac)219cae86d4aSKarsten Keil isac_retransmit(struct isac_hw *isac)
220cae86d4aSKarsten Keil {
221cae86d4aSKarsten Keil 	if (test_and_clear_bit(FLG_BUSY_TIMER, &isac->dch.Flags))
222cae86d4aSKarsten Keil 		del_timer(&isac->dch.timer);
223cae86d4aSKarsten Keil 	if (test_bit(FLG_TX_BUSY, &isac->dch.Flags)) {
224cae86d4aSKarsten Keil 		/* Restart frame */
225cae86d4aSKarsten Keil 		isac->dch.tx_idx = 0;
226cae86d4aSKarsten Keil 		isac_fill_fifo(isac);
227cae86d4aSKarsten Keil 	} else if (isac->dch.tx_skb) { /* should not happen */
228cae86d4aSKarsten Keil 		pr_info("%s: tx_skb exist but not busy\n", isac->name);
229cae86d4aSKarsten Keil 		test_and_set_bit(FLG_TX_BUSY, &isac->dch.Flags);
230cae86d4aSKarsten Keil 		isac->dch.tx_idx = 0;
231cae86d4aSKarsten Keil 		isac_fill_fifo(isac);
232cae86d4aSKarsten Keil 	} else {
233cae86d4aSKarsten Keil 		pr_info("%s: ISAC XDU no TX_BUSY\n", isac->name);
234cae86d4aSKarsten Keil 		if (get_next_dframe(&isac->dch))
235cae86d4aSKarsten Keil 			isac_fill_fifo(isac);
236cae86d4aSKarsten Keil 	}
237cae86d4aSKarsten Keil }
238cae86d4aSKarsten Keil 
239cae86d4aSKarsten Keil static void
isac_mos_irq(struct isac_hw * isac)240cae86d4aSKarsten Keil isac_mos_irq(struct isac_hw *isac)
241cae86d4aSKarsten Keil {
242cae86d4aSKarsten Keil 	u8 val;
243cae86d4aSKarsten Keil 	int ret;
244cae86d4aSKarsten Keil 
245cae86d4aSKarsten Keil 	val = ReadISAC(isac, ISAC_MOSR);
246cae86d4aSKarsten Keil 	pr_debug("%s: ISAC MOSR %02x\n", isac->name, val);
247cae86d4aSKarsten Keil #if ARCOFI_USE
248cae86d4aSKarsten Keil 	if (val & 0x08) {
249cae86d4aSKarsten Keil 		if (!isac->mon_rx) {
250cae86d4aSKarsten Keil 			isac->mon_rx = kmalloc(MAX_MON_FRAME, GFP_ATOMIC);
251cae86d4aSKarsten Keil 			if (!isac->mon_rx) {
252cae86d4aSKarsten Keil 				pr_info("%s: ISAC MON RX out of memory!\n",
253cae86d4aSKarsten Keil 					isac->name);
254cae86d4aSKarsten Keil 				isac->mocr &= 0xf0;
255cae86d4aSKarsten Keil 				isac->mocr |= 0x0a;
256cae86d4aSKarsten Keil 				WriteISAC(isac, ISAC_MOCR, isac->mocr);
257cae86d4aSKarsten Keil 				goto afterMONR0;
258cae86d4aSKarsten Keil 			} else
259cae86d4aSKarsten Keil 				isac->mon_rxp = 0;
260cae86d4aSKarsten Keil 		}
261cae86d4aSKarsten Keil 		if (isac->mon_rxp >= MAX_MON_FRAME) {
262cae86d4aSKarsten Keil 			isac->mocr &= 0xf0;
263cae86d4aSKarsten Keil 			isac->mocr |= 0x0a;
264cae86d4aSKarsten Keil 			WriteISAC(isac, ISAC_MOCR, isac->mocr);
265cae86d4aSKarsten Keil 			isac->mon_rxp = 0;
266cae86d4aSKarsten Keil 			pr_debug("%s: ISAC MON RX overflow!\n", isac->name);
267cae86d4aSKarsten Keil 			goto afterMONR0;
268cae86d4aSKarsten Keil 		}
269cae86d4aSKarsten Keil 		isac->mon_rx[isac->mon_rxp++] = ReadISAC(isac, ISAC_MOR0);
270cae86d4aSKarsten Keil 		pr_debug("%s: ISAC MOR0 %02x\n", isac->name,
271cae86d4aSKarsten Keil 			 isac->mon_rx[isac->mon_rxp - 1]);
272cae86d4aSKarsten Keil 		if (isac->mon_rxp == 1) {
273cae86d4aSKarsten Keil 			isac->mocr |= 0x04;
274cae86d4aSKarsten Keil 			WriteISAC(isac, ISAC_MOCR, isac->mocr);
275cae86d4aSKarsten Keil 		}
276cae86d4aSKarsten Keil 	}
277cae86d4aSKarsten Keil afterMONR0:
278cae86d4aSKarsten Keil 	if (val & 0x80) {
279cae86d4aSKarsten Keil 		if (!isac->mon_rx) {
280cae86d4aSKarsten Keil 			isac->mon_rx = kmalloc(MAX_MON_FRAME, GFP_ATOMIC);
281cae86d4aSKarsten Keil 			if (!isac->mon_rx) {
282cae86d4aSKarsten Keil 				pr_info("%s: ISAC MON RX out of memory!\n",
283cae86d4aSKarsten Keil 					isac->name);
284cae86d4aSKarsten Keil 				isac->mocr &= 0x0f;
285cae86d4aSKarsten Keil 				isac->mocr |= 0xa0;
286cae86d4aSKarsten Keil 				WriteISAC(isac, ISAC_MOCR, isac->mocr);
287cae86d4aSKarsten Keil 				goto afterMONR1;
288cae86d4aSKarsten Keil 			} else
289cae86d4aSKarsten Keil 				isac->mon_rxp = 0;
290cae86d4aSKarsten Keil 		}
291cae86d4aSKarsten Keil 		if (isac->mon_rxp >= MAX_MON_FRAME) {
292cae86d4aSKarsten Keil 			isac->mocr &= 0x0f;
293cae86d4aSKarsten Keil 			isac->mocr |= 0xa0;
294cae86d4aSKarsten Keil 			WriteISAC(isac, ISAC_MOCR, isac->mocr);
295cae86d4aSKarsten Keil 			isac->mon_rxp = 0;
296cae86d4aSKarsten Keil 			pr_debug("%s: ISAC MON RX overflow!\n", isac->name);
297cae86d4aSKarsten Keil 			goto afterMONR1;
298cae86d4aSKarsten Keil 		}
299cae86d4aSKarsten Keil 		isac->mon_rx[isac->mon_rxp++] = ReadISAC(isac, ISAC_MOR1);
300cae86d4aSKarsten Keil 		pr_debug("%s: ISAC MOR1 %02x\n", isac->name,
301cae86d4aSKarsten Keil 			 isac->mon_rx[isac->mon_rxp - 1]);
302cae86d4aSKarsten Keil 		isac->mocr |= 0x40;
303cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_MOCR, isac->mocr);
304cae86d4aSKarsten Keil 	}
305cae86d4aSKarsten Keil afterMONR1:
306cae86d4aSKarsten Keil 	if (val & 0x04) {
307cae86d4aSKarsten Keil 		isac->mocr &= 0xf0;
308cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_MOCR, isac->mocr);
309cae86d4aSKarsten Keil 		isac->mocr |= 0x0a;
310cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_MOCR, isac->mocr);
311cae86d4aSKarsten Keil 		if (isac->monitor) {
312cae86d4aSKarsten Keil 			ret = isac->monitor(isac->dch.hw, MONITOR_RX_0,
313cae86d4aSKarsten Keil 					    isac->mon_rx, isac->mon_rxp);
314cae86d4aSKarsten Keil 			if (ret)
315cae86d4aSKarsten Keil 				kfree(isac->mon_rx);
316cae86d4aSKarsten Keil 		} else {
317cae86d4aSKarsten Keil 			pr_info("%s: MONITOR 0 received %d but no user\n",
318cae86d4aSKarsten Keil 				isac->name, isac->mon_rxp);
319cae86d4aSKarsten Keil 			kfree(isac->mon_rx);
320cae86d4aSKarsten Keil 		}
321cae86d4aSKarsten Keil 		isac->mon_rx = NULL;
322cae86d4aSKarsten Keil 		isac->mon_rxp = 0;
323cae86d4aSKarsten Keil 	}
324cae86d4aSKarsten Keil 	if (val & 0x40) {
325cae86d4aSKarsten Keil 		isac->mocr &= 0x0f;
326cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_MOCR, isac->mocr);
327cae86d4aSKarsten Keil 		isac->mocr |= 0xa0;
328cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_MOCR, isac->mocr);
329cae86d4aSKarsten Keil 		if (isac->monitor) {
330cae86d4aSKarsten Keil 			ret = isac->monitor(isac->dch.hw, MONITOR_RX_1,
331cae86d4aSKarsten Keil 					    isac->mon_rx, isac->mon_rxp);
332cae86d4aSKarsten Keil 			if (ret)
333cae86d4aSKarsten Keil 				kfree(isac->mon_rx);
334cae86d4aSKarsten Keil 		} else {
335cae86d4aSKarsten Keil 			pr_info("%s: MONITOR 1 received %d but no user\n",
336cae86d4aSKarsten Keil 				isac->name, isac->mon_rxp);
337cae86d4aSKarsten Keil 			kfree(isac->mon_rx);
338cae86d4aSKarsten Keil 		}
339cae86d4aSKarsten Keil 		isac->mon_rx = NULL;
340cae86d4aSKarsten Keil 		isac->mon_rxp = 0;
341cae86d4aSKarsten Keil 	}
342cae86d4aSKarsten Keil 	if (val & 0x02) {
343cae86d4aSKarsten Keil 		if ((!isac->mon_tx) || (isac->mon_txc &&
344cae86d4aSKarsten Keil 					(isac->mon_txp >= isac->mon_txc) && !(val & 0x08))) {
345cae86d4aSKarsten Keil 			isac->mocr &= 0xf0;
346cae86d4aSKarsten Keil 			WriteISAC(isac, ISAC_MOCR, isac->mocr);
347cae86d4aSKarsten Keil 			isac->mocr |= 0x0a;
348cae86d4aSKarsten Keil 			WriteISAC(isac, ISAC_MOCR, isac->mocr);
349cae86d4aSKarsten Keil 			if (isac->mon_txc && (isac->mon_txp >= isac->mon_txc)) {
350cae86d4aSKarsten Keil 				if (isac->monitor)
35102417f47SGustavo A. R. Silva 					isac->monitor(isac->dch.hw,
352cae86d4aSKarsten Keil 						      MONITOR_TX_0, NULL, 0);
353cae86d4aSKarsten Keil 			}
354cae86d4aSKarsten Keil 			kfree(isac->mon_tx);
355cae86d4aSKarsten Keil 			isac->mon_tx = NULL;
356cae86d4aSKarsten Keil 			isac->mon_txc = 0;
357cae86d4aSKarsten Keil 			isac->mon_txp = 0;
358cae86d4aSKarsten Keil 			goto AfterMOX0;
359cae86d4aSKarsten Keil 		}
360cae86d4aSKarsten Keil 		if (isac->mon_txc && (isac->mon_txp >= isac->mon_txc)) {
361cae86d4aSKarsten Keil 			if (isac->monitor)
36202417f47SGustavo A. R. Silva 				isac->monitor(isac->dch.hw,
363cae86d4aSKarsten Keil 					      MONITOR_TX_0, NULL, 0);
364cae86d4aSKarsten Keil 			kfree(isac->mon_tx);
365cae86d4aSKarsten Keil 			isac->mon_tx = NULL;
366cae86d4aSKarsten Keil 			isac->mon_txc = 0;
367cae86d4aSKarsten Keil 			isac->mon_txp = 0;
368cae86d4aSKarsten Keil 			goto AfterMOX0;
369cae86d4aSKarsten Keil 		}
370cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_MOX0, isac->mon_tx[isac->mon_txp++]);
371cae86d4aSKarsten Keil 		pr_debug("%s: ISAC %02x -> MOX0\n", isac->name,
372cae86d4aSKarsten Keil 			 isac->mon_tx[isac->mon_txp - 1]);
373cae86d4aSKarsten Keil 	}
374cae86d4aSKarsten Keil AfterMOX0:
375cae86d4aSKarsten Keil 	if (val & 0x20) {
376cae86d4aSKarsten Keil 		if ((!isac->mon_tx) || (isac->mon_txc &&
377cae86d4aSKarsten Keil 					(isac->mon_txp >= isac->mon_txc) && !(val & 0x80))) {
378cae86d4aSKarsten Keil 			isac->mocr &= 0x0f;
379cae86d4aSKarsten Keil 			WriteISAC(isac, ISAC_MOCR, isac->mocr);
380cae86d4aSKarsten Keil 			isac->mocr |= 0xa0;
381cae86d4aSKarsten Keil 			WriteISAC(isac, ISAC_MOCR, isac->mocr);
382cae86d4aSKarsten Keil 			if (isac->mon_txc && (isac->mon_txp >= isac->mon_txc)) {
383cae86d4aSKarsten Keil 				if (isac->monitor)
38402417f47SGustavo A. R. Silva 					isac->monitor(isac->dch.hw,
385cae86d4aSKarsten Keil 						      MONITOR_TX_1, NULL, 0);
386cae86d4aSKarsten Keil 			}
387cae86d4aSKarsten Keil 			kfree(isac->mon_tx);
388cae86d4aSKarsten Keil 			isac->mon_tx = NULL;
389cae86d4aSKarsten Keil 			isac->mon_txc = 0;
390cae86d4aSKarsten Keil 			isac->mon_txp = 0;
391cae86d4aSKarsten Keil 			goto AfterMOX1;
392cae86d4aSKarsten Keil 		}
393cae86d4aSKarsten Keil 		if (isac->mon_txc && (isac->mon_txp >= isac->mon_txc)) {
394cae86d4aSKarsten Keil 			if (isac->monitor)
39502417f47SGustavo A. R. Silva 				isac->monitor(isac->dch.hw,
396cae86d4aSKarsten Keil 					      MONITOR_TX_1, NULL, 0);
397cae86d4aSKarsten Keil 			kfree(isac->mon_tx);
398cae86d4aSKarsten Keil 			isac->mon_tx = NULL;
399cae86d4aSKarsten Keil 			isac->mon_txc = 0;
400cae86d4aSKarsten Keil 			isac->mon_txp = 0;
401cae86d4aSKarsten Keil 			goto AfterMOX1;
402cae86d4aSKarsten Keil 		}
403cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_MOX1, isac->mon_tx[isac->mon_txp++]);
404cae86d4aSKarsten Keil 		pr_debug("%s: ISAC %02x -> MOX1\n", isac->name,
405cae86d4aSKarsten Keil 			 isac->mon_tx[isac->mon_txp - 1]);
406cae86d4aSKarsten Keil 	}
407cae86d4aSKarsten Keil AfterMOX1:
408cae86d4aSKarsten Keil 	val = 0; /* dummy to avoid warning */
409cae86d4aSKarsten Keil #endif
410cae86d4aSKarsten Keil }
411cae86d4aSKarsten Keil 
412cae86d4aSKarsten Keil static void
isac_cisq_irq(struct isac_hw * isac)413cae86d4aSKarsten Keil isac_cisq_irq(struct isac_hw *isac) {
414cae86d4aSKarsten Keil 	u8 val;
415cae86d4aSKarsten Keil 
416cae86d4aSKarsten Keil 	val = ReadISAC(isac, ISAC_CIR0);
417cae86d4aSKarsten Keil 	pr_debug("%s: ISAC CIR0 %02X\n", isac->name, val);
418cae86d4aSKarsten Keil 	if (val & 2) {
419cae86d4aSKarsten Keil 		pr_debug("%s: ph_state change %x->%x\n", isac->name,
420cae86d4aSKarsten Keil 			 isac->state, (val >> 2) & 0xf);
421cae86d4aSKarsten Keil 		isac->state = (val >> 2) & 0xf;
422cae86d4aSKarsten Keil 		isac_ph_state_change(isac);
423cae86d4aSKarsten Keil 	}
424cae86d4aSKarsten Keil 	if (val & 1) {
425cae86d4aSKarsten Keil 		val = ReadISAC(isac, ISAC_CIR1);
426cae86d4aSKarsten Keil 		pr_debug("%s: ISAC CIR1 %02X\n", isac->name, val);
427cae86d4aSKarsten Keil 	}
428cae86d4aSKarsten Keil }
429cae86d4aSKarsten Keil 
430cae86d4aSKarsten Keil static void
isacsx_cic_irq(struct isac_hw * isac)431cae86d4aSKarsten Keil isacsx_cic_irq(struct isac_hw *isac)
432cae86d4aSKarsten Keil {
433cae86d4aSKarsten Keil 	u8 val;
434cae86d4aSKarsten Keil 
435cae86d4aSKarsten Keil 	val = ReadISAC(isac, ISACX_CIR0);
436cae86d4aSKarsten Keil 	pr_debug("%s: ISACX CIR0 %02X\n", isac->name, val);
437cae86d4aSKarsten Keil 	if (val & ISACX_CIR0_CIC0) {
438cae86d4aSKarsten Keil 		pr_debug("%s: ph_state change %x->%x\n", isac->name,
439cae86d4aSKarsten Keil 			 isac->state, val >> 4);
440cae86d4aSKarsten Keil 		isac->state = val >> 4;
441cae86d4aSKarsten Keil 		isac_ph_state_change(isac);
442cae86d4aSKarsten Keil 	}
443cae86d4aSKarsten Keil }
444cae86d4aSKarsten Keil 
445cae86d4aSKarsten Keil static void
isacsx_rme_irq(struct isac_hw * isac)446cae86d4aSKarsten Keil isacsx_rme_irq(struct isac_hw *isac)
447cae86d4aSKarsten Keil {
448cae86d4aSKarsten Keil 	int count;
449cae86d4aSKarsten Keil 	u8 val;
450cae86d4aSKarsten Keil 
451cae86d4aSKarsten Keil 	val = ReadISAC(isac, ISACX_RSTAD);
452cae86d4aSKarsten Keil 	if ((val & (ISACX_RSTAD_VFR |
453cae86d4aSKarsten Keil 		    ISACX_RSTAD_RDO |
454cae86d4aSKarsten Keil 		    ISACX_RSTAD_CRC |
455cae86d4aSKarsten Keil 		    ISACX_RSTAD_RAB))
456cae86d4aSKarsten Keil 	    != (ISACX_RSTAD_VFR | ISACX_RSTAD_CRC)) {
457cae86d4aSKarsten Keil 		pr_debug("%s: RSTAD %#x, dropped\n", isac->name, val);
458cae86d4aSKarsten Keil #ifdef ERROR_STATISTIC
459cae86d4aSKarsten Keil 		if (val & ISACX_RSTAD_CRC)
460cae86d4aSKarsten Keil 			isac->dch.err_rx++;
461cae86d4aSKarsten Keil 		else
462cae86d4aSKarsten Keil 			isac->dch.err_crc++;
463cae86d4aSKarsten Keil #endif
464cae86d4aSKarsten Keil 		WriteISAC(isac, ISACX_CMDRD, ISACX_CMDRD_RMC);
465cae86d4aSKarsten Keil 		dev_kfree_skb(isac->dch.rx_skb);
466cae86d4aSKarsten Keil 		isac->dch.rx_skb = NULL;
467cae86d4aSKarsten Keil 	} else {
468cae86d4aSKarsten Keil 		count = ReadISAC(isac, ISACX_RBCLD) & 0x1f;
469cae86d4aSKarsten Keil 		if (count == 0)
470cae86d4aSKarsten Keil 			count = 32;
471cae86d4aSKarsten Keil 		isac_empty_fifo(isac, count);
472cae86d4aSKarsten Keil 		if (isac->dch.rx_skb) {
473cae86d4aSKarsten Keil 			skb_trim(isac->dch.rx_skb, isac->dch.rx_skb->len - 1);
474cae86d4aSKarsten Keil 			pr_debug("%s: dchannel received %d\n", isac->name,
475cae86d4aSKarsten Keil 				 isac->dch.rx_skb->len);
476cae86d4aSKarsten Keil 			recv_Dchannel(&isac->dch);
477cae86d4aSKarsten Keil 		}
478cae86d4aSKarsten Keil 	}
479cae86d4aSKarsten Keil }
480cae86d4aSKarsten Keil 
481cae86d4aSKarsten Keil irqreturn_t
mISDNisac_irq(struct isac_hw * isac,u8 val)482cae86d4aSKarsten Keil mISDNisac_irq(struct isac_hw *isac, u8 val)
483cae86d4aSKarsten Keil {
484cae86d4aSKarsten Keil 	if (unlikely(!val))
485cae86d4aSKarsten Keil 		return IRQ_NONE;
486cae86d4aSKarsten Keil 	pr_debug("%s: ISAC interrupt %02x\n", isac->name, val);
487cae86d4aSKarsten Keil 	if (isac->type & IPAC_TYPE_ISACX) {
488cae86d4aSKarsten Keil 		if (val & ISACX__CIC)
489cae86d4aSKarsten Keil 			isacsx_cic_irq(isac);
490cae86d4aSKarsten Keil 		if (val & ISACX__ICD) {
491cae86d4aSKarsten Keil 			val = ReadISAC(isac, ISACX_ISTAD);
492cae86d4aSKarsten Keil 			pr_debug("%s: ISTAD %02x\n", isac->name, val);
493cae86d4aSKarsten Keil 			if (val & ISACX_D_XDU) {
494cae86d4aSKarsten Keil 				pr_debug("%s: ISAC XDU\n", isac->name);
495cae86d4aSKarsten Keil #ifdef ERROR_STATISTIC
496cae86d4aSKarsten Keil 				isac->dch.err_tx++;
497cae86d4aSKarsten Keil #endif
498cae86d4aSKarsten Keil 				isac_retransmit(isac);
499cae86d4aSKarsten Keil 			}
500cae86d4aSKarsten Keil 			if (val & ISACX_D_XMR) {
501cae86d4aSKarsten Keil 				pr_debug("%s: ISAC XMR\n", isac->name);
502cae86d4aSKarsten Keil #ifdef ERROR_STATISTIC
503cae86d4aSKarsten Keil 				isac->dch.err_tx++;
504cae86d4aSKarsten Keil #endif
505cae86d4aSKarsten Keil 				isac_retransmit(isac);
506cae86d4aSKarsten Keil 			}
507cae86d4aSKarsten Keil 			if (val & ISACX_D_XPR)
508cae86d4aSKarsten Keil 				isac_xpr_irq(isac);
509cae86d4aSKarsten Keil 			if (val & ISACX_D_RFO) {
510cae86d4aSKarsten Keil 				pr_debug("%s: ISAC RFO\n", isac->name);
511cae86d4aSKarsten Keil 				WriteISAC(isac, ISACX_CMDRD, ISACX_CMDRD_RMC);
512cae86d4aSKarsten Keil 			}
513cae86d4aSKarsten Keil 			if (val & ISACX_D_RME)
514cae86d4aSKarsten Keil 				isacsx_rme_irq(isac);
515cae86d4aSKarsten Keil 			if (val & ISACX_D_RPF)
516cae86d4aSKarsten Keil 				isac_empty_fifo(isac, 0x20);
517cae86d4aSKarsten Keil 		}
518cae86d4aSKarsten Keil 	} else {
519cae86d4aSKarsten Keil 		if (val & 0x80)	/* RME */
520cae86d4aSKarsten Keil 			isac_rme_irq(isac);
521cae86d4aSKarsten Keil 		if (val & 0x40)	/* RPF */
522cae86d4aSKarsten Keil 			isac_empty_fifo(isac, 32);
523cae86d4aSKarsten Keil 		if (val & 0x10)	/* XPR */
524cae86d4aSKarsten Keil 			isac_xpr_irq(isac);
525cae86d4aSKarsten Keil 		if (val & 0x04)	/* CISQ */
526cae86d4aSKarsten Keil 			isac_cisq_irq(isac);
527cae86d4aSKarsten Keil 		if (val & 0x20)	/* RSC - never */
528cae86d4aSKarsten Keil 			pr_debug("%s: ISAC RSC interrupt\n", isac->name);
529cae86d4aSKarsten Keil 		if (val & 0x02)	/* SIN - never */
530cae86d4aSKarsten Keil 			pr_debug("%s: ISAC SIN interrupt\n", isac->name);
531cae86d4aSKarsten Keil 		if (val & 0x01) {	/* EXI */
532cae86d4aSKarsten Keil 			val = ReadISAC(isac, ISAC_EXIR);
533cae86d4aSKarsten Keil 			pr_debug("%s: ISAC EXIR %02x\n", isac->name, val);
534cae86d4aSKarsten Keil 			if (val & 0x80)	/* XMR */
535cae86d4aSKarsten Keil 				pr_debug("%s: ISAC XMR\n", isac->name);
536cae86d4aSKarsten Keil 			if (val & 0x40) { /* XDU */
537cae86d4aSKarsten Keil 				pr_debug("%s: ISAC XDU\n", isac->name);
538cae86d4aSKarsten Keil #ifdef ERROR_STATISTIC
539cae86d4aSKarsten Keil 				isac->dch.err_tx++;
540cae86d4aSKarsten Keil #endif
541cae86d4aSKarsten Keil 				isac_retransmit(isac);
542cae86d4aSKarsten Keil 			}
543cae86d4aSKarsten Keil 			if (val & 0x04)	/* MOS */
544cae86d4aSKarsten Keil 				isac_mos_irq(isac);
545cae86d4aSKarsten Keil 		}
546cae86d4aSKarsten Keil 	}
547cae86d4aSKarsten Keil 	return IRQ_HANDLED;
548cae86d4aSKarsten Keil }
549cae86d4aSKarsten Keil EXPORT_SYMBOL(mISDNisac_irq);
550cae86d4aSKarsten Keil 
551cae86d4aSKarsten Keil static int
isac_l1hw(struct mISDNchannel * ch,struct sk_buff * skb)552cae86d4aSKarsten Keil isac_l1hw(struct mISDNchannel *ch, struct sk_buff *skb)
553cae86d4aSKarsten Keil {
554cae86d4aSKarsten Keil 	struct mISDNdevice	*dev = container_of(ch, struct mISDNdevice, D);
555cae86d4aSKarsten Keil 	struct dchannel		*dch = container_of(dev, struct dchannel, dev);
556cae86d4aSKarsten Keil 	struct isac_hw		*isac = container_of(dch, struct isac_hw, dch);
557cae86d4aSKarsten Keil 	int			ret = -EINVAL;
558cae86d4aSKarsten Keil 	struct mISDNhead	*hh = mISDN_HEAD_P(skb);
559cae86d4aSKarsten Keil 	u32			id;
560cae86d4aSKarsten Keil 	u_long			flags;
561cae86d4aSKarsten Keil 
562cae86d4aSKarsten Keil 	switch (hh->prim) {
563cae86d4aSKarsten Keil 	case PH_DATA_REQ:
564cae86d4aSKarsten Keil 		spin_lock_irqsave(isac->hwlock, flags);
565cae86d4aSKarsten Keil 		ret = dchannel_senddata(dch, skb);
566cae86d4aSKarsten Keil 		if (ret > 0) { /* direct TX */
567cae86d4aSKarsten Keil 			id = hh->id; /* skb can be freed */
568cae86d4aSKarsten Keil 			isac_fill_fifo(isac);
569cae86d4aSKarsten Keil 			ret = 0;
570cae86d4aSKarsten Keil 			spin_unlock_irqrestore(isac->hwlock, flags);
571cae86d4aSKarsten Keil 			queue_ch_frame(ch, PH_DATA_CNF, id, NULL);
572cae86d4aSKarsten Keil 		} else
573cae86d4aSKarsten Keil 			spin_unlock_irqrestore(isac->hwlock, flags);
574cae86d4aSKarsten Keil 		return ret;
575cae86d4aSKarsten Keil 	case PH_ACTIVATE_REQ:
576cae86d4aSKarsten Keil 		ret = l1_event(dch->l1, hh->prim);
577cae86d4aSKarsten Keil 		break;
578cae86d4aSKarsten Keil 	case PH_DEACTIVATE_REQ:
579cae86d4aSKarsten Keil 		test_and_clear_bit(FLG_L2_ACTIVATED, &dch->Flags);
580cae86d4aSKarsten Keil 		ret = l1_event(dch->l1, hh->prim);
581cae86d4aSKarsten Keil 		break;
582cae86d4aSKarsten Keil 	}
583cae86d4aSKarsten Keil 
584cae86d4aSKarsten Keil 	if (!ret)
585cae86d4aSKarsten Keil 		dev_kfree_skb(skb);
586cae86d4aSKarsten Keil 	return ret;
587cae86d4aSKarsten Keil }
588cae86d4aSKarsten Keil 
589cae86d4aSKarsten Keil static int
isac_ctrl(struct isac_hw * isac,u32 cmd,unsigned long para)590c626c127SKarsten Keil isac_ctrl(struct isac_hw *isac, u32 cmd, unsigned long para)
591cae86d4aSKarsten Keil {
592cae86d4aSKarsten Keil 	u8 tl = 0;
593c626c127SKarsten Keil 	unsigned long flags;
594c626c127SKarsten Keil 	int ret = 0;
595cae86d4aSKarsten Keil 
596cae86d4aSKarsten Keil 	switch (cmd) {
597cae86d4aSKarsten Keil 	case HW_TESTLOOP:
598cae86d4aSKarsten Keil 		spin_lock_irqsave(isac->hwlock, flags);
599cae86d4aSKarsten Keil 		if (!(isac->type & IPAC_TYPE_ISACX)) {
600cae86d4aSKarsten Keil 			/* TODO: implement for IPAC_TYPE_ISACX */
601cae86d4aSKarsten Keil 			if (para & 1) /* B1 */
602cae86d4aSKarsten Keil 				tl |= 0x0c;
603cae86d4aSKarsten Keil 			else if (para & 2) /* B2 */
604cae86d4aSKarsten Keil 				tl |= 0x3;
605cae86d4aSKarsten Keil 			/* we only support IOM2 mode */
606cae86d4aSKarsten Keil 			WriteISAC(isac, ISAC_SPCR, tl);
607cae86d4aSKarsten Keil 			if (tl)
608cae86d4aSKarsten Keil 				WriteISAC(isac, ISAC_ADF1, 0x8);
609cae86d4aSKarsten Keil 			else
610cae86d4aSKarsten Keil 				WriteISAC(isac, ISAC_ADF1, 0x0);
611cae86d4aSKarsten Keil 		}
612cae86d4aSKarsten Keil 		spin_unlock_irqrestore(isac->hwlock, flags);
613cae86d4aSKarsten Keil 		break;
614c626c127SKarsten Keil 	case HW_TIMER3_VALUE:
615c626c127SKarsten Keil 		ret = l1_event(isac->dch.l1, HW_TIMER3_VALUE | (para & 0xff));
616c626c127SKarsten Keil 		break;
617cae86d4aSKarsten Keil 	default:
618cae86d4aSKarsten Keil 		pr_debug("%s: %s unknown command %x %lx\n", isac->name,
619cae86d4aSKarsten Keil 			 __func__, cmd, para);
620c626c127SKarsten Keil 		ret = -1;
621cae86d4aSKarsten Keil 	}
622c626c127SKarsten Keil 	return ret;
623cae86d4aSKarsten Keil }
624cae86d4aSKarsten Keil 
625cae86d4aSKarsten Keil static int
isac_l1cmd(struct dchannel * dch,u32 cmd)626cae86d4aSKarsten Keil isac_l1cmd(struct dchannel *dch, u32 cmd)
627cae86d4aSKarsten Keil {
628cae86d4aSKarsten Keil 	struct isac_hw *isac = container_of(dch, struct isac_hw, dch);
629cae86d4aSKarsten Keil 	u_long flags;
630cae86d4aSKarsten Keil 
631cae86d4aSKarsten Keil 	pr_debug("%s: cmd(%x) state(%02x)\n", isac->name, cmd, isac->state);
632cae86d4aSKarsten Keil 	switch (cmd) {
633cae86d4aSKarsten Keil 	case INFO3_P8:
634cae86d4aSKarsten Keil 		spin_lock_irqsave(isac->hwlock, flags);
635cae86d4aSKarsten Keil 		ph_command(isac, ISAC_CMD_AR8);
636cae86d4aSKarsten Keil 		spin_unlock_irqrestore(isac->hwlock, flags);
637cae86d4aSKarsten Keil 		break;
638cae86d4aSKarsten Keil 	case INFO3_P10:
639cae86d4aSKarsten Keil 		spin_lock_irqsave(isac->hwlock, flags);
640cae86d4aSKarsten Keil 		ph_command(isac, ISAC_CMD_AR10);
641cae86d4aSKarsten Keil 		spin_unlock_irqrestore(isac->hwlock, flags);
642cae86d4aSKarsten Keil 		break;
643cae86d4aSKarsten Keil 	case HW_RESET_REQ:
644cae86d4aSKarsten Keil 		spin_lock_irqsave(isac->hwlock, flags);
645cae86d4aSKarsten Keil 		if ((isac->state == ISAC_IND_EI) ||
646cae86d4aSKarsten Keil 		    (isac->state == ISAC_IND_DR) ||
6471e1589adSMaciej S. Szmigiero 		    (isac->state == ISAC_IND_DR6) ||
648cae86d4aSKarsten Keil 		    (isac->state == ISAC_IND_RS))
649cae86d4aSKarsten Keil 			ph_command(isac, ISAC_CMD_TIM);
650cae86d4aSKarsten Keil 		else
651cae86d4aSKarsten Keil 			ph_command(isac, ISAC_CMD_RS);
652cae86d4aSKarsten Keil 		spin_unlock_irqrestore(isac->hwlock, flags);
653cae86d4aSKarsten Keil 		break;
654cae86d4aSKarsten Keil 	case HW_DEACT_REQ:
655cae86d4aSKarsten Keil 		skb_queue_purge(&dch->squeue);
656cae86d4aSKarsten Keil 		if (dch->tx_skb) {
657cae86d4aSKarsten Keil 			dev_kfree_skb(dch->tx_skb);
658cae86d4aSKarsten Keil 			dch->tx_skb = NULL;
659cae86d4aSKarsten Keil 		}
660cae86d4aSKarsten Keil 		dch->tx_idx = 0;
661cae86d4aSKarsten Keil 		if (dch->rx_skb) {
662cae86d4aSKarsten Keil 			dev_kfree_skb(dch->rx_skb);
663cae86d4aSKarsten Keil 			dch->rx_skb = NULL;
664cae86d4aSKarsten Keil 		}
665cae86d4aSKarsten Keil 		test_and_clear_bit(FLG_TX_BUSY, &dch->Flags);
666cae86d4aSKarsten Keil 		if (test_and_clear_bit(FLG_BUSY_TIMER, &dch->Flags))
667cae86d4aSKarsten Keil 			del_timer(&dch->timer);
668cae86d4aSKarsten Keil 		break;
669cae86d4aSKarsten Keil 	case HW_POWERUP_REQ:
670cae86d4aSKarsten Keil 		spin_lock_irqsave(isac->hwlock, flags);
671cae86d4aSKarsten Keil 		ph_command(isac, ISAC_CMD_TIM);
672cae86d4aSKarsten Keil 		spin_unlock_irqrestore(isac->hwlock, flags);
673cae86d4aSKarsten Keil 		break;
674cae86d4aSKarsten Keil 	case PH_ACTIVATE_IND:
675cae86d4aSKarsten Keil 		test_and_set_bit(FLG_ACTIVE, &dch->Flags);
676cae86d4aSKarsten Keil 		_queue_data(&dch->dev.D, cmd, MISDN_ID_ANY, 0, NULL,
677cae86d4aSKarsten Keil 			    GFP_ATOMIC);
678cae86d4aSKarsten Keil 		break;
679cae86d4aSKarsten Keil 	case PH_DEACTIVATE_IND:
680cae86d4aSKarsten Keil 		test_and_clear_bit(FLG_ACTIVE, &dch->Flags);
681cae86d4aSKarsten Keil 		_queue_data(&dch->dev.D, cmd, MISDN_ID_ANY, 0, NULL,
682cae86d4aSKarsten Keil 			    GFP_ATOMIC);
683cae86d4aSKarsten Keil 		break;
684cae86d4aSKarsten Keil 	default:
685cae86d4aSKarsten Keil 		pr_debug("%s: %s unknown command %x\n", isac->name,
686cae86d4aSKarsten Keil 			 __func__, cmd);
687cae86d4aSKarsten Keil 		return -1;
688cae86d4aSKarsten Keil 	}
689cae86d4aSKarsten Keil 	return 0;
690cae86d4aSKarsten Keil }
691cae86d4aSKarsten Keil 
692cae86d4aSKarsten Keil static void
isac_release(struct isac_hw * isac)693cae86d4aSKarsten Keil isac_release(struct isac_hw *isac)
694cae86d4aSKarsten Keil {
695cae86d4aSKarsten Keil 	if (isac->type & IPAC_TYPE_ISACX)
696cae86d4aSKarsten Keil 		WriteISAC(isac, ISACX_MASK, 0xff);
697a9f81244STong Zhang 	else if (isac->type != 0)
698cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_MASK, 0xff);
699cae86d4aSKarsten Keil 	if (isac->dch.timer.function != NULL) {
700cae86d4aSKarsten Keil 		del_timer(&isac->dch.timer);
701cae86d4aSKarsten Keil 		isac->dch.timer.function = NULL;
702cae86d4aSKarsten Keil 	}
703cae86d4aSKarsten Keil 	kfree(isac->mon_rx);
704cae86d4aSKarsten Keil 	isac->mon_rx = NULL;
705cae86d4aSKarsten Keil 	kfree(isac->mon_tx);
706cae86d4aSKarsten Keil 	isac->mon_tx = NULL;
707cae86d4aSKarsten Keil 	if (isac->dch.l1)
708cae86d4aSKarsten Keil 		l1_event(isac->dch.l1, CLOSE_CHANNEL);
709cae86d4aSKarsten Keil 	mISDN_freedchannel(&isac->dch);
710cae86d4aSKarsten Keil }
711cae86d4aSKarsten Keil 
712cae86d4aSKarsten Keil static void
dbusy_timer_handler(struct timer_list * t)713e313ac12SKees Cook dbusy_timer_handler(struct timer_list *t)
714cae86d4aSKarsten Keil {
715e313ac12SKees Cook 	struct isac_hw *isac = from_timer(isac, t, dch.timer);
716cae86d4aSKarsten Keil 	int rbch, star;
717cae86d4aSKarsten Keil 	u_long flags;
718cae86d4aSKarsten Keil 
719cae86d4aSKarsten Keil 	if (test_bit(FLG_BUSY_TIMER, &isac->dch.Flags)) {
720cae86d4aSKarsten Keil 		spin_lock_irqsave(isac->hwlock, flags);
721cae86d4aSKarsten Keil 		rbch = ReadISAC(isac, ISAC_RBCH);
722cae86d4aSKarsten Keil 		star = ReadISAC(isac, ISAC_STAR);
723cae86d4aSKarsten Keil 		pr_debug("%s: D-Channel Busy RBCH %02x STAR %02x\n",
724cae86d4aSKarsten Keil 			 isac->name, rbch, star);
725cae86d4aSKarsten Keil 		if (rbch & ISAC_RBCH_XAC) /* D-Channel Busy */
726cae86d4aSKarsten Keil 			test_and_set_bit(FLG_L1_BUSY, &isac->dch.Flags);
727cae86d4aSKarsten Keil 		else {
728cae86d4aSKarsten Keil 			/* discard frame; reset transceiver */
729cae86d4aSKarsten Keil 			test_and_clear_bit(FLG_BUSY_TIMER, &isac->dch.Flags);
730cae86d4aSKarsten Keil 			if (isac->dch.tx_idx)
731cae86d4aSKarsten Keil 				isac->dch.tx_idx = 0;
732cae86d4aSKarsten Keil 			else
733cae86d4aSKarsten Keil 				pr_info("%s: ISAC D-Channel Busy no tx_idx\n",
734cae86d4aSKarsten Keil 					isac->name);
735cae86d4aSKarsten Keil 			/* Transmitter reset */
736cae86d4aSKarsten Keil 			WriteISAC(isac, ISAC_CMDR, 0x01);
737cae86d4aSKarsten Keil 		}
738cae86d4aSKarsten Keil 		spin_unlock_irqrestore(isac->hwlock, flags);
739cae86d4aSKarsten Keil 	}
740cae86d4aSKarsten Keil }
741cae86d4aSKarsten Keil 
742cae86d4aSKarsten Keil static int
open_dchannel_caller(struct isac_hw * isac,struct channel_req * rq,void * caller)7433e7a8716SArnd Bergmann open_dchannel_caller(struct isac_hw *isac, struct channel_req *rq, void *caller)
744cae86d4aSKarsten Keil {
745cae86d4aSKarsten Keil 	pr_debug("%s: %s dev(%d) open from %p\n", isac->name, __func__,
7463e7a8716SArnd Bergmann 		 isac->dch.dev.id, caller);
747cae86d4aSKarsten Keil 	if (rq->protocol != ISDN_P_TE_S0)
748cae86d4aSKarsten Keil 		return -EINVAL;
749cae86d4aSKarsten Keil 	if (rq->adr.channel == 1)
750cae86d4aSKarsten Keil 		/* E-Channel not supported */
751cae86d4aSKarsten Keil 		return -EINVAL;
752cae86d4aSKarsten Keil 	rq->ch = &isac->dch.dev.D;
753cae86d4aSKarsten Keil 	rq->ch->protocol = rq->protocol;
754cae86d4aSKarsten Keil 	if (isac->dch.state == 7)
755cae86d4aSKarsten Keil 		_queue_data(rq->ch, PH_ACTIVATE_IND, MISDN_ID_ANY,
756cae86d4aSKarsten Keil 			    0, NULL, GFP_KERNEL);
757cae86d4aSKarsten Keil 	return 0;
758cae86d4aSKarsten Keil }
759cae86d4aSKarsten Keil 
7603e7a8716SArnd Bergmann static int
open_dchannel(struct isac_hw * isac,struct channel_req * rq)7613e7a8716SArnd Bergmann open_dchannel(struct isac_hw *isac, struct channel_req *rq)
7623e7a8716SArnd Bergmann {
7633e7a8716SArnd Bergmann 	return open_dchannel_caller(isac, rq, __builtin_return_address(0));
7643e7a8716SArnd Bergmann }
7653e7a8716SArnd Bergmann 
766cae86d4aSKarsten Keil static const char *ISACVer[] =
767cae86d4aSKarsten Keil {"2086/2186 V1.1", "2085 B1", "2085 B2",
768cae86d4aSKarsten Keil  "2085 V2.3"};
769cae86d4aSKarsten Keil 
770cae86d4aSKarsten Keil static int
isac_init(struct isac_hw * isac)771cae86d4aSKarsten Keil isac_init(struct isac_hw *isac)
772cae86d4aSKarsten Keil {
773cae86d4aSKarsten Keil 	u8 val;
774cae86d4aSKarsten Keil 	int err = 0;
775cae86d4aSKarsten Keil 
776cae86d4aSKarsten Keil 	if (!isac->dch.l1) {
777cae86d4aSKarsten Keil 		err = create_l1(&isac->dch, isac_l1cmd);
778cae86d4aSKarsten Keil 		if (err)
779cae86d4aSKarsten Keil 			return err;
780cae86d4aSKarsten Keil 	}
781cae86d4aSKarsten Keil 	isac->mon_tx = NULL;
782cae86d4aSKarsten Keil 	isac->mon_rx = NULL;
783e313ac12SKees Cook 	timer_setup(&isac->dch.timer, dbusy_timer_handler, 0);
784cae86d4aSKarsten Keil 	isac->mocr = 0xaa;
785cae86d4aSKarsten Keil 	if (isac->type & IPAC_TYPE_ISACX) {
786cae86d4aSKarsten Keil 		/* Disable all IRQ */
787cae86d4aSKarsten Keil 		WriteISAC(isac, ISACX_MASK, 0xff);
788cae86d4aSKarsten Keil 		val = ReadISAC(isac, ISACX_STARD);
789cae86d4aSKarsten Keil 		pr_debug("%s: ISACX STARD %x\n", isac->name, val);
790cae86d4aSKarsten Keil 		val = ReadISAC(isac, ISACX_ISTAD);
791cae86d4aSKarsten Keil 		pr_debug("%s: ISACX ISTAD %x\n", isac->name, val);
792cae86d4aSKarsten Keil 		val = ReadISAC(isac, ISACX_ISTA);
793cae86d4aSKarsten Keil 		pr_debug("%s: ISACX ISTA %x\n", isac->name, val);
794cae86d4aSKarsten Keil 		/* clear LDD */
795cae86d4aSKarsten Keil 		WriteISAC(isac, ISACX_TR_CONF0, 0x00);
796cae86d4aSKarsten Keil 		/* enable transmitter */
797cae86d4aSKarsten Keil 		WriteISAC(isac, ISACX_TR_CONF2, 0x00);
798cae86d4aSKarsten Keil 		/* transparent mode 0, RAC, stop/go */
799cae86d4aSKarsten Keil 		WriteISAC(isac, ISACX_MODED, 0xc9);
800cae86d4aSKarsten Keil 		/* all HDLC IRQ unmasked */
801cae86d4aSKarsten Keil 		val = ReadISAC(isac, ISACX_ID);
802cae86d4aSKarsten Keil 		if (isac->dch.debug & DEBUG_HW)
803cae86d4aSKarsten Keil 			pr_notice("%s: ISACX Design ID %x\n",
804cae86d4aSKarsten Keil 				  isac->name, val & 0x3f);
805cae86d4aSKarsten Keil 		val = ReadISAC(isac, ISACX_CIR0);
806cae86d4aSKarsten Keil 		pr_debug("%s: ISACX CIR0 %02X\n", isac->name, val);
807cae86d4aSKarsten Keil 		isac->state = val >> 4;
808cae86d4aSKarsten Keil 		isac_ph_state_change(isac);
809cae86d4aSKarsten Keil 		ph_command(isac, ISAC_CMD_RS);
810cae86d4aSKarsten Keil 		WriteISAC(isac, ISACX_MASK, IPACX__ON);
811cae86d4aSKarsten Keil 		WriteISAC(isac, ISACX_MASKD, 0x00);
812cae86d4aSKarsten Keil 	} else { /* old isac */
813cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_MASK, 0xff);
814cae86d4aSKarsten Keil 		val = ReadISAC(isac, ISAC_STAR);
815cae86d4aSKarsten Keil 		pr_debug("%s: ISAC STAR %x\n", isac->name, val);
816cae86d4aSKarsten Keil 		val = ReadISAC(isac, ISAC_MODE);
817cae86d4aSKarsten Keil 		pr_debug("%s: ISAC MODE %x\n", isac->name, val);
818cae86d4aSKarsten Keil 		val = ReadISAC(isac, ISAC_ADF2);
819cae86d4aSKarsten Keil 		pr_debug("%s: ISAC ADF2 %x\n", isac->name, val);
820cae86d4aSKarsten Keil 		val = ReadISAC(isac, ISAC_ISTA);
821cae86d4aSKarsten Keil 		pr_debug("%s: ISAC ISTA %x\n", isac->name, val);
822cae86d4aSKarsten Keil 		if (val & 0x01) {
823cae86d4aSKarsten Keil 			val = ReadISAC(isac, ISAC_EXIR);
824cae86d4aSKarsten Keil 			pr_debug("%s: ISAC EXIR %x\n", isac->name, val);
825cae86d4aSKarsten Keil 		}
826cae86d4aSKarsten Keil 		val = ReadISAC(isac, ISAC_RBCH);
827cae86d4aSKarsten Keil 		if (isac->dch.debug & DEBUG_HW)
828cae86d4aSKarsten Keil 			pr_notice("%s: ISAC version (%x): %s\n", isac->name,
829cae86d4aSKarsten Keil 				  val, ISACVer[(val >> 5) & 3]);
830cae86d4aSKarsten Keil 		isac->type |= ((val >> 5) & 3);
831cae86d4aSKarsten Keil 		if (!isac->adf2)
832cae86d4aSKarsten Keil 			isac->adf2 = 0x80;
833cae86d4aSKarsten Keil 		if (!(isac->adf2 & 0x80)) { /* only IOM 2 Mode */
834cae86d4aSKarsten Keil 			pr_info("%s: only support IOM2 mode but adf2=%02x\n",
835cae86d4aSKarsten Keil 				isac->name, isac->adf2);
836cae86d4aSKarsten Keil 			isac_release(isac);
837cae86d4aSKarsten Keil 			return -EINVAL;
838cae86d4aSKarsten Keil 		}
839cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_ADF2, isac->adf2);
840cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_SQXR, 0x2f);
841cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_SPCR, 0x00);
842cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_STCR, 0x70);
843cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_MODE, 0xc9);
844cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_TIMR, 0x00);
845cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_ADF1, 0x00);
846cae86d4aSKarsten Keil 		val = ReadISAC(isac, ISAC_CIR0);
847cae86d4aSKarsten Keil 		pr_debug("%s: ISAC CIR0 %x\n", isac->name, val);
848cae86d4aSKarsten Keil 		isac->state = (val >> 2) & 0xf;
849cae86d4aSKarsten Keil 		isac_ph_state_change(isac);
850cae86d4aSKarsten Keil 		ph_command(isac, ISAC_CMD_RS);
851cae86d4aSKarsten Keil 		WriteISAC(isac, ISAC_MASK, 0);
852cae86d4aSKarsten Keil 	}
853cae86d4aSKarsten Keil 	return err;
854cae86d4aSKarsten Keil }
855cae86d4aSKarsten Keil 
856cae86d4aSKarsten Keil int
mISDNisac_init(struct isac_hw * isac,void * hw)857cae86d4aSKarsten Keil mISDNisac_init(struct isac_hw *isac, void *hw)
858cae86d4aSKarsten Keil {
859cae86d4aSKarsten Keil 	mISDN_initdchannel(&isac->dch, MAX_DFRAME_LEN_L1, isac_ph_state_bh);
860cae86d4aSKarsten Keil 	isac->dch.hw = hw;
861cae86d4aSKarsten Keil 	isac->dch.dev.D.send = isac_l1hw;
862cae86d4aSKarsten Keil 	isac->init = isac_init;
863cae86d4aSKarsten Keil 	isac->release = isac_release;
864cae86d4aSKarsten Keil 	isac->ctrl = isac_ctrl;
865cae86d4aSKarsten Keil 	isac->open = open_dchannel;
866cae86d4aSKarsten Keil 	isac->dch.dev.Dprotocols = (1 << ISDN_P_TE_S0);
867cae86d4aSKarsten Keil 	isac->dch.dev.nrbchan = 2;
868cae86d4aSKarsten Keil 	return 0;
869cae86d4aSKarsten Keil }
870cae86d4aSKarsten Keil EXPORT_SYMBOL(mISDNisac_init);
871cae86d4aSKarsten Keil 
872cae86d4aSKarsten Keil static void
waitforCEC(struct hscx_hw * hx)873cae86d4aSKarsten Keil waitforCEC(struct hscx_hw *hx)
874cae86d4aSKarsten Keil {
875cae86d4aSKarsten Keil 	u8 starb, to = 50;
876cae86d4aSKarsten Keil 
877cae86d4aSKarsten Keil 	while (to) {
878cae86d4aSKarsten Keil 		starb = ReadHSCX(hx, IPAC_STARB);
879cae86d4aSKarsten Keil 		if (!(starb & 0x04))
880cae86d4aSKarsten Keil 			break;
881cae86d4aSKarsten Keil 		udelay(1);
882cae86d4aSKarsten Keil 		to--;
883cae86d4aSKarsten Keil 	}
884cae86d4aSKarsten Keil 	if (to < 50)
885cae86d4aSKarsten Keil 		pr_debug("%s: B%1d CEC %d us\n", hx->ip->name, hx->bch.nr,
886cae86d4aSKarsten Keil 			 50 - to);
887cae86d4aSKarsten Keil 	if (!to)
888cae86d4aSKarsten Keil 		pr_info("%s: B%1d CEC timeout\n", hx->ip->name, hx->bch.nr);
889cae86d4aSKarsten Keil }
890cae86d4aSKarsten Keil 
891cae86d4aSKarsten Keil 
892cae86d4aSKarsten Keil static void
waitforXFW(struct hscx_hw * hx)893cae86d4aSKarsten Keil waitforXFW(struct hscx_hw *hx)
894cae86d4aSKarsten Keil {
895cae86d4aSKarsten Keil 	u8 starb, to = 50;
896cae86d4aSKarsten Keil 
897cae86d4aSKarsten Keil 	while (to) {
898cae86d4aSKarsten Keil 		starb = ReadHSCX(hx, IPAC_STARB);
899cae86d4aSKarsten Keil 		if ((starb & 0x44) == 0x40)
900cae86d4aSKarsten Keil 			break;
901cae86d4aSKarsten Keil 		udelay(1);
902cae86d4aSKarsten Keil 		to--;
903cae86d4aSKarsten Keil 	}
904cae86d4aSKarsten Keil 	if (to < 50)
905cae86d4aSKarsten Keil 		pr_debug("%s: B%1d XFW %d us\n", hx->ip->name, hx->bch.nr,
906cae86d4aSKarsten Keil 			 50 - to);
907cae86d4aSKarsten Keil 	if (!to)
908cae86d4aSKarsten Keil 		pr_info("%s: B%1d XFW timeout\n", hx->ip->name, hx->bch.nr);
909cae86d4aSKarsten Keil }
910cae86d4aSKarsten Keil 
911cae86d4aSKarsten Keil static void
hscx_cmdr(struct hscx_hw * hx,u8 cmd)912cae86d4aSKarsten Keil hscx_cmdr(struct hscx_hw *hx, u8 cmd)
913cae86d4aSKarsten Keil {
914cae86d4aSKarsten Keil 	if (hx->ip->type & IPAC_TYPE_IPACX)
915cae86d4aSKarsten Keil 		WriteHSCX(hx, IPACX_CMDRB, cmd);
916cae86d4aSKarsten Keil 	else {
917cae86d4aSKarsten Keil 		waitforCEC(hx);
918cae86d4aSKarsten Keil 		WriteHSCX(hx, IPAC_CMDRB, cmd);
919cae86d4aSKarsten Keil 	}
920cae86d4aSKarsten Keil }
921cae86d4aSKarsten Keil 
922cae86d4aSKarsten Keil static void
hscx_empty_fifo(struct hscx_hw * hscx,u8 count)923cae86d4aSKarsten Keil hscx_empty_fifo(struct hscx_hw *hscx, u8 count)
924cae86d4aSKarsten Keil {
925cae86d4aSKarsten Keil 	u8 *p;
9267206e659SKarsten Keil 	int maxlen;
927cae86d4aSKarsten Keil 
928cae86d4aSKarsten Keil 	pr_debug("%s: B%1d %d\n", hscx->ip->name, hscx->bch.nr, count);
929c27b46e7SKarsten Keil 	if (test_bit(FLG_RX_OFF, &hscx->bch.Flags)) {
930c27b46e7SKarsten Keil 		hscx->bch.dropcnt += count;
931c27b46e7SKarsten Keil 		hscx_cmdr(hscx, 0x80); /* RMC */
932c27b46e7SKarsten Keil 		return;
933c27b46e7SKarsten Keil 	}
9347206e659SKarsten Keil 	maxlen = bchannel_get_rxbuf(&hscx->bch, count);
9357206e659SKarsten Keil 	if (maxlen < 0) {
936cae86d4aSKarsten Keil 		hscx_cmdr(hscx, 0x80); /* RMC */
9377206e659SKarsten Keil 		if (hscx->bch.rx_skb)
938cae86d4aSKarsten Keil 			skb_trim(hscx->bch.rx_skb, 0);
939257daba4SKefeng Wang 		pr_warn("%s.B%d: No bufferspace for %d bytes\n",
9407206e659SKarsten Keil 			hscx->ip->name, hscx->bch.nr, count);
941cae86d4aSKarsten Keil 		return;
942cae86d4aSKarsten Keil 	}
943cae86d4aSKarsten Keil 	p = skb_put(hscx->bch.rx_skb, count);
944cae86d4aSKarsten Keil 
945cae86d4aSKarsten Keil 	if (hscx->ip->type & IPAC_TYPE_IPACX)
946cae86d4aSKarsten Keil 		hscx->ip->read_fifo(hscx->ip->hw,
947cae86d4aSKarsten Keil 				    hscx->off + IPACX_RFIFOB, p, count);
948cae86d4aSKarsten Keil 	else
949cae86d4aSKarsten Keil 		hscx->ip->read_fifo(hscx->ip->hw,
950cae86d4aSKarsten Keil 				    hscx->off, p, count);
951cae86d4aSKarsten Keil 
952cae86d4aSKarsten Keil 	hscx_cmdr(hscx, 0x80); /* RMC */
953cae86d4aSKarsten Keil 
954cae86d4aSKarsten Keil 	if (hscx->bch.debug & DEBUG_HW_BFIFO) {
955cae86d4aSKarsten Keil 		snprintf(hscx->log, 64, "B%1d-recv %s %d ",
956cae86d4aSKarsten Keil 			 hscx->bch.nr, hscx->ip->name, count);
957cae86d4aSKarsten Keil 		print_hex_dump_bytes(hscx->log, DUMP_PREFIX_OFFSET, p, count);
958cae86d4aSKarsten Keil 	}
959cae86d4aSKarsten Keil }
960cae86d4aSKarsten Keil 
961cae86d4aSKarsten Keil static void
hscx_fill_fifo(struct hscx_hw * hscx)962cae86d4aSKarsten Keil hscx_fill_fifo(struct hscx_hw *hscx)
963cae86d4aSKarsten Keil {
964cae86d4aSKarsten Keil 	int count, more;
965cae86d4aSKarsten Keil 	u8 *p;
966cae86d4aSKarsten Keil 
9676d1ee48fSKarsten Keil 	if (!hscx->bch.tx_skb) {
9686d1ee48fSKarsten Keil 		if (!test_bit(FLG_TX_EMPTY, &hscx->bch.Flags))
969cae86d4aSKarsten Keil 			return;
9706d1ee48fSKarsten Keil 		count = hscx->fifo_size;
9716d1ee48fSKarsten Keil 		more = 1;
9726d1ee48fSKarsten Keil 		p = hscx->log;
9736d1ee48fSKarsten Keil 		memset(p, hscx->bch.fill[0], count);
9746d1ee48fSKarsten Keil 	} else {
975cae86d4aSKarsten Keil 		count = hscx->bch.tx_skb->len - hscx->bch.tx_idx;
976cae86d4aSKarsten Keil 		if (count <= 0)
977cae86d4aSKarsten Keil 			return;
978cae86d4aSKarsten Keil 		p = hscx->bch.tx_skb->data + hscx->bch.tx_idx;
979cae86d4aSKarsten Keil 
980cae86d4aSKarsten Keil 		more = test_bit(FLG_TRANSPARENT, &hscx->bch.Flags) ? 1 : 0;
981cae86d4aSKarsten Keil 		if (count > hscx->fifo_size) {
982cae86d4aSKarsten Keil 			count = hscx->fifo_size;
983cae86d4aSKarsten Keil 			more = 1;
984cae86d4aSKarsten Keil 		}
9856d1ee48fSKarsten Keil 		pr_debug("%s: B%1d %d/%d/%d\n", hscx->ip->name, hscx->bch.nr,
9866d1ee48fSKarsten Keil 			 count, hscx->bch.tx_idx, hscx->bch.tx_skb->len);
987cae86d4aSKarsten Keil 		hscx->bch.tx_idx += count;
9886d1ee48fSKarsten Keil 	}
989cae86d4aSKarsten Keil 	if (hscx->ip->type & IPAC_TYPE_IPACX)
990cae86d4aSKarsten Keil 		hscx->ip->write_fifo(hscx->ip->hw,
991cae86d4aSKarsten Keil 				     hscx->off + IPACX_XFIFOB, p, count);
992cae86d4aSKarsten Keil 	else {
993cae86d4aSKarsten Keil 		waitforXFW(hscx);
994cae86d4aSKarsten Keil 		hscx->ip->write_fifo(hscx->ip->hw,
995cae86d4aSKarsten Keil 				     hscx->off, p, count);
996cae86d4aSKarsten Keil 	}
997cae86d4aSKarsten Keil 	hscx_cmdr(hscx, more ? 0x08 : 0x0a);
998cae86d4aSKarsten Keil 
9996d1ee48fSKarsten Keil 	if (hscx->bch.tx_skb && (hscx->bch.debug & DEBUG_HW_BFIFO)) {
1000cae86d4aSKarsten Keil 		snprintf(hscx->log, 64, "B%1d-send %s %d ",
1001cae86d4aSKarsten Keil 			 hscx->bch.nr, hscx->ip->name, count);
1002cae86d4aSKarsten Keil 		print_hex_dump_bytes(hscx->log, DUMP_PREFIX_OFFSET, p, count);
1003cae86d4aSKarsten Keil 	}
1004cae86d4aSKarsten Keil }
1005cae86d4aSKarsten Keil 
1006cae86d4aSKarsten Keil static void
hscx_xpr(struct hscx_hw * hx)1007cae86d4aSKarsten Keil hscx_xpr(struct hscx_hw *hx)
1008cae86d4aSKarsten Keil {
10098bfddfbeSKarsten Keil 	if (hx->bch.tx_skb && hx->bch.tx_idx < hx->bch.tx_skb->len) {
1010cae86d4aSKarsten Keil 		hscx_fill_fifo(hx);
10118bfddfbeSKarsten Keil 	} else {
1012cae86d4aSKarsten Keil 		dev_kfree_skb(hx->bch.tx_skb);
10136d1ee48fSKarsten Keil 		if (get_next_bframe(&hx->bch)) {
1014cae86d4aSKarsten Keil 			hscx_fill_fifo(hx);
10156d1ee48fSKarsten Keil 			test_and_clear_bit(FLG_TX_EMPTY, &hx->bch.Flags);
10166d1ee48fSKarsten Keil 		} else if (test_bit(FLG_TX_EMPTY, &hx->bch.Flags)) {
10176d1ee48fSKarsten Keil 			hscx_fill_fifo(hx);
10186d1ee48fSKarsten Keil 		}
1019cae86d4aSKarsten Keil 	}
1020cae86d4aSKarsten Keil }
1021cae86d4aSKarsten Keil 
1022cae86d4aSKarsten Keil static void
ipac_rme(struct hscx_hw * hx)1023cae86d4aSKarsten Keil ipac_rme(struct hscx_hw *hx)
1024cae86d4aSKarsten Keil {
1025cae86d4aSKarsten Keil 	int count;
1026cae86d4aSKarsten Keil 	u8 rstab;
1027cae86d4aSKarsten Keil 
1028cae86d4aSKarsten Keil 	if (hx->ip->type & IPAC_TYPE_IPACX)
1029cae86d4aSKarsten Keil 		rstab = ReadHSCX(hx, IPACX_RSTAB);
1030cae86d4aSKarsten Keil 	else
1031cae86d4aSKarsten Keil 		rstab = ReadHSCX(hx, IPAC_RSTAB);
1032cae86d4aSKarsten Keil 	pr_debug("%s: B%1d RSTAB %02x\n", hx->ip->name, hx->bch.nr, rstab);
1033cae86d4aSKarsten Keil 	if ((rstab & 0xf0) != 0xa0) {
1034cae86d4aSKarsten Keil 		/* !(VFR && !RDO && CRC && !RAB) */
1035cae86d4aSKarsten Keil 		if (!(rstab & 0x80)) {
1036cae86d4aSKarsten Keil 			if (hx->bch.debug & DEBUG_HW_BCHANNEL)
1037cae86d4aSKarsten Keil 				pr_notice("%s: B%1d invalid frame\n",
1038cae86d4aSKarsten Keil 					  hx->ip->name, hx->bch.nr);
1039cae86d4aSKarsten Keil 		}
1040cae86d4aSKarsten Keil 		if (rstab & 0x40) {
1041cae86d4aSKarsten Keil 			if (hx->bch.debug & DEBUG_HW_BCHANNEL)
1042cae86d4aSKarsten Keil 				pr_notice("%s: B%1d RDO proto=%x\n",
1043cae86d4aSKarsten Keil 					  hx->ip->name, hx->bch.nr,
1044cae86d4aSKarsten Keil 					  hx->bch.state);
1045cae86d4aSKarsten Keil 		}
1046cae86d4aSKarsten Keil 		if (!(rstab & 0x20)) {
1047cae86d4aSKarsten Keil 			if (hx->bch.debug & DEBUG_HW_BCHANNEL)
1048cae86d4aSKarsten Keil 				pr_notice("%s: B%1d CRC error\n",
1049cae86d4aSKarsten Keil 					  hx->ip->name, hx->bch.nr);
1050cae86d4aSKarsten Keil 		}
1051cae86d4aSKarsten Keil 		hscx_cmdr(hx, 0x80); /* Do RMC */
1052cae86d4aSKarsten Keil 		return;
1053cae86d4aSKarsten Keil 	}
1054cae86d4aSKarsten Keil 	if (hx->ip->type & IPAC_TYPE_IPACX)
1055cae86d4aSKarsten Keil 		count = ReadHSCX(hx, IPACX_RBCLB);
1056cae86d4aSKarsten Keil 	else
1057cae86d4aSKarsten Keil 		count = ReadHSCX(hx, IPAC_RBCLB);
1058cae86d4aSKarsten Keil 	count &= (hx->fifo_size - 1);
1059cae86d4aSKarsten Keil 	if (count == 0)
1060cae86d4aSKarsten Keil 		count = hx->fifo_size;
1061cae86d4aSKarsten Keil 	hscx_empty_fifo(hx, count);
1062cae86d4aSKarsten Keil 	if (!hx->bch.rx_skb)
1063cae86d4aSKarsten Keil 		return;
1064cae86d4aSKarsten Keil 	if (hx->bch.rx_skb->len < 2) {
1065*dc978706STong Zhang 		pr_debug("%s: B%1d frame too short %d\n",
1066cae86d4aSKarsten Keil 			 hx->ip->name, hx->bch.nr, hx->bch.rx_skb->len);
1067cae86d4aSKarsten Keil 		skb_trim(hx->bch.rx_skb, 0);
1068cae86d4aSKarsten Keil 	} else {
1069cae86d4aSKarsten Keil 		skb_trim(hx->bch.rx_skb, hx->bch.rx_skb->len - 1);
1070034005a0SKarsten Keil 		recv_Bchannel(&hx->bch, 0, false);
1071cae86d4aSKarsten Keil 	}
1072cae86d4aSKarsten Keil }
1073cae86d4aSKarsten Keil 
1074cae86d4aSKarsten Keil static void
ipac_irq(struct hscx_hw * hx,u8 ista)1075cae86d4aSKarsten Keil ipac_irq(struct hscx_hw *hx, u8 ista)
1076cae86d4aSKarsten Keil {
1077cae86d4aSKarsten Keil 	u8 istab, m, exirb = 0;
1078cae86d4aSKarsten Keil 
1079cae86d4aSKarsten Keil 	if (hx->ip->type & IPAC_TYPE_IPACX)
1080cae86d4aSKarsten Keil 		istab = ReadHSCX(hx, IPACX_ISTAB);
1081cae86d4aSKarsten Keil 	else if (hx->ip->type & IPAC_TYPE_IPAC) {
1082cae86d4aSKarsten Keil 		istab = ReadHSCX(hx, IPAC_ISTAB);
1083cae86d4aSKarsten Keil 		m = (hx->bch.nr & 1) ? IPAC__EXA : IPAC__EXB;
1084cae86d4aSKarsten Keil 		if (m & ista) {
1085cae86d4aSKarsten Keil 			exirb = ReadHSCX(hx, IPAC_EXIRB);
1086cae86d4aSKarsten Keil 			pr_debug("%s: B%1d EXIRB %02x\n", hx->ip->name,
1087cae86d4aSKarsten Keil 				 hx->bch.nr, exirb);
1088cae86d4aSKarsten Keil 		}
1089cae86d4aSKarsten Keil 	} else if (hx->bch.nr & 2) { /* HSCX B */
1090cae86d4aSKarsten Keil 		if (ista & (HSCX__EXA | HSCX__ICA))
1091cae86d4aSKarsten Keil 			ipac_irq(&hx->ip->hscx[0], ista);
1092cae86d4aSKarsten Keil 		if (ista & HSCX__EXB) {
1093cae86d4aSKarsten Keil 			exirb = ReadHSCX(hx, IPAC_EXIRB);
1094cae86d4aSKarsten Keil 			pr_debug("%s: B%1d EXIRB %02x\n", hx->ip->name,
1095cae86d4aSKarsten Keil 				 hx->bch.nr, exirb);
1096cae86d4aSKarsten Keil 		}
1097cae86d4aSKarsten Keil 		istab = ista & 0xF8;
1098cae86d4aSKarsten Keil 	} else { /* HSCX A */
1099cae86d4aSKarsten Keil 		istab = ReadHSCX(hx, IPAC_ISTAB);
1100cae86d4aSKarsten Keil 		if (ista & HSCX__EXA) {
1101cae86d4aSKarsten Keil 			exirb = ReadHSCX(hx, IPAC_EXIRB);
1102cae86d4aSKarsten Keil 			pr_debug("%s: B%1d EXIRB %02x\n", hx->ip->name,
1103cae86d4aSKarsten Keil 				 hx->bch.nr, exirb);
1104cae86d4aSKarsten Keil 		}
1105cae86d4aSKarsten Keil 		istab = istab & 0xF8;
1106cae86d4aSKarsten Keil 	}
1107cae86d4aSKarsten Keil 	if (exirb & IPAC_B_XDU)
1108cae86d4aSKarsten Keil 		istab |= IPACX_B_XDU;
1109cae86d4aSKarsten Keil 	if (exirb & IPAC_B_RFO)
1110cae86d4aSKarsten Keil 		istab |= IPACX_B_RFO;
1111cae86d4aSKarsten Keil 	pr_debug("%s: B%1d ISTAB %02x\n", hx->ip->name, hx->bch.nr, istab);
1112cae86d4aSKarsten Keil 
1113cae86d4aSKarsten Keil 	if (!test_bit(FLG_ACTIVE, &hx->bch.Flags))
1114cae86d4aSKarsten Keil 		return;
1115cae86d4aSKarsten Keil 
1116cae86d4aSKarsten Keil 	if (istab & IPACX_B_RME)
1117cae86d4aSKarsten Keil 		ipac_rme(hx);
1118cae86d4aSKarsten Keil 
1119cae86d4aSKarsten Keil 	if (istab & IPACX_B_RPF) {
1120cae86d4aSKarsten Keil 		hscx_empty_fifo(hx, hx->fifo_size);
1121034005a0SKarsten Keil 		if (test_bit(FLG_TRANSPARENT, &hx->bch.Flags))
1122034005a0SKarsten Keil 			recv_Bchannel(&hx->bch, 0, false);
1123cae86d4aSKarsten Keil 	}
1124cae86d4aSKarsten Keil 
1125cae86d4aSKarsten Keil 	if (istab & IPACX_B_RFO) {
1126cae86d4aSKarsten Keil 		pr_debug("%s: B%1d RFO error\n", hx->ip->name, hx->bch.nr);
1127cae86d4aSKarsten Keil 		hscx_cmdr(hx, 0x40);	/* RRES */
1128cae86d4aSKarsten Keil 	}
1129cae86d4aSKarsten Keil 
1130cae86d4aSKarsten Keil 	if (istab & IPACX_B_XPR)
1131cae86d4aSKarsten Keil 		hscx_xpr(hx);
1132cae86d4aSKarsten Keil 
1133cae86d4aSKarsten Keil 	if (istab & IPACX_B_XDU) {
1134cae86d4aSKarsten Keil 		if (test_bit(FLG_TRANSPARENT, &hx->bch.Flags)) {
11356d1ee48fSKarsten Keil 			if (test_bit(FLG_FILLEMPTY, &hx->bch.Flags))
11366d1ee48fSKarsten Keil 				test_and_set_bit(FLG_TX_EMPTY, &hx->bch.Flags);
11376d1ee48fSKarsten Keil 			hscx_xpr(hx);
1138cae86d4aSKarsten Keil 			return;
1139cae86d4aSKarsten Keil 		}
1140cae86d4aSKarsten Keil 		pr_debug("%s: B%1d XDU error at len %d\n", hx->ip->name,
1141cae86d4aSKarsten Keil 			 hx->bch.nr, hx->bch.tx_idx);
1142cae86d4aSKarsten Keil 		hx->bch.tx_idx = 0;
1143cae86d4aSKarsten Keil 		hscx_cmdr(hx, 0x01);	/* XRES */
1144cae86d4aSKarsten Keil 	}
1145cae86d4aSKarsten Keil }
1146cae86d4aSKarsten Keil 
1147cae86d4aSKarsten Keil irqreturn_t
mISDNipac_irq(struct ipac_hw * ipac,int maxloop)1148cae86d4aSKarsten Keil mISDNipac_irq(struct ipac_hw *ipac, int maxloop)
1149cae86d4aSKarsten Keil {
1150cae86d4aSKarsten Keil 	int cnt = maxloop + 1;
1151cae86d4aSKarsten Keil 	u8 ista, istad;
1152cae86d4aSKarsten Keil 	struct isac_hw  *isac = &ipac->isac;
1153cae86d4aSKarsten Keil 
1154cae86d4aSKarsten Keil 	if (ipac->type & IPAC_TYPE_IPACX) {
1155cae86d4aSKarsten Keil 		ista = ReadIPAC(ipac, ISACX_ISTA);
115640d24c4dSDan Carpenter 		while (ista && --cnt) {
1157cae86d4aSKarsten Keil 			pr_debug("%s: ISTA %02x\n", ipac->name, ista);
1158cae86d4aSKarsten Keil 			if (ista & IPACX__ICA)
1159cae86d4aSKarsten Keil 				ipac_irq(&ipac->hscx[0], ista);
1160cae86d4aSKarsten Keil 			if (ista & IPACX__ICB)
1161cae86d4aSKarsten Keil 				ipac_irq(&ipac->hscx[1], ista);
1162cae86d4aSKarsten Keil 			if (ista & (ISACX__ICD | ISACX__CIC))
1163cae86d4aSKarsten Keil 				mISDNisac_irq(&ipac->isac, ista);
1164cae86d4aSKarsten Keil 			ista = ReadIPAC(ipac, ISACX_ISTA);
1165cae86d4aSKarsten Keil 		}
1166cae86d4aSKarsten Keil 	} else if (ipac->type & IPAC_TYPE_IPAC) {
1167cae86d4aSKarsten Keil 		ista = ReadIPAC(ipac, IPAC_ISTA);
116840d24c4dSDan Carpenter 		while (ista && --cnt) {
1169cae86d4aSKarsten Keil 			pr_debug("%s: ISTA %02x\n", ipac->name, ista);
1170cae86d4aSKarsten Keil 			if (ista & (IPAC__ICD | IPAC__EXD)) {
1171cae86d4aSKarsten Keil 				istad = ReadISAC(isac, ISAC_ISTA);
1172cae86d4aSKarsten Keil 				pr_debug("%s: ISTAD %02x\n", ipac->name, istad);
1173cae86d4aSKarsten Keil 				if (istad & IPAC_D_TIN2)
1174cae86d4aSKarsten Keil 					pr_debug("%s TIN2 irq\n", ipac->name);
1175cae86d4aSKarsten Keil 				if (ista & IPAC__EXD)
1176cae86d4aSKarsten Keil 					istad |= 1; /* ISAC EXI */
1177cae86d4aSKarsten Keil 				mISDNisac_irq(isac, istad);
1178cae86d4aSKarsten Keil 			}
1179cae86d4aSKarsten Keil 			if (ista & (IPAC__ICA | IPAC__EXA))
1180cae86d4aSKarsten Keil 				ipac_irq(&ipac->hscx[0], ista);
1181cae86d4aSKarsten Keil 			if (ista & (IPAC__ICB | IPAC__EXB))
1182cae86d4aSKarsten Keil 				ipac_irq(&ipac->hscx[1], ista);
1183cae86d4aSKarsten Keil 			ista = ReadIPAC(ipac, IPAC_ISTA);
1184cae86d4aSKarsten Keil 		}
1185cae86d4aSKarsten Keil 	} else if (ipac->type & IPAC_TYPE_HSCX) {
118640d24c4dSDan Carpenter 		while (--cnt) {
1187cae86d4aSKarsten Keil 			ista = ReadIPAC(ipac, IPAC_ISTAB + ipac->hscx[1].off);
1188cae86d4aSKarsten Keil 			pr_debug("%s: B2 ISTA %02x\n", ipac->name, ista);
1189cae86d4aSKarsten Keil 			if (ista)
1190cae86d4aSKarsten Keil 				ipac_irq(&ipac->hscx[1], ista);
1191cae86d4aSKarsten Keil 			istad = ReadISAC(isac, ISAC_ISTA);
1192cae86d4aSKarsten Keil 			pr_debug("%s: ISTAD %02x\n", ipac->name, istad);
1193cae86d4aSKarsten Keil 			if (istad)
1194cae86d4aSKarsten Keil 				mISDNisac_irq(isac, istad);
1195cae86d4aSKarsten Keil 			if (0 == (ista | istad))
1196cae86d4aSKarsten Keil 				break;
1197cae86d4aSKarsten Keil 		}
1198cae86d4aSKarsten Keil 	}
1199cae86d4aSKarsten Keil 	if (cnt > maxloop) /* only for ISAC/HSCX without PCI IRQ test */
1200cae86d4aSKarsten Keil 		return IRQ_NONE;
1201cae86d4aSKarsten Keil 	if (cnt < maxloop)
1202cae86d4aSKarsten Keil 		pr_debug("%s: %d irqloops cpu%d\n", ipac->name,
1203cae86d4aSKarsten Keil 			 maxloop - cnt, smp_processor_id());
1204cae86d4aSKarsten Keil 	if (maxloop && !cnt)
1205cae86d4aSKarsten Keil 		pr_notice("%s: %d IRQ LOOP cpu%d\n", ipac->name,
1206cae86d4aSKarsten Keil 			  maxloop, smp_processor_id());
1207cae86d4aSKarsten Keil 	return IRQ_HANDLED;
1208cae86d4aSKarsten Keil }
1209cae86d4aSKarsten Keil EXPORT_SYMBOL(mISDNipac_irq);
1210cae86d4aSKarsten Keil 
1211cae86d4aSKarsten Keil static int
hscx_mode(struct hscx_hw * hscx,u32 bprotocol)1212cae86d4aSKarsten Keil hscx_mode(struct hscx_hw *hscx, u32 bprotocol)
1213cae86d4aSKarsten Keil {
1214cae86d4aSKarsten Keil 	pr_debug("%s: HSCX %c protocol %x-->%x ch %d\n", hscx->ip->name,
1215cae86d4aSKarsten Keil 		 '@' + hscx->bch.nr, hscx->bch.state, bprotocol, hscx->bch.nr);
1216cae86d4aSKarsten Keil 	if (hscx->ip->type & IPAC_TYPE_IPACX) {
1217cae86d4aSKarsten Keil 		if (hscx->bch.nr & 1) { /* B1 and ICA */
1218cae86d4aSKarsten Keil 			WriteIPAC(hscx->ip, ISACX_BCHA_TSDP_BC1, 0x80);
1219cae86d4aSKarsten Keil 			WriteIPAC(hscx->ip, ISACX_BCHA_CR, 0x88);
1220cae86d4aSKarsten Keil 		} else { /* B2 and ICB */
1221cae86d4aSKarsten Keil 			WriteIPAC(hscx->ip, ISACX_BCHB_TSDP_BC1, 0x81);
1222cae86d4aSKarsten Keil 			WriteIPAC(hscx->ip, ISACX_BCHB_CR, 0x88);
1223cae86d4aSKarsten Keil 		}
1224cae86d4aSKarsten Keil 		switch (bprotocol) {
1225cae86d4aSKarsten Keil 		case ISDN_P_NONE: /* init */
1226cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPACX_MODEB, 0xC0);	/* rec off */
1227cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPACX_EXMB,  0x30);	/* std adj. */
1228cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPACX_MASKB, 0xFF);	/* ints off */
1229cae86d4aSKarsten Keil 			hscx_cmdr(hscx, 0x41);
1230cae86d4aSKarsten Keil 			test_and_clear_bit(FLG_HDLC, &hscx->bch.Flags);
1231cae86d4aSKarsten Keil 			test_and_clear_bit(FLG_TRANSPARENT, &hscx->bch.Flags);
1232cae86d4aSKarsten Keil 			break;
1233cae86d4aSKarsten Keil 		case ISDN_P_B_RAW:
1234cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPACX_MODEB, 0x88);	/* ex trans */
1235cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPACX_EXMB,  0x00);	/* trans */
1236cae86d4aSKarsten Keil 			hscx_cmdr(hscx, 0x41);
1237cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPACX_MASKB, IPACX_B_ON);
1238cae86d4aSKarsten Keil 			test_and_set_bit(FLG_TRANSPARENT, &hscx->bch.Flags);
1239cae86d4aSKarsten Keil 			break;
1240cae86d4aSKarsten Keil 		case ISDN_P_B_HDLC:
1241cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPACX_MODEB, 0xC0);	/* trans */
1242cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPACX_EXMB,  0x00);	/* hdlc,crc */
1243cae86d4aSKarsten Keil 			hscx_cmdr(hscx, 0x41);
1244cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPACX_MASKB, IPACX_B_ON);
1245cae86d4aSKarsten Keil 			test_and_set_bit(FLG_HDLC, &hscx->bch.Flags);
1246cae86d4aSKarsten Keil 			break;
1247cae86d4aSKarsten Keil 		default:
1248cae86d4aSKarsten Keil 			pr_info("%s: protocol not known %x\n", hscx->ip->name,
1249cae86d4aSKarsten Keil 				bprotocol);
1250cae86d4aSKarsten Keil 			return -ENOPROTOOPT;
1251cae86d4aSKarsten Keil 		}
1252cae86d4aSKarsten Keil 	} else if (hscx->ip->type & IPAC_TYPE_IPAC) { /* IPAC */
1253cae86d4aSKarsten Keil 		WriteHSCX(hscx, IPAC_CCR1, 0x82);
1254cae86d4aSKarsten Keil 		WriteHSCX(hscx, IPAC_CCR2, 0x30);
1255cae86d4aSKarsten Keil 		WriteHSCX(hscx, IPAC_XCCR, 0x07);
1256cae86d4aSKarsten Keil 		WriteHSCX(hscx, IPAC_RCCR, 0x07);
1257cae86d4aSKarsten Keil 		WriteHSCX(hscx, IPAC_TSAX, hscx->slot);
1258cae86d4aSKarsten Keil 		WriteHSCX(hscx, IPAC_TSAR, hscx->slot);
1259cae86d4aSKarsten Keil 		switch (bprotocol) {
1260cae86d4aSKarsten Keil 		case ISDN_P_NONE:
1261cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_TSAX, 0x1F);
1262cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_TSAR, 0x1F);
1263cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_MODEB, 0x84);
1264cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_CCR1, 0x82);
1265cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_MASKB, 0xFF);	/* ints off */
1266cae86d4aSKarsten Keil 			test_and_clear_bit(FLG_HDLC, &hscx->bch.Flags);
1267cae86d4aSKarsten Keil 			test_and_clear_bit(FLG_TRANSPARENT, &hscx->bch.Flags);
1268cae86d4aSKarsten Keil 			break;
1269cae86d4aSKarsten Keil 		case ISDN_P_B_RAW:
1270cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_MODEB, 0xe4);	/* ex trans */
1271cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_CCR1, 0x82);
1272cae86d4aSKarsten Keil 			hscx_cmdr(hscx, 0x41);
1273cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_MASKB, 0);
1274cae86d4aSKarsten Keil 			test_and_set_bit(FLG_TRANSPARENT, &hscx->bch.Flags);
1275cae86d4aSKarsten Keil 			break;
1276cae86d4aSKarsten Keil 		case ISDN_P_B_HDLC:
1277cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_MODEB, 0x8c);
1278cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_CCR1, 0x8a);
1279cae86d4aSKarsten Keil 			hscx_cmdr(hscx, 0x41);
1280cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_MASKB, 0);
1281cae86d4aSKarsten Keil 			test_and_set_bit(FLG_HDLC, &hscx->bch.Flags);
1282cae86d4aSKarsten Keil 			break;
1283cae86d4aSKarsten Keil 		default:
1284cae86d4aSKarsten Keil 			pr_info("%s: protocol not known %x\n", hscx->ip->name,
1285cae86d4aSKarsten Keil 				bprotocol);
1286cae86d4aSKarsten Keil 			return -ENOPROTOOPT;
1287cae86d4aSKarsten Keil 		}
1288cae86d4aSKarsten Keil 	} else if (hscx->ip->type & IPAC_TYPE_HSCX) { /* HSCX */
1289cae86d4aSKarsten Keil 		WriteHSCX(hscx, IPAC_CCR1, 0x85);
1290cae86d4aSKarsten Keil 		WriteHSCX(hscx, IPAC_CCR2, 0x30);
1291cae86d4aSKarsten Keil 		WriteHSCX(hscx, IPAC_XCCR, 0x07);
1292cae86d4aSKarsten Keil 		WriteHSCX(hscx, IPAC_RCCR, 0x07);
1293cae86d4aSKarsten Keil 		WriteHSCX(hscx, IPAC_TSAX, hscx->slot);
1294cae86d4aSKarsten Keil 		WriteHSCX(hscx, IPAC_TSAR, hscx->slot);
1295cae86d4aSKarsten Keil 		switch (bprotocol) {
1296cae86d4aSKarsten Keil 		case ISDN_P_NONE:
1297cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_TSAX, 0x1F);
1298cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_TSAR, 0x1F);
1299cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_MODEB, 0x84);
1300cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_CCR1, 0x85);
1301cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_MASKB, 0xFF);	/* ints off */
1302cae86d4aSKarsten Keil 			test_and_clear_bit(FLG_HDLC, &hscx->bch.Flags);
1303cae86d4aSKarsten Keil 			test_and_clear_bit(FLG_TRANSPARENT, &hscx->bch.Flags);
1304cae86d4aSKarsten Keil 			break;
1305cae86d4aSKarsten Keil 		case ISDN_P_B_RAW:
1306cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_MODEB, 0xe4);	/* ex trans */
1307cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_CCR1, 0x85);
1308cae86d4aSKarsten Keil 			hscx_cmdr(hscx, 0x41);
1309cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_MASKB, 0);
1310cae86d4aSKarsten Keil 			test_and_set_bit(FLG_TRANSPARENT, &hscx->bch.Flags);
1311cae86d4aSKarsten Keil 			break;
1312cae86d4aSKarsten Keil 		case ISDN_P_B_HDLC:
1313cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_MODEB, 0x8c);
1314cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_CCR1, 0x8d);
1315cae86d4aSKarsten Keil 			hscx_cmdr(hscx, 0x41);
1316cae86d4aSKarsten Keil 			WriteHSCX(hscx, IPAC_MASKB, 0);
1317cae86d4aSKarsten Keil 			test_and_set_bit(FLG_HDLC, &hscx->bch.Flags);
1318cae86d4aSKarsten Keil 			break;
1319cae86d4aSKarsten Keil 		default:
1320cae86d4aSKarsten Keil 			pr_info("%s: protocol not known %x\n", hscx->ip->name,
1321cae86d4aSKarsten Keil 				bprotocol);
1322cae86d4aSKarsten Keil 			return -ENOPROTOOPT;
1323cae86d4aSKarsten Keil 		}
1324cae86d4aSKarsten Keil 	} else
1325cae86d4aSKarsten Keil 		return -EINVAL;
1326cae86d4aSKarsten Keil 	hscx->bch.state = bprotocol;
1327cae86d4aSKarsten Keil 	return 0;
1328cae86d4aSKarsten Keil }
1329cae86d4aSKarsten Keil 
1330cae86d4aSKarsten Keil static int
hscx_l2l1(struct mISDNchannel * ch,struct sk_buff * skb)1331cae86d4aSKarsten Keil hscx_l2l1(struct mISDNchannel *ch, struct sk_buff *skb)
1332cae86d4aSKarsten Keil {
1333cae86d4aSKarsten Keil 	struct bchannel *bch = container_of(ch, struct bchannel, ch);
1334cae86d4aSKarsten Keil 	struct hscx_hw	*hx = container_of(bch, struct hscx_hw, bch);
1335cae86d4aSKarsten Keil 	int ret = -EINVAL;
1336cae86d4aSKarsten Keil 	struct mISDNhead *hh = mISDN_HEAD_P(skb);
13378bfddfbeSKarsten Keil 	unsigned long flags;
1338cae86d4aSKarsten Keil 
1339cae86d4aSKarsten Keil 	switch (hh->prim) {
1340cae86d4aSKarsten Keil 	case PH_DATA_REQ:
1341cae86d4aSKarsten Keil 		spin_lock_irqsave(hx->ip->hwlock, flags);
1342cae86d4aSKarsten Keil 		ret = bchannel_senddata(bch, skb);
1343cae86d4aSKarsten Keil 		if (ret > 0) { /* direct TX */
1344cae86d4aSKarsten Keil 			ret = 0;
1345cae86d4aSKarsten Keil 			hscx_fill_fifo(hx);
13468bfddfbeSKarsten Keil 		}
1347cae86d4aSKarsten Keil 		spin_unlock_irqrestore(hx->ip->hwlock, flags);
1348cae86d4aSKarsten Keil 		return ret;
1349cae86d4aSKarsten Keil 	case PH_ACTIVATE_REQ:
1350cae86d4aSKarsten Keil 		spin_lock_irqsave(hx->ip->hwlock, flags);
1351cae86d4aSKarsten Keil 		if (!test_and_set_bit(FLG_ACTIVE, &bch->Flags))
1352cae86d4aSKarsten Keil 			ret = hscx_mode(hx, ch->protocol);
1353cae86d4aSKarsten Keil 		else
1354cae86d4aSKarsten Keil 			ret = 0;
1355cae86d4aSKarsten Keil 		spin_unlock_irqrestore(hx->ip->hwlock, flags);
1356cae86d4aSKarsten Keil 		if (!ret)
1357cae86d4aSKarsten Keil 			_queue_data(ch, PH_ACTIVATE_IND, MISDN_ID_ANY, 0,
1358cae86d4aSKarsten Keil 				    NULL, GFP_KERNEL);
1359cae86d4aSKarsten Keil 		break;
1360cae86d4aSKarsten Keil 	case PH_DEACTIVATE_REQ:
1361cae86d4aSKarsten Keil 		spin_lock_irqsave(hx->ip->hwlock, flags);
1362cae86d4aSKarsten Keil 		mISDN_clear_bchannel(bch);
1363cae86d4aSKarsten Keil 		hscx_mode(hx, ISDN_P_NONE);
1364cae86d4aSKarsten Keil 		spin_unlock_irqrestore(hx->ip->hwlock, flags);
1365cae86d4aSKarsten Keil 		_queue_data(ch, PH_DEACTIVATE_IND, MISDN_ID_ANY, 0,
1366cae86d4aSKarsten Keil 			    NULL, GFP_KERNEL);
1367cae86d4aSKarsten Keil 		ret = 0;
1368cae86d4aSKarsten Keil 		break;
1369cae86d4aSKarsten Keil 	default:
1370cae86d4aSKarsten Keil 		pr_info("%s: %s unknown prim(%x,%x)\n",
1371cae86d4aSKarsten Keil 			hx->ip->name, __func__, hh->prim, hh->id);
1372cae86d4aSKarsten Keil 		ret = -EINVAL;
1373cae86d4aSKarsten Keil 	}
1374cae86d4aSKarsten Keil 	if (!ret)
1375cae86d4aSKarsten Keil 		dev_kfree_skb(skb);
1376cae86d4aSKarsten Keil 	return ret;
1377cae86d4aSKarsten Keil }
1378cae86d4aSKarsten Keil 
1379cae86d4aSKarsten Keil static int
channel_bctrl(struct bchannel * bch,struct mISDN_ctrl_req * cq)1380cae86d4aSKarsten Keil channel_bctrl(struct bchannel *bch, struct mISDN_ctrl_req *cq)
1381cae86d4aSKarsten Keil {
1382034005a0SKarsten Keil 	return mISDN_ctrl_bchannel(bch, cq);
1383cae86d4aSKarsten Keil }
1384cae86d4aSKarsten Keil 
1385cae86d4aSKarsten Keil static int
hscx_bctrl(struct mISDNchannel * ch,u32 cmd,void * arg)1386cae86d4aSKarsten Keil hscx_bctrl(struct mISDNchannel *ch, u32 cmd, void *arg)
1387cae86d4aSKarsten Keil {
1388cae86d4aSKarsten Keil 	struct bchannel *bch = container_of(ch, struct bchannel, ch);
1389cae86d4aSKarsten Keil 	struct hscx_hw	*hx = container_of(bch, struct hscx_hw, bch);
1390cae86d4aSKarsten Keil 	int ret = -EINVAL;
1391cae86d4aSKarsten Keil 	u_long flags;
1392cae86d4aSKarsten Keil 
1393cae86d4aSKarsten Keil 	pr_debug("%s: %s cmd:%x %p\n", hx->ip->name, __func__, cmd, arg);
1394cae86d4aSKarsten Keil 	switch (cmd) {
1395cae86d4aSKarsten Keil 	case CLOSE_CHANNEL:
1396cae86d4aSKarsten Keil 		test_and_clear_bit(FLG_OPEN, &bch->Flags);
13974b921edaSKarsten Keil 		cancel_work_sync(&bch->workq);
1398cae86d4aSKarsten Keil 		spin_lock_irqsave(hx->ip->hwlock, flags);
13994b921edaSKarsten Keil 		mISDN_clear_bchannel(bch);
1400cae86d4aSKarsten Keil 		hscx_mode(hx, ISDN_P_NONE);
1401cae86d4aSKarsten Keil 		spin_unlock_irqrestore(hx->ip->hwlock, flags);
1402cae86d4aSKarsten Keil 		ch->protocol = ISDN_P_NONE;
1403cae86d4aSKarsten Keil 		ch->peer = NULL;
1404cae86d4aSKarsten Keil 		module_put(hx->ip->owner);
1405cae86d4aSKarsten Keil 		ret = 0;
1406cae86d4aSKarsten Keil 		break;
1407cae86d4aSKarsten Keil 	case CONTROL_CHANNEL:
1408cae86d4aSKarsten Keil 		ret = channel_bctrl(bch, arg);
1409cae86d4aSKarsten Keil 		break;
1410cae86d4aSKarsten Keil 	default:
1411cae86d4aSKarsten Keil 		pr_info("%s: %s unknown prim(%x)\n",
1412cae86d4aSKarsten Keil 			hx->ip->name, __func__, cmd);
1413cae86d4aSKarsten Keil 	}
1414cae86d4aSKarsten Keil 	return ret;
1415cae86d4aSKarsten Keil }
1416cae86d4aSKarsten Keil 
1417cae86d4aSKarsten Keil static void
free_ipac(struct ipac_hw * ipac)1418cae86d4aSKarsten Keil free_ipac(struct ipac_hw *ipac)
1419cae86d4aSKarsten Keil {
1420cae86d4aSKarsten Keil 	isac_release(&ipac->isac);
1421cae86d4aSKarsten Keil }
1422cae86d4aSKarsten Keil 
1423cae86d4aSKarsten Keil static const char *HSCXVer[] =
1424cae86d4aSKarsten Keil {"A1", "?1", "A2", "?3", "A3", "V2.1", "?6", "?7",
1425cae86d4aSKarsten Keil  "?8", "?9", "?10", "?11", "?12", "?13", "?14", "???"};
1426cae86d4aSKarsten Keil 
1427cae86d4aSKarsten Keil 
1428cae86d4aSKarsten Keil 
1429cae86d4aSKarsten Keil static void
hscx_init(struct hscx_hw * hx)1430cae86d4aSKarsten Keil hscx_init(struct hscx_hw *hx)
1431cae86d4aSKarsten Keil {
1432cae86d4aSKarsten Keil 	u8 val;
1433cae86d4aSKarsten Keil 
1434cae86d4aSKarsten Keil 	WriteHSCX(hx, IPAC_RAH2, 0xFF);
1435cae86d4aSKarsten Keil 	WriteHSCX(hx, IPAC_XBCH, 0x00);
1436cae86d4aSKarsten Keil 	WriteHSCX(hx, IPAC_RLCR, 0x00);
1437cae86d4aSKarsten Keil 
1438cae86d4aSKarsten Keil 	if (hx->ip->type & IPAC_TYPE_HSCX) {
1439cae86d4aSKarsten Keil 		WriteHSCX(hx, IPAC_CCR1, 0x85);
1440cae86d4aSKarsten Keil 		val = ReadHSCX(hx, HSCX_VSTR);
1441cae86d4aSKarsten Keil 		pr_debug("%s: HSCX VSTR %02x\n", hx->ip->name, val);
1442cae86d4aSKarsten Keil 		if (hx->bch.debug & DEBUG_HW)
1443cae86d4aSKarsten Keil 			pr_notice("%s: HSCX version %s\n", hx->ip->name,
1444cae86d4aSKarsten Keil 				  HSCXVer[val & 0x0f]);
1445cae86d4aSKarsten Keil 	} else
1446cae86d4aSKarsten Keil 		WriteHSCX(hx, IPAC_CCR1, 0x82);
1447cae86d4aSKarsten Keil 	WriteHSCX(hx, IPAC_CCR2, 0x30);
1448cae86d4aSKarsten Keil 	WriteHSCX(hx, IPAC_XCCR, 0x07);
1449cae86d4aSKarsten Keil 	WriteHSCX(hx, IPAC_RCCR, 0x07);
1450cae86d4aSKarsten Keil }
1451cae86d4aSKarsten Keil 
1452cae86d4aSKarsten Keil static int
ipac_init(struct ipac_hw * ipac)1453cae86d4aSKarsten Keil ipac_init(struct ipac_hw *ipac)
1454cae86d4aSKarsten Keil {
1455cae86d4aSKarsten Keil 	u8 val;
1456cae86d4aSKarsten Keil 
1457cae86d4aSKarsten Keil 	if (ipac->type & IPAC_TYPE_HSCX) {
1458cae86d4aSKarsten Keil 		hscx_init(&ipac->hscx[0]);
1459cae86d4aSKarsten Keil 		hscx_init(&ipac->hscx[1]);
1460cae86d4aSKarsten Keil 		val = ReadIPAC(ipac, IPAC_ID);
1461cae86d4aSKarsten Keil 	} else if (ipac->type & IPAC_TYPE_IPAC) {
1462cae86d4aSKarsten Keil 		hscx_init(&ipac->hscx[0]);
1463cae86d4aSKarsten Keil 		hscx_init(&ipac->hscx[1]);
1464cae86d4aSKarsten Keil 		WriteIPAC(ipac, IPAC_MASK, IPAC__ON);
1465cae86d4aSKarsten Keil 		val = ReadIPAC(ipac, IPAC_CONF);
1466cae86d4aSKarsten Keil 		/* conf is default 0, but can be overwritten by card setup */
1467cae86d4aSKarsten Keil 		pr_debug("%s: IPAC CONF %02x/%02x\n", ipac->name,
1468cae86d4aSKarsten Keil 			 val, ipac->conf);
1469cae86d4aSKarsten Keil 		WriteIPAC(ipac, IPAC_CONF, ipac->conf);
1470cae86d4aSKarsten Keil 		val = ReadIPAC(ipac, IPAC_ID);
1471cae86d4aSKarsten Keil 		if (ipac->hscx[0].bch.debug & DEBUG_HW)
1472cae86d4aSKarsten Keil 			pr_notice("%s: IPAC Design ID %02x\n", ipac->name, val);
1473cae86d4aSKarsten Keil 	}
1474cae86d4aSKarsten Keil 	/* nothing special for IPACX to do here */
1475cae86d4aSKarsten Keil 	return isac_init(&ipac->isac);
1476cae86d4aSKarsten Keil }
1477cae86d4aSKarsten Keil 
1478cae86d4aSKarsten Keil static int
open_bchannel(struct ipac_hw * ipac,struct channel_req * rq)1479cae86d4aSKarsten Keil open_bchannel(struct ipac_hw *ipac, struct channel_req *rq)
1480cae86d4aSKarsten Keil {
1481cae86d4aSKarsten Keil 	struct bchannel		*bch;
1482cae86d4aSKarsten Keil 
1483819a1008SDan Carpenter 	if (rq->adr.channel == 0 || rq->adr.channel > 2)
1484cae86d4aSKarsten Keil 		return -EINVAL;
1485cae86d4aSKarsten Keil 	if (rq->protocol == ISDN_P_NONE)
1486cae86d4aSKarsten Keil 		return -EINVAL;
1487cae86d4aSKarsten Keil 	bch = &ipac->hscx[rq->adr.channel - 1].bch;
1488cae86d4aSKarsten Keil 	if (test_and_set_bit(FLG_OPEN, &bch->Flags))
1489cae86d4aSKarsten Keil 		return -EBUSY; /* b-channel can be only open once */
1490cae86d4aSKarsten Keil 	test_and_clear_bit(FLG_FILLEMPTY, &bch->Flags);
1491cae86d4aSKarsten Keil 	bch->ch.protocol = rq->protocol;
1492cae86d4aSKarsten Keil 	rq->ch = &bch->ch;
1493cae86d4aSKarsten Keil 	return 0;
1494cae86d4aSKarsten Keil }
1495cae86d4aSKarsten Keil 
1496cae86d4aSKarsten Keil static int
channel_ctrl(struct ipac_hw * ipac,struct mISDN_ctrl_req * cq)1497cae86d4aSKarsten Keil channel_ctrl(struct ipac_hw *ipac, struct mISDN_ctrl_req *cq)
1498cae86d4aSKarsten Keil {
1499cae86d4aSKarsten Keil 	int	ret = 0;
1500cae86d4aSKarsten Keil 
1501cae86d4aSKarsten Keil 	switch (cq->op) {
1502cae86d4aSKarsten Keil 	case MISDN_CTRL_GETOP:
1503c626c127SKarsten Keil 		cq->op = MISDN_CTRL_LOOP | MISDN_CTRL_L1_TIMER3;
1504cae86d4aSKarsten Keil 		break;
1505cae86d4aSKarsten Keil 	case MISDN_CTRL_LOOP:
1506cae86d4aSKarsten Keil 		/* cq->channel: 0 disable, 1 B1 loop 2 B2 loop, 3 both */
1507cae86d4aSKarsten Keil 		if (cq->channel < 0 || cq->channel > 3) {
1508cae86d4aSKarsten Keil 			ret = -EINVAL;
1509cae86d4aSKarsten Keil 			break;
1510cae86d4aSKarsten Keil 		}
1511cae86d4aSKarsten Keil 		ret = ipac->ctrl(ipac, HW_TESTLOOP, cq->channel);
1512cae86d4aSKarsten Keil 		break;
1513c626c127SKarsten Keil 	case MISDN_CTRL_L1_TIMER3:
1514c626c127SKarsten Keil 		ret = ipac->isac.ctrl(&ipac->isac, HW_TIMER3_VALUE, cq->p1);
1515c626c127SKarsten Keil 		break;
1516cae86d4aSKarsten Keil 	default:
1517cae86d4aSKarsten Keil 		pr_info("%s: unknown CTRL OP %x\n", ipac->name, cq->op);
1518cae86d4aSKarsten Keil 		ret = -EINVAL;
1519cae86d4aSKarsten Keil 		break;
1520cae86d4aSKarsten Keil 	}
1521cae86d4aSKarsten Keil 	return ret;
1522cae86d4aSKarsten Keil }
1523cae86d4aSKarsten Keil 
1524cae86d4aSKarsten Keil static int
ipac_dctrl(struct mISDNchannel * ch,u32 cmd,void * arg)1525cae86d4aSKarsten Keil ipac_dctrl(struct mISDNchannel *ch, u32 cmd, void *arg)
1526cae86d4aSKarsten Keil {
1527cae86d4aSKarsten Keil 	struct mISDNdevice *dev = container_of(ch, struct mISDNdevice, D);
1528cae86d4aSKarsten Keil 	struct dchannel *dch = container_of(dev, struct dchannel, dev);
1529cae86d4aSKarsten Keil 	struct isac_hw *isac = container_of(dch, struct isac_hw, dch);
1530cae86d4aSKarsten Keil 	struct ipac_hw *ipac = container_of(isac, struct ipac_hw, isac);
1531cae86d4aSKarsten Keil 	struct channel_req *rq;
1532cae86d4aSKarsten Keil 	int err = 0;
1533cae86d4aSKarsten Keil 
1534cae86d4aSKarsten Keil 	pr_debug("%s: DCTRL: %x %p\n", ipac->name, cmd, arg);
1535cae86d4aSKarsten Keil 	switch (cmd) {
1536cae86d4aSKarsten Keil 	case OPEN_CHANNEL:
1537cae86d4aSKarsten Keil 		rq = arg;
1538cae86d4aSKarsten Keil 		if (rq->protocol == ISDN_P_TE_S0)
15393e7a8716SArnd Bergmann 			err = open_dchannel_caller(isac, rq, __builtin_return_address(0));
1540cae86d4aSKarsten Keil 		else
1541cae86d4aSKarsten Keil 			err = open_bchannel(ipac, rq);
1542cae86d4aSKarsten Keil 		if (err)
1543cae86d4aSKarsten Keil 			break;
1544cae86d4aSKarsten Keil 		if (!try_module_get(ipac->owner))
1545cae86d4aSKarsten Keil 			pr_info("%s: cannot get module\n", ipac->name);
1546cae86d4aSKarsten Keil 		break;
1547cae86d4aSKarsten Keil 	case CLOSE_CHANNEL:
1548cae86d4aSKarsten Keil 		pr_debug("%s: dev(%d) close from %p\n", ipac->name,
1549cae86d4aSKarsten Keil 			 dch->dev.id, __builtin_return_address(0));
1550cae86d4aSKarsten Keil 		module_put(ipac->owner);
1551cae86d4aSKarsten Keil 		break;
1552cae86d4aSKarsten Keil 	case CONTROL_CHANNEL:
1553cae86d4aSKarsten Keil 		err = channel_ctrl(ipac, arg);
1554cae86d4aSKarsten Keil 		break;
1555cae86d4aSKarsten Keil 	default:
1556cae86d4aSKarsten Keil 		pr_debug("%s: unknown DCTRL command %x\n", ipac->name, cmd);
1557cae86d4aSKarsten Keil 		return -EINVAL;
1558cae86d4aSKarsten Keil 	}
1559cae86d4aSKarsten Keil 	return err;
1560cae86d4aSKarsten Keil }
1561cae86d4aSKarsten Keil 
1562cae86d4aSKarsten Keil u32
mISDNipac_init(struct ipac_hw * ipac,void * hw)1563cae86d4aSKarsten Keil mISDNipac_init(struct ipac_hw *ipac, void *hw)
1564cae86d4aSKarsten Keil {
1565cae86d4aSKarsten Keil 	u32 ret;
1566cae86d4aSKarsten Keil 	u8 i;
1567cae86d4aSKarsten Keil 
1568cae86d4aSKarsten Keil 	ipac->hw = hw;
1569cae86d4aSKarsten Keil 	if (ipac->isac.dch.debug & DEBUG_HW)
1570cae86d4aSKarsten Keil 		pr_notice("%s: ipac type %x\n", ipac->name, ipac->type);
1571cae86d4aSKarsten Keil 	if (ipac->type & IPAC_TYPE_HSCX) {
1572cae86d4aSKarsten Keil 		ipac->isac.type = IPAC_TYPE_ISAC;
1573cae86d4aSKarsten Keil 		ipac->hscx[0].off = 0;
1574cae86d4aSKarsten Keil 		ipac->hscx[1].off = 0x40;
1575cae86d4aSKarsten Keil 		ipac->hscx[0].fifo_size = 32;
1576cae86d4aSKarsten Keil 		ipac->hscx[1].fifo_size = 32;
1577cae86d4aSKarsten Keil 	} else if (ipac->type & IPAC_TYPE_IPAC) {
1578cae86d4aSKarsten Keil 		ipac->isac.type = IPAC_TYPE_IPAC | IPAC_TYPE_ISAC;
1579cae86d4aSKarsten Keil 		ipac->hscx[0].off = 0;
1580cae86d4aSKarsten Keil 		ipac->hscx[1].off = 0x40;
1581cae86d4aSKarsten Keil 		ipac->hscx[0].fifo_size = 64;
1582cae86d4aSKarsten Keil 		ipac->hscx[1].fifo_size = 64;
1583cae86d4aSKarsten Keil 	} else if (ipac->type & IPAC_TYPE_IPACX) {
1584cae86d4aSKarsten Keil 		ipac->isac.type = IPAC_TYPE_IPACX | IPAC_TYPE_ISACX;
1585cae86d4aSKarsten Keil 		ipac->hscx[0].off = IPACX_OFF_ICA;
1586cae86d4aSKarsten Keil 		ipac->hscx[1].off = IPACX_OFF_ICB;
1587cae86d4aSKarsten Keil 		ipac->hscx[0].fifo_size = 64;
1588cae86d4aSKarsten Keil 		ipac->hscx[1].fifo_size = 64;
1589cae86d4aSKarsten Keil 	} else
1590cae86d4aSKarsten Keil 		return 0;
1591cae86d4aSKarsten Keil 
1592cae86d4aSKarsten Keil 	mISDNisac_init(&ipac->isac, hw);
1593cae86d4aSKarsten Keil 
1594cae86d4aSKarsten Keil 	ipac->isac.dch.dev.D.ctrl = ipac_dctrl;
1595cae86d4aSKarsten Keil 
1596cae86d4aSKarsten Keil 	for (i = 0; i < 2; i++) {
1597cae86d4aSKarsten Keil 		ipac->hscx[i].bch.nr = i + 1;
1598cae86d4aSKarsten Keil 		set_channelmap(i + 1, ipac->isac.dch.dev.channelmap);
1599cae86d4aSKarsten Keil 		list_add(&ipac->hscx[i].bch.ch.list,
1600cae86d4aSKarsten Keil 			 &ipac->isac.dch.dev.bchannels);
1601034005a0SKarsten Keil 		mISDN_initbchannel(&ipac->hscx[i].bch, MAX_DATA_MEM,
1602034005a0SKarsten Keil 				   ipac->hscx[i].fifo_size);
1603cae86d4aSKarsten Keil 		ipac->hscx[i].bch.ch.nr = i + 1;
1604cae86d4aSKarsten Keil 		ipac->hscx[i].bch.ch.send = &hscx_l2l1;
1605cae86d4aSKarsten Keil 		ipac->hscx[i].bch.ch.ctrl = hscx_bctrl;
1606cae86d4aSKarsten Keil 		ipac->hscx[i].bch.hw = hw;
1607cae86d4aSKarsten Keil 		ipac->hscx[i].ip = ipac;
1608cae86d4aSKarsten Keil 		/* default values for IOM time slots
1609b8a14f33SMasahiro Yamada 		 * can be overwritten by card */
1610cae86d4aSKarsten Keil 		ipac->hscx[i].slot = (i == 0) ? 0x2f : 0x03;
1611cae86d4aSKarsten Keil 	}
1612cae86d4aSKarsten Keil 
1613cae86d4aSKarsten Keil 	ipac->init = ipac_init;
1614cae86d4aSKarsten Keil 	ipac->release = free_ipac;
1615cae86d4aSKarsten Keil 
1616cae86d4aSKarsten Keil 	ret =	(1 << (ISDN_P_B_RAW & ISDN_P_B_MASK)) |
1617cae86d4aSKarsten Keil 		(1 << (ISDN_P_B_HDLC & ISDN_P_B_MASK));
1618cae86d4aSKarsten Keil 	return ret;
1619cae86d4aSKarsten Keil }
1620cae86d4aSKarsten Keil EXPORT_SYMBOL(mISDNipac_init);
1621cae86d4aSKarsten Keil 
1622cae86d4aSKarsten Keil static int __init
isac_mod_init(void)1623cae86d4aSKarsten Keil isac_mod_init(void)
1624cae86d4aSKarsten Keil {
1625cae86d4aSKarsten Keil 	pr_notice("mISDNipac module version %s\n", ISAC_REV);
1626cae86d4aSKarsten Keil 	return 0;
1627cae86d4aSKarsten Keil }
1628cae86d4aSKarsten Keil 
1629cae86d4aSKarsten Keil static void __exit
isac_mod_cleanup(void)1630cae86d4aSKarsten Keil isac_mod_cleanup(void)
1631cae86d4aSKarsten Keil {
1632cae86d4aSKarsten Keil 	pr_notice("mISDNipac module unloaded\n");
1633cae86d4aSKarsten Keil }
1634cae86d4aSKarsten Keil module_init(isac_mod_init);
1635cae86d4aSKarsten Keil module_exit(isac_mod_cleanup);
1636