2019-05-27 13:55:02 +07:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2009-01-09 22:20:51 +07:00
|
|
|
/* hfcsusb.c
|
|
|
|
* mISDN driver for Colognechip HFC-S USB chip
|
|
|
|
*
|
|
|
|
* Copyright 2001 by Peter Sprenger (sprenger@moving-bytes.de)
|
|
|
|
* Copyright 2008 by Martin Bachem (info@bachem-it.com)
|
|
|
|
*
|
|
|
|
* module params
|
|
|
|
* debug=<n>, default=0, with n=0xHHHHGGGG
|
|
|
|
* H - l1 driver flags described in hfcsusb.h
|
|
|
|
* G - common mISDN debug flags described at mISDNhw.h
|
|
|
|
*
|
|
|
|
* poll=<n>, default 128
|
|
|
|
* n : burst size of PH_DATA_IND at transparent rx data
|
|
|
|
*
|
2012-02-10 12:01:08 +07:00
|
|
|
* Revision: 0.3.3 (socket), 2008-11-05
|
2009-01-09 22:20:51 +07:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/usb.h>
|
|
|
|
#include <linux/mISDNhw.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 15:04:11 +07:00
|
|
|
#include <linux/slab.h>
|
2009-01-09 22:20:51 +07:00
|
|
|
#include "hfcsusb.h"
|
|
|
|
|
|
|
|
static unsigned int debug;
|
|
|
|
static int poll = DEFAULT_TRANSP_BURST_SZ;
|
|
|
|
|
|
|
|
static LIST_HEAD(HFClist);
|
|
|
|
static DEFINE_RWLOCK(HFClock);
|
|
|
|
|
|
|
|
|
|
|
|
MODULE_AUTHOR("Martin Bachem");
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
module_param(debug, uint, S_IRUGO | S_IWUSR);
|
|
|
|
module_param(poll, int, 0);
|
|
|
|
|
|
|
|
static int hfcsusb_cnt;
|
|
|
|
|
|
|
|
/* some function prototypes */
|
|
|
|
static void hfcsusb_ph_command(struct hfcsusb *hw, u_char command);
|
|
|
|
static void release_hw(struct hfcsusb *hw);
|
|
|
|
static void reset_hfcsusb(struct hfcsusb *hw);
|
|
|
|
static void setPortMode(struct hfcsusb *hw);
|
|
|
|
static void hfcsusb_start_endpoint(struct hfcsusb *hw, int channel);
|
|
|
|
static void hfcsusb_stop_endpoint(struct hfcsusb *hw, int channel);
|
|
|
|
static int hfcsusb_setup_bch(struct bchannel *bch, int protocol);
|
|
|
|
static void deactivate_bchannel(struct bchannel *bch);
|
|
|
|
static void hfcsusb_ph_info(struct hfcsusb *hw);
|
|
|
|
|
|
|
|
/* start next background transfer for control channel */
|
|
|
|
static void
|
|
|
|
ctrl_start_transfer(struct hfcsusb *hw)
|
|
|
|
{
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s\n", hw->name, __func__);
|
|
|
|
|
|
|
|
if (hw->ctrl_cnt) {
|
|
|
|
hw->ctrl_urb->pipe = hw->ctrl_out_pipe;
|
|
|
|
hw->ctrl_urb->setup_packet = (u_char *)&hw->ctrl_write;
|
|
|
|
hw->ctrl_urb->transfer_buffer = NULL;
|
|
|
|
hw->ctrl_urb->transfer_buffer_length = 0;
|
|
|
|
hw->ctrl_write.wIndex =
|
2012-02-20 10:52:38 +07:00
|
|
|
cpu_to_le16(hw->ctrl_buff[hw->ctrl_out_idx].hfcs_reg);
|
2009-01-09 22:20:51 +07:00
|
|
|
hw->ctrl_write.wValue =
|
2012-02-20 10:52:38 +07:00
|
|
|
cpu_to_le16(hw->ctrl_buff[hw->ctrl_out_idx].reg_val);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
usb_submit_urb(hw->ctrl_urb, GFP_ATOMIC);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* queue a control transfer request to write HFC-S USB
|
|
|
|
* chip register using CTRL resuest queue
|
|
|
|
*/
|
|
|
|
static int write_reg(struct hfcsusb *hw, __u8 reg, __u8 val)
|
|
|
|
{
|
|
|
|
struct ctrl_buf *buf;
|
|
|
|
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s reg(0x%02x) val(0x%02x)\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, reg, val);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
spin_lock(&hw->ctrl_lock);
|
2010-05-26 12:55:10 +07:00
|
|
|
if (hw->ctrl_cnt >= HFC_CTRL_BUFSIZE) {
|
|
|
|
spin_unlock(&hw->ctrl_lock);
|
2009-01-09 22:20:51 +07:00
|
|
|
return 1;
|
2010-05-26 12:55:10 +07:00
|
|
|
}
|
2009-01-09 22:20:51 +07:00
|
|
|
buf = &hw->ctrl_buff[hw->ctrl_in_idx];
|
|
|
|
buf->hfcs_reg = reg;
|
|
|
|
buf->reg_val = val;
|
|
|
|
if (++hw->ctrl_in_idx >= HFC_CTRL_BUFSIZE)
|
|
|
|
hw->ctrl_in_idx = 0;
|
|
|
|
if (++hw->ctrl_cnt == 1)
|
|
|
|
ctrl_start_transfer(hw);
|
|
|
|
spin_unlock(&hw->ctrl_lock);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* control completion routine handling background control cmds */
|
|
|
|
static void
|
|
|
|
ctrl_complete(struct urb *urb)
|
|
|
|
{
|
|
|
|
struct hfcsusb *hw = (struct hfcsusb *) urb->context;
|
|
|
|
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s\n", hw->name, __func__);
|
|
|
|
|
|
|
|
urb->dev = hw->dev;
|
|
|
|
if (hw->ctrl_cnt) {
|
|
|
|
hw->ctrl_cnt--; /* decrement actual count */
|
|
|
|
if (++hw->ctrl_out_idx >= HFC_CTRL_BUFSIZE)
|
|
|
|
hw->ctrl_out_idx = 0; /* pointer wrap */
|
|
|
|
|
|
|
|
ctrl_start_transfer(hw); /* start next transfer */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* handle LED bits */
|
|
|
|
static void
|
|
|
|
set_led_bit(struct hfcsusb *hw, signed short led_bits, int set_on)
|
|
|
|
{
|
|
|
|
if (set_on) {
|
|
|
|
if (led_bits < 0)
|
|
|
|
hw->led_state &= ~abs(led_bits);
|
|
|
|
else
|
|
|
|
hw->led_state |= led_bits;
|
|
|
|
} else {
|
|
|
|
if (led_bits < 0)
|
|
|
|
hw->led_state |= abs(led_bits);
|
|
|
|
else
|
|
|
|
hw->led_state &= ~led_bits;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* handle LED requests */
|
|
|
|
static void
|
|
|
|
handle_led(struct hfcsusb *hw, int event)
|
|
|
|
{
|
|
|
|
struct hfcsusb_vdata *driver_info = (struct hfcsusb_vdata *)
|
|
|
|
hfcsusb_idtab[hw->vend_idx].driver_info;
|
|
|
|
__u8 tmpled;
|
|
|
|
|
|
|
|
if (driver_info->led_scheme == LED_OFF)
|
|
|
|
return;
|
|
|
|
tmpled = hw->led_state;
|
|
|
|
|
|
|
|
switch (event) {
|
|
|
|
case LED_POWER_ON:
|
|
|
|
set_led_bit(hw, driver_info->led_bits[0], 1);
|
|
|
|
set_led_bit(hw, driver_info->led_bits[1], 0);
|
|
|
|
set_led_bit(hw, driver_info->led_bits[2], 0);
|
|
|
|
set_led_bit(hw, driver_info->led_bits[3], 0);
|
|
|
|
break;
|
|
|
|
case LED_POWER_OFF:
|
|
|
|
set_led_bit(hw, driver_info->led_bits[0], 0);
|
|
|
|
set_led_bit(hw, driver_info->led_bits[1], 0);
|
|
|
|
set_led_bit(hw, driver_info->led_bits[2], 0);
|
|
|
|
set_led_bit(hw, driver_info->led_bits[3], 0);
|
|
|
|
break;
|
|
|
|
case LED_S0_ON:
|
|
|
|
set_led_bit(hw, driver_info->led_bits[1], 1);
|
|
|
|
break;
|
|
|
|
case LED_S0_OFF:
|
|
|
|
set_led_bit(hw, driver_info->led_bits[1], 0);
|
|
|
|
break;
|
|
|
|
case LED_B1_ON:
|
|
|
|
set_led_bit(hw, driver_info->led_bits[2], 1);
|
|
|
|
break;
|
|
|
|
case LED_B1_OFF:
|
|
|
|
set_led_bit(hw, driver_info->led_bits[2], 0);
|
|
|
|
break;
|
|
|
|
case LED_B2_ON:
|
|
|
|
set_led_bit(hw, driver_info->led_bits[3], 1);
|
|
|
|
break;
|
|
|
|
case LED_B2_OFF:
|
|
|
|
set_led_bit(hw, driver_info->led_bits[3], 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hw->led_state != tmpled) {
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s reg(0x%02x) val(x%02x)\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__,
|
|
|
|
HFCUSB_P_DATA, hw->led_state);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
write_reg(hw, HFCUSB_P_DATA, hw->led_state);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Layer2 -> Layer 1 Bchannel data
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
hfcusb_l2l1B(struct mISDNchannel *ch, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct bchannel *bch = container_of(ch, struct bchannel, ch);
|
|
|
|
struct hfcsusb *hw = bch->hw;
|
|
|
|
int ret = -EINVAL;
|
|
|
|
struct mISDNhead *hh = mISDN_HEAD_P(skb);
|
|
|
|
u_long flags;
|
|
|
|
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s\n", hw->name, __func__);
|
|
|
|
|
|
|
|
switch (hh->prim) {
|
|
|
|
case PH_DATA_REQ:
|
|
|
|
spin_lock_irqsave(&hw->lock, flags);
|
|
|
|
ret = bchannel_senddata(bch, skb);
|
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s PH_DATA_REQ ret(%i)\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, ret);
|
2012-05-16 06:51:02 +07:00
|
|
|
if (ret > 0)
|
2009-01-09 22:20:51 +07:00
|
|
|
ret = 0;
|
|
|
|
return ret;
|
|
|
|
case PH_ACTIVATE_REQ:
|
|
|
|
if (!test_and_set_bit(FLG_ACTIVE, &bch->Flags)) {
|
2012-05-16 06:51:04 +07:00
|
|
|
hfcsusb_start_endpoint(hw, bch->nr - 1);
|
2009-01-09 22:20:51 +07:00
|
|
|
ret = hfcsusb_setup_bch(bch, ch->protocol);
|
|
|
|
} else
|
|
|
|
ret = 0;
|
|
|
|
if (!ret)
|
|
|
|
_queue_data(ch, PH_ACTIVATE_IND, MISDN_ID_ANY,
|
2012-02-20 10:52:38 +07:00
|
|
|
0, NULL, GFP_KERNEL);
|
2009-01-09 22:20:51 +07:00
|
|
|
break;
|
|
|
|
case PH_DEACTIVATE_REQ:
|
|
|
|
deactivate_bchannel(bch);
|
|
|
|
_queue_data(ch, PH_DEACTIVATE_IND, MISDN_ID_ANY,
|
2012-02-20 10:52:38 +07:00
|
|
|
0, NULL, GFP_KERNEL);
|
2009-01-09 22:20:51 +07:00
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!ret)
|
|
|
|
dev_kfree_skb(skb);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* send full D/B channel status information
|
|
|
|
* as MPH_INFORMATION_IND
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
hfcsusb_ph_info(struct hfcsusb *hw)
|
|
|
|
{
|
|
|
|
struct ph_info *phi;
|
|
|
|
struct dchannel *dch = &hw->dch;
|
|
|
|
int i;
|
|
|
|
|
2019-01-09 04:27:05 +07:00
|
|
|
phi = kzalloc(struct_size(phi, bch, dch->dev.nrbchan), GFP_ATOMIC);
|
2019-03-03 04:20:43 +07:00
|
|
|
if (!phi)
|
|
|
|
return;
|
|
|
|
|
2009-01-09 22:20:51 +07:00
|
|
|
phi->dch.ch.protocol = hw->protocol;
|
|
|
|
phi->dch.ch.Flags = dch->Flags;
|
|
|
|
phi->dch.state = dch->state;
|
|
|
|
phi->dch.num_bch = dch->dev.nrbchan;
|
|
|
|
for (i = 0; i < dch->dev.nrbchan; i++) {
|
|
|
|
phi->bch[i].protocol = hw->bch[i].ch.protocol;
|
|
|
|
phi->bch[i].Flags = hw->bch[i].Flags;
|
|
|
|
}
|
|
|
|
_queue_data(&dch->dev.D, MPH_INFORMATION_IND, MISDN_ID_ANY,
|
2020-06-18 06:15:57 +07:00
|
|
|
struct_size(phi, bch, dch->dev.nrbchan), phi, GFP_ATOMIC);
|
2011-06-11 13:36:42 +07:00
|
|
|
kfree(phi);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Layer2 -> Layer 1 Dchannel data
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
hfcusb_l2l1D(struct mISDNchannel *ch, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct mISDNdevice *dev = container_of(ch, struct mISDNdevice, D);
|
|
|
|
struct dchannel *dch = container_of(dev, struct dchannel, dev);
|
|
|
|
struct mISDNhead *hh = mISDN_HEAD_P(skb);
|
|
|
|
struct hfcsusb *hw = dch->hw;
|
|
|
|
int ret = -EINVAL;
|
|
|
|
u_long flags;
|
|
|
|
|
|
|
|
switch (hh->prim) {
|
|
|
|
case PH_DATA_REQ:
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s: PH_DATA_REQ\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
spin_lock_irqsave(&hw->lock, flags);
|
|
|
|
ret = dchannel_senddata(dch, skb);
|
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
|
|
|
if (ret > 0) {
|
|
|
|
ret = 0;
|
|
|
|
queue_ch_frame(ch, PH_DATA_CNF, hh->id, NULL);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PH_ACTIVATE_REQ:
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s: PH_ACTIVATE_REQ %s\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__,
|
|
|
|
(hw->protocol == ISDN_P_NT_S0) ? "NT" : "TE");
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
if (hw->protocol == ISDN_P_NT_S0) {
|
|
|
|
ret = 0;
|
|
|
|
if (test_bit(FLG_ACTIVE, &dch->Flags)) {
|
|
|
|
_queue_data(&dch->dev.D,
|
2012-02-20 10:52:38 +07:00
|
|
|
PH_ACTIVATE_IND, MISDN_ID_ANY, 0,
|
|
|
|
NULL, GFP_ATOMIC);
|
2009-01-09 22:20:51 +07:00
|
|
|
} else {
|
|
|
|
hfcsusb_ph_command(hw,
|
2012-02-20 10:52:38 +07:00
|
|
|
HFC_L1_ACTIVATE_NT);
|
2009-01-09 22:20:51 +07:00
|
|
|
test_and_set_bit(FLG_L2_ACTIVATED,
|
2012-02-20 10:52:38 +07:00
|
|
|
&dch->Flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
hfcsusb_ph_command(hw, HFC_L1_ACTIVATE_TE);
|
|
|
|
ret = l1_event(dch->l1, hh->prim);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PH_DEACTIVATE_REQ:
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s: PH_DEACTIVATE_REQ\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__);
|
2009-01-09 22:20:51 +07:00
|
|
|
test_and_clear_bit(FLG_L2_ACTIVATED, &dch->Flags);
|
|
|
|
|
|
|
|
if (hw->protocol == ISDN_P_NT_S0) {
|
|
|
|
hfcsusb_ph_command(hw, HFC_L1_DEACTIVATE_NT);
|
|
|
|
spin_lock_irqsave(&hw->lock, flags);
|
|
|
|
skb_queue_purge(&dch->squeue);
|
|
|
|
if (dch->tx_skb) {
|
|
|
|
dev_kfree_skb(dch->tx_skb);
|
|
|
|
dch->tx_skb = NULL;
|
|
|
|
}
|
|
|
|
dch->tx_idx = 0;
|
|
|
|
if (dch->rx_skb) {
|
|
|
|
dev_kfree_skb(dch->rx_skb);
|
|
|
|
dch->rx_skb = NULL;
|
|
|
|
}
|
|
|
|
test_and_clear_bit(FLG_TX_BUSY, &dch->Flags);
|
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
|
|
|
#ifdef FIXME
|
|
|
|
if (test_and_clear_bit(FLG_L1_BUSY, &dch->Flags))
|
|
|
|
dchannel_sched_event(&hc->dch, D_CLEARBUSY);
|
|
|
|
#endif
|
|
|
|
ret = 0;
|
|
|
|
} else
|
|
|
|
ret = l1_event(dch->l1, hh->prim);
|
|
|
|
break;
|
|
|
|
case MPH_INFORMATION_REQ:
|
|
|
|
hfcsusb_ph_info(hw);
|
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Layer 1 callback function
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
hfc_l1callback(struct dchannel *dch, u_int cmd)
|
|
|
|
{
|
|
|
|
struct hfcsusb *hw = dch->hw;
|
|
|
|
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s cmd 0x%x\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, cmd);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
switch (cmd) {
|
|
|
|
case INFO3_P8:
|
|
|
|
case INFO3_P10:
|
|
|
|
case HW_RESET_REQ:
|
|
|
|
case HW_POWERUP_REQ:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HW_DEACT_REQ:
|
|
|
|
skb_queue_purge(&dch->squeue);
|
|
|
|
if (dch->tx_skb) {
|
|
|
|
dev_kfree_skb(dch->tx_skb);
|
|
|
|
dch->tx_skb = NULL;
|
|
|
|
}
|
|
|
|
dch->tx_idx = 0;
|
|
|
|
if (dch->rx_skb) {
|
|
|
|
dev_kfree_skb(dch->rx_skb);
|
|
|
|
dch->rx_skb = NULL;
|
|
|
|
}
|
|
|
|
test_and_clear_bit(FLG_TX_BUSY, &dch->Flags);
|
|
|
|
break;
|
|
|
|
case PH_ACTIVATE_IND:
|
|
|
|
test_and_set_bit(FLG_ACTIVE, &dch->Flags);
|
|
|
|
_queue_data(&dch->dev.D, cmd, MISDN_ID_ANY, 0, NULL,
|
2012-02-20 10:52:38 +07:00
|
|
|
GFP_ATOMIC);
|
2009-01-09 22:20:51 +07:00
|
|
|
break;
|
|
|
|
case PH_DEACTIVATE_IND:
|
|
|
|
test_and_clear_bit(FLG_ACTIVE, &dch->Flags);
|
|
|
|
_queue_data(&dch->dev.D, cmd, MISDN_ID_ANY, 0, NULL,
|
2012-02-20 10:52:38 +07:00
|
|
|
GFP_ATOMIC);
|
2009-01-09 22:20:51 +07:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (dch->debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG "%s: %s: unknown cmd %x\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, cmd);
|
2009-01-09 22:20:51 +07:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
hfcsusb_ph_info(hw);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
open_dchannel(struct hfcsusb *hw, struct mISDNchannel *ch,
|
2012-02-20 10:52:38 +07:00
|
|
|
struct channel_req *rq)
|
2009-01-09 22:20:51 +07:00
|
|
|
{
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
if (debug & DEBUG_HW_OPEN)
|
|
|
|
printk(KERN_DEBUG "%s: %s: dev(%d) open addr(%i) from %p\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, hw->dch.dev.id, rq->adr.channel,
|
|
|
|
__builtin_return_address(0));
|
2009-01-09 22:20:51 +07:00
|
|
|
if (rq->protocol == ISDN_P_NONE)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
test_and_clear_bit(FLG_ACTIVE, &hw->dch.Flags);
|
|
|
|
test_and_clear_bit(FLG_ACTIVE, &hw->ech.Flags);
|
|
|
|
hfcsusb_start_endpoint(hw, HFC_CHAN_D);
|
|
|
|
|
|
|
|
/* E-Channel logging */
|
|
|
|
if (rq->adr.channel == 1) {
|
|
|
|
if (hw->fifos[HFCUSB_PCM_RX].pipe) {
|
|
|
|
hfcsusb_start_endpoint(hw, HFC_CHAN_E);
|
|
|
|
set_bit(FLG_ACTIVE, &hw->ech.Flags);
|
|
|
|
_queue_data(&hw->ech.dev.D, PH_ACTIVATE_IND,
|
2012-02-20 10:52:38 +07:00
|
|
|
MISDN_ID_ANY, 0, NULL, GFP_ATOMIC);
|
2009-01-09 22:20:51 +07:00
|
|
|
} else
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!hw->initdone) {
|
|
|
|
hw->protocol = rq->protocol;
|
|
|
|
if (rq->protocol == ISDN_P_TE_S0) {
|
|
|
|
err = create_l1(&hw->dch, hfc_l1callback);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
setPortMode(hw);
|
|
|
|
ch->protocol = rq->protocol;
|
|
|
|
hw->initdone = 1;
|
|
|
|
} else {
|
|
|
|
if (rq->protocol != ch->protocol)
|
|
|
|
return -EPROTONOSUPPORT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (((ch->protocol == ISDN_P_NT_S0) && (hw->dch.state == 3)) ||
|
|
|
|
((ch->protocol == ISDN_P_TE_S0) && (hw->dch.state == 7)))
|
|
|
|
_queue_data(ch, PH_ACTIVATE_IND, MISDN_ID_ANY,
|
2012-02-20 10:52:38 +07:00
|
|
|
0, NULL, GFP_KERNEL);
|
2009-01-09 22:20:51 +07:00
|
|
|
rq->ch = ch;
|
|
|
|
if (!try_module_get(THIS_MODULE))
|
|
|
|
printk(KERN_WARNING "%s: %s: cannot get module\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__);
|
2009-01-09 22:20:51 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
open_bchannel(struct hfcsusb *hw, struct channel_req *rq)
|
|
|
|
{
|
|
|
|
struct bchannel *bch;
|
|
|
|
|
2012-03-27 04:20:48 +07:00
|
|
|
if (rq->adr.channel == 0 || rq->adr.channel > 2)
|
2009-01-09 22:20:51 +07:00
|
|
|
return -EINVAL;
|
|
|
|
if (rq->protocol == ISDN_P_NONE)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s B%i\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, rq->adr.channel);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
bch = &hw->bch[rq->adr.channel - 1];
|
|
|
|
if (test_and_set_bit(FLG_OPEN, &bch->Flags))
|
|
|
|
return -EBUSY; /* b-channel can be only open once */
|
|
|
|
bch->ch.protocol = rq->protocol;
|
|
|
|
rq->ch = &bch->ch;
|
|
|
|
|
|
|
|
if (!try_module_get(THIS_MODULE))
|
|
|
|
printk(KERN_WARNING "%s: %s:cannot get module\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__);
|
2009-01-09 22:20:51 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
channel_ctrl(struct hfcsusb *hw, struct mISDN_ctrl_req *cq)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s op(0x%x) channel(0x%x)\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, (cq->op), (cq->channel));
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
switch (cq->op) {
|
|
|
|
case MISDN_CTRL_GETOP:
|
|
|
|
cq->op = MISDN_CTRL_LOOP | MISDN_CTRL_CONNECT |
|
2012-02-20 10:52:38 +07:00
|
|
|
MISDN_CTRL_DISCONNECT;
|
2009-01-09 22:20:51 +07:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
printk(KERN_WARNING "%s: %s: unknown Op %x\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, cq->op);
|
2009-01-09 22:20:51 +07:00
|
|
|
ret = -EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* device control function
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
hfc_dctrl(struct mISDNchannel *ch, u_int cmd, void *arg)
|
|
|
|
{
|
|
|
|
struct mISDNdevice *dev = container_of(ch, struct mISDNdevice, D);
|
|
|
|
struct dchannel *dch = container_of(dev, struct dchannel, dev);
|
|
|
|
struct hfcsusb *hw = dch->hw;
|
|
|
|
struct channel_req *rq;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
if (dch->debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG "%s: %s: cmd:%x %p\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, cmd, arg);
|
2009-01-09 22:20:51 +07:00
|
|
|
switch (cmd) {
|
|
|
|
case OPEN_CHANNEL:
|
|
|
|
rq = arg;
|
|
|
|
if ((rq->protocol == ISDN_P_TE_S0) ||
|
|
|
|
(rq->protocol == ISDN_P_NT_S0))
|
|
|
|
err = open_dchannel(hw, ch, rq);
|
|
|
|
else
|
|
|
|
err = open_bchannel(hw, rq);
|
|
|
|
if (!err)
|
|
|
|
hw->open++;
|
|
|
|
break;
|
|
|
|
case CLOSE_CHANNEL:
|
|
|
|
hw->open--;
|
|
|
|
if (debug & DEBUG_HW_OPEN)
|
|
|
|
printk(KERN_DEBUG
|
2012-02-20 10:52:38 +07:00
|
|
|
"%s: %s: dev(%d) close from %p (open %d)\n",
|
|
|
|
hw->name, __func__, hw->dch.dev.id,
|
|
|
|
__builtin_return_address(0), hw->open);
|
2009-01-09 22:20:51 +07:00
|
|
|
if (!hw->open) {
|
|
|
|
hfcsusb_stop_endpoint(hw, HFC_CHAN_D);
|
|
|
|
if (hw->fifos[HFCUSB_PCM_RX].pipe)
|
|
|
|
hfcsusb_stop_endpoint(hw, HFC_CHAN_E);
|
|
|
|
handle_led(hw, LED_POWER_ON);
|
|
|
|
}
|
|
|
|
module_put(THIS_MODULE);
|
|
|
|
break;
|
|
|
|
case CONTROL_CHANNEL:
|
|
|
|
err = channel_ctrl(hw, arg);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (dch->debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG "%s: %s: unknown command %x\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, cmd);
|
2009-01-09 22:20:51 +07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* S0 TE state change event handler
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
ph_state_te(struct dchannel *dch)
|
|
|
|
{
|
|
|
|
struct hfcsusb *hw = dch->hw;
|
|
|
|
|
|
|
|
if (debug & DEBUG_HW) {
|
|
|
|
if (dch->state <= HFC_MAX_TE_LAYER1_STATE)
|
|
|
|
printk(KERN_DEBUG "%s: %s: %s\n", hw->name, __func__,
|
2012-02-20 10:52:38 +07:00
|
|
|
HFC_TE_LAYER1_STATES[dch->state]);
|
2009-01-09 22:20:51 +07:00
|
|
|
else
|
|
|
|
printk(KERN_DEBUG "%s: %s: TE F%d\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, dch->state);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (dch->state) {
|
|
|
|
case 0:
|
|
|
|
l1_event(dch->l1, HW_RESET_IND);
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
l1_event(dch->l1, HW_DEACT_IND);
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
case 8:
|
|
|
|
l1_event(dch->l1, ANYSIGNAL);
|
|
|
|
break;
|
|
|
|
case 6:
|
|
|
|
l1_event(dch->l1, INFO2);
|
|
|
|
break;
|
|
|
|
case 7:
|
|
|
|
l1_event(dch->l1, INFO4_P8);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (dch->state == 7)
|
|
|
|
handle_led(hw, LED_S0_ON);
|
|
|
|
else
|
|
|
|
handle_led(hw, LED_S0_OFF);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* S0 NT state change event handler
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
ph_state_nt(struct dchannel *dch)
|
|
|
|
{
|
|
|
|
struct hfcsusb *hw = dch->hw;
|
|
|
|
|
|
|
|
if (debug & DEBUG_HW) {
|
|
|
|
if (dch->state <= HFC_MAX_NT_LAYER1_STATE)
|
|
|
|
printk(KERN_DEBUG "%s: %s: %s\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__,
|
|
|
|
HFC_NT_LAYER1_STATES[dch->state]);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
else
|
|
|
|
printk(KERN_INFO DRIVER_NAME "%s: %s: NT G%d\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, dch->state);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (dch->state) {
|
|
|
|
case (1):
|
|
|
|
test_and_clear_bit(FLG_ACTIVE, &dch->Flags);
|
|
|
|
test_and_clear_bit(FLG_L2_ACTIVATED, &dch->Flags);
|
|
|
|
hw->nt_timer = 0;
|
|
|
|
hw->timers &= ~NT_ACTIVATION_TIMER;
|
|
|
|
handle_led(hw, LED_S0_OFF);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case (2):
|
|
|
|
if (hw->nt_timer < 0) {
|
|
|
|
hw->nt_timer = 0;
|
|
|
|
hw->timers &= ~NT_ACTIVATION_TIMER;
|
|
|
|
hfcsusb_ph_command(dch->hw, HFC_L1_DEACTIVATE_NT);
|
|
|
|
} else {
|
|
|
|
hw->timers |= NT_ACTIVATION_TIMER;
|
|
|
|
hw->nt_timer = NT_T1_COUNT;
|
|
|
|
/* allow G2 -> G3 transition */
|
|
|
|
write_reg(hw, HFCUSB_STATES, 2 | HFCUSB_NT_G2_G3);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case (3):
|
|
|
|
hw->nt_timer = 0;
|
|
|
|
hw->timers &= ~NT_ACTIVATION_TIMER;
|
|
|
|
test_and_set_bit(FLG_ACTIVE, &dch->Flags);
|
|
|
|
_queue_data(&dch->dev.D, PH_ACTIVATE_IND,
|
2012-02-20 10:52:38 +07:00
|
|
|
MISDN_ID_ANY, 0, NULL, GFP_ATOMIC);
|
2009-01-09 22:20:51 +07:00
|
|
|
handle_led(hw, LED_S0_ON);
|
|
|
|
break;
|
|
|
|
case (4):
|
|
|
|
hw->nt_timer = 0;
|
|
|
|
hw->timers &= ~NT_ACTIVATION_TIMER;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
hfcsusb_ph_info(hw);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
ph_state(struct dchannel *dch)
|
|
|
|
{
|
|
|
|
struct hfcsusb *hw = dch->hw;
|
|
|
|
|
|
|
|
if (hw->protocol == ISDN_P_NT_S0)
|
|
|
|
ph_state_nt(dch);
|
|
|
|
else if (hw->protocol == ISDN_P_TE_S0)
|
|
|
|
ph_state_te(dch);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* disable/enable BChannel for desired protocoll
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
hfcsusb_setup_bch(struct bchannel *bch, int protocol)
|
|
|
|
{
|
|
|
|
struct hfcsusb *hw = bch->hw;
|
|
|
|
__u8 conhdlc, sctrl, sctrl_r;
|
|
|
|
|
|
|
|
if (debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG "%s: %s: protocol %x-->%x B%d\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, bch->state, protocol,
|
|
|
|
bch->nr);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
/* setup val for CON_HDLC */
|
|
|
|
conhdlc = 0;
|
|
|
|
if (protocol > ISDN_P_NONE)
|
|
|
|
conhdlc = 8; /* enable FIFO */
|
|
|
|
|
|
|
|
switch (protocol) {
|
|
|
|
case (-1): /* used for init */
|
|
|
|
bch->state = -1;
|
2020-08-24 05:36:59 +07:00
|
|
|
fallthrough;
|
2009-01-09 22:20:51 +07:00
|
|
|
case (ISDN_P_NONE):
|
|
|
|
if (bch->state == ISDN_P_NONE)
|
|
|
|
return 0; /* already in idle state */
|
|
|
|
bch->state = ISDN_P_NONE;
|
|
|
|
clear_bit(FLG_HDLC, &bch->Flags);
|
|
|
|
clear_bit(FLG_TRANSPARENT, &bch->Flags);
|
|
|
|
break;
|
|
|
|
case (ISDN_P_B_RAW):
|
|
|
|
conhdlc |= 2;
|
|
|
|
bch->state = protocol;
|
|
|
|
set_bit(FLG_TRANSPARENT, &bch->Flags);
|
|
|
|
break;
|
|
|
|
case (ISDN_P_B_HDLC):
|
|
|
|
bch->state = protocol;
|
|
|
|
set_bit(FLG_HDLC, &bch->Flags);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG "%s: %s: prot not known %x\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, protocol);
|
2009-01-09 22:20:51 +07:00
|
|
|
return -ENOPROTOOPT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (protocol >= ISDN_P_NONE) {
|
|
|
|
write_reg(hw, HFCUSB_FIFO, (bch->nr == 1) ? 0 : 2);
|
|
|
|
write_reg(hw, HFCUSB_CON_HDLC, conhdlc);
|
|
|
|
write_reg(hw, HFCUSB_INC_RES_F, 2);
|
|
|
|
write_reg(hw, HFCUSB_FIFO, (bch->nr == 1) ? 1 : 3);
|
|
|
|
write_reg(hw, HFCUSB_CON_HDLC, conhdlc);
|
|
|
|
write_reg(hw, HFCUSB_INC_RES_F, 2);
|
|
|
|
|
|
|
|
sctrl = 0x40 + ((hw->protocol == ISDN_P_TE_S0) ? 0x00 : 0x04);
|
|
|
|
sctrl_r = 0x0;
|
|
|
|
if (test_bit(FLG_ACTIVE, &hw->bch[0].Flags)) {
|
|
|
|
sctrl |= 1;
|
|
|
|
sctrl_r |= 1;
|
|
|
|
}
|
|
|
|
if (test_bit(FLG_ACTIVE, &hw->bch[1].Flags)) {
|
|
|
|
sctrl |= 2;
|
|
|
|
sctrl_r |= 2;
|
|
|
|
}
|
|
|
|
write_reg(hw, HFCUSB_SCTRL, sctrl);
|
|
|
|
write_reg(hw, HFCUSB_SCTRL_R, sctrl_r);
|
|
|
|
|
|
|
|
if (protocol > ISDN_P_NONE)
|
|
|
|
handle_led(hw, (bch->nr == 1) ? LED_B1_ON : LED_B2_ON);
|
|
|
|
else
|
|
|
|
handle_led(hw, (bch->nr == 1) ? LED_B1_OFF :
|
2012-02-20 10:52:38 +07:00
|
|
|
LED_B2_OFF);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
hfcsusb_ph_info(hw);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
hfcsusb_ph_command(struct hfcsusb *hw, u_char command)
|
|
|
|
{
|
|
|
|
if (debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG "%s: %s: %x\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, command);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
switch (command) {
|
|
|
|
case HFC_L1_ACTIVATE_TE:
|
|
|
|
/* force sending sending INFO1 */
|
|
|
|
write_reg(hw, HFCUSB_STATES, 0x14);
|
|
|
|
/* start l1 activation */
|
|
|
|
write_reg(hw, HFCUSB_STATES, 0x04);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HFC_L1_FORCE_DEACTIVATE_TE:
|
|
|
|
write_reg(hw, HFCUSB_STATES, 0x10);
|
|
|
|
write_reg(hw, HFCUSB_STATES, 0x03);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HFC_L1_ACTIVATE_NT:
|
|
|
|
if (hw->dch.state == 3)
|
|
|
|
_queue_data(&hw->dch.dev.D, PH_ACTIVATE_IND,
|
2012-02-20 10:52:38 +07:00
|
|
|
MISDN_ID_ANY, 0, NULL, GFP_ATOMIC);
|
2009-01-09 22:20:51 +07:00
|
|
|
else
|
|
|
|
write_reg(hw, HFCUSB_STATES, HFCUSB_ACTIVATE |
|
2012-02-20 10:52:38 +07:00
|
|
|
HFCUSB_DO_ACTION | HFCUSB_NT_G2_G3);
|
2009-01-09 22:20:51 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case HFC_L1_DEACTIVATE_NT:
|
|
|
|
write_reg(hw, HFCUSB_STATES,
|
2012-02-20 10:52:38 +07:00
|
|
|
HFCUSB_DO_ACTION);
|
2009-01-09 22:20:51 +07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Layer 1 B-channel hardware access
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
channel_bctrl(struct bchannel *bch, struct mISDN_ctrl_req *cq)
|
|
|
|
{
|
2012-05-16 06:51:07 +07:00
|
|
|
return mISDN_ctrl_bchannel(bch, cq);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* collect data from incoming interrupt or isochron USB data */
|
|
|
|
static void
|
|
|
|
hfcsusb_rx_frame(struct usb_fifo *fifo, __u8 *data, unsigned int len,
|
2012-02-20 10:52:38 +07:00
|
|
|
int finish)
|
2009-01-09 22:20:51 +07:00
|
|
|
{
|
|
|
|
struct hfcsusb *hw = fifo->hw;
|
|
|
|
struct sk_buff *rx_skb = NULL;
|
|
|
|
int maxlen = 0;
|
|
|
|
int fifon = fifo->fifonum;
|
|
|
|
int i;
|
|
|
|
int hdlc = 0;
|
2018-06-20 17:40:28 +07:00
|
|
|
unsigned long flags;
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s: fifo(%i) len(%i) "
|
2012-02-20 10:52:38 +07:00
|
|
|
"dch(%p) bch(%p) ech(%p)\n",
|
|
|
|
hw->name, __func__, fifon, len,
|
|
|
|
fifo->dch, fifo->bch, fifo->ech);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
if (!len)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if ((!!fifo->dch + !!fifo->bch + !!fifo->ech) != 1) {
|
|
|
|
printk(KERN_DEBUG "%s: %s: undefined channel\n",
|
|
|
|
hw->name, __func__);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_lock_irqsave(&hw->lock, flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
if (fifo->dch) {
|
|
|
|
rx_skb = fifo->dch->rx_skb;
|
|
|
|
maxlen = fifo->dch->maxlen;
|
|
|
|
hdlc = 1;
|
|
|
|
}
|
|
|
|
if (fifo->bch) {
|
2012-05-16 06:51:08 +07:00
|
|
|
if (test_bit(FLG_RX_OFF, &fifo->bch->Flags)) {
|
|
|
|
fifo->bch->dropcnt += len;
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
2012-05-16 06:51:08 +07:00
|
|
|
return;
|
|
|
|
}
|
2012-05-16 06:51:05 +07:00
|
|
|
maxlen = bchannel_get_rxbuf(fifo->bch, len);
|
2009-01-09 22:20:51 +07:00
|
|
|
rx_skb = fifo->bch->rx_skb;
|
2012-05-16 06:51:05 +07:00
|
|
|
if (maxlen < 0) {
|
|
|
|
if (rx_skb)
|
|
|
|
skb_trim(rx_skb, 0);
|
2019-10-18 10:18:31 +07:00
|
|
|
pr_warn("%s.B%d: No bufferspace for %d bytes\n",
|
|
|
|
hw->name, fifo->bch->nr, len);
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
2012-05-16 06:51:05 +07:00
|
|
|
return;
|
|
|
|
}
|
2009-01-09 22:20:51 +07:00
|
|
|
maxlen = fifo->bch->maxlen;
|
|
|
|
hdlc = test_bit(FLG_HDLC, &fifo->bch->Flags);
|
|
|
|
}
|
|
|
|
if (fifo->ech) {
|
|
|
|
rx_skb = fifo->ech->rx_skb;
|
|
|
|
maxlen = fifo->ech->maxlen;
|
|
|
|
hdlc = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fifo->dch || fifo->ech) {
|
2012-05-16 06:51:05 +07:00
|
|
|
if (!rx_skb) {
|
|
|
|
rx_skb = mI_alloc_skb(maxlen, GFP_ATOMIC);
|
|
|
|
if (rx_skb) {
|
|
|
|
if (fifo->dch)
|
|
|
|
fifo->dch->rx_skb = rx_skb;
|
|
|
|
if (fifo->ech)
|
|
|
|
fifo->ech->rx_skb = rx_skb;
|
|
|
|
skb_trim(rx_skb, 0);
|
|
|
|
} else {
|
|
|
|
printk(KERN_DEBUG "%s: %s: No mem for rx_skb\n",
|
|
|
|
hw->name, __func__);
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
2012-05-16 06:51:05 +07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2009-01-09 22:20:51 +07:00
|
|
|
/* D/E-Channel SKB range check */
|
|
|
|
if ((rx_skb->len + len) >= MAX_DFRAME_LEN_L1) {
|
|
|
|
printk(KERN_DEBUG "%s: %s: sbk mem exceeded "
|
2012-02-20 10:52:38 +07:00
|
|
|
"for fifo(%d) HFCUSB_D_RX\n",
|
|
|
|
hw->name, __func__, fifon);
|
2009-01-09 22:20:51 +07:00
|
|
|
skb_trim(rx_skb, 0);
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
networking: introduce and use skb_put_data()
A common pattern with skb_put() is to just want to memcpy()
some data into the new space, introduce skb_put_data() for
this.
An spatch similar to the one for skb_put_zero() converts many
of the places using it:
@@
identifier p, p2;
expression len, skb, data;
type t, t2;
@@
(
-p = skb_put(skb, len);
+p = skb_put_data(skb, data, len);
|
-p = (t)skb_put(skb, len);
+p = skb_put_data(skb, data, len);
)
(
p2 = (t2)p;
-memcpy(p2, data, len);
|
-memcpy(p, data, len);
)
@@
type t, t2;
identifier p, p2;
expression skb, data;
@@
t *p;
...
(
-p = skb_put(skb, sizeof(t));
+p = skb_put_data(skb, data, sizeof(t));
|
-p = (t *)skb_put(skb, sizeof(t));
+p = skb_put_data(skb, data, sizeof(t));
)
(
p2 = (t2)p;
-memcpy(p2, data, sizeof(*p));
|
-memcpy(p, data, sizeof(*p));
)
@@
expression skb, len, data;
@@
-memcpy(skb_put(skb, len), data, len);
+skb_put_data(skb, data, len);
(again, manually post-processed to retain some comments)
Reviewed-by: Stephen Hemminger <stephen@networkplumber.org>
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2017-06-16 19:29:20 +07:00
|
|
|
skb_put_data(rx_skb, data, len);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
if (hdlc) {
|
|
|
|
/* we have a complete hdlc packet */
|
|
|
|
if (finish) {
|
|
|
|
if ((rx_skb->len > 3) &&
|
2012-02-20 10:52:38 +07:00
|
|
|
(!(rx_skb->data[rx_skb->len - 1]))) {
|
2009-01-09 22:20:51 +07:00
|
|
|
if (debug & DBG_HFC_FIFO_VERBOSE) {
|
|
|
|
printk(KERN_DEBUG "%s: %s: fifon(%i)"
|
2012-02-20 10:52:38 +07:00
|
|
|
" new RX len(%i): ",
|
|
|
|
hw->name, __func__, fifon,
|
|
|
|
rx_skb->len);
|
2009-01-09 22:20:51 +07:00
|
|
|
i = 0;
|
|
|
|
while (i < rx_skb->len)
|
|
|
|
printk("%02x ",
|
2012-02-20 10:52:38 +07:00
|
|
|
rx_skb->data[i++]);
|
2009-01-09 22:20:51 +07:00
|
|
|
printk("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* remove CRC & status */
|
|
|
|
skb_trim(rx_skb, rx_skb->len - 3);
|
|
|
|
|
|
|
|
if (fifo->dch)
|
|
|
|
recv_Dchannel(fifo->dch);
|
|
|
|
if (fifo->bch)
|
2012-05-16 06:51:06 +07:00
|
|
|
recv_Bchannel(fifo->bch, MISDN_ID_ANY,
|
|
|
|
0);
|
2009-01-09 22:20:51 +07:00
|
|
|
if (fifo->ech)
|
|
|
|
recv_Echannel(fifo->ech,
|
2012-02-20 10:52:38 +07:00
|
|
|
&hw->dch);
|
2009-01-09 22:20:51 +07:00
|
|
|
} else {
|
|
|
|
if (debug & DBG_HFC_FIFO_VERBOSE) {
|
|
|
|
printk(KERN_DEBUG
|
2012-02-20 10:52:38 +07:00
|
|
|
"%s: CRC or minlen ERROR fifon(%i) "
|
|
|
|
"RX len(%i): ",
|
|
|
|
hw->name, fifon, rx_skb->len);
|
2009-01-09 22:20:51 +07:00
|
|
|
i = 0;
|
|
|
|
while (i < rx_skb->len)
|
|
|
|
printk("%02x ",
|
2012-02-20 10:52:38 +07:00
|
|
|
rx_skb->data[i++]);
|
2009-01-09 22:20:51 +07:00
|
|
|
printk("\n");
|
|
|
|
}
|
|
|
|
skb_trim(rx_skb, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* deliver transparent data to layer2 */
|
2012-05-16 06:51:06 +07:00
|
|
|
recv_Bchannel(fifo->bch, MISDN_ID_ANY, false);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
|
2009-02-12 16:28:40 +07:00
|
|
|
static void
|
2009-01-09 22:20:51 +07:00
|
|
|
fill_isoc_urb(struct urb *urb, struct usb_device *dev, unsigned int pipe,
|
|
|
|
void *buf, int num_packets, int packet_size, int interval,
|
|
|
|
usb_complete_t complete, void *context)
|
|
|
|
{
|
|
|
|
int k;
|
|
|
|
|
|
|
|
usb_fill_bulk_urb(urb, dev, pipe, buf, packet_size * num_packets,
|
2012-02-20 10:52:38 +07:00
|
|
|
complete, context);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
urb->number_of_packets = num_packets;
|
|
|
|
urb->transfer_flags = URB_ISO_ASAP;
|
|
|
|
urb->actual_length = 0;
|
|
|
|
urb->interval = interval;
|
|
|
|
|
|
|
|
for (k = 0; k < num_packets; k++) {
|
|
|
|
urb->iso_frame_desc[k].offset = packet_size * k;
|
|
|
|
urb->iso_frame_desc[k].length = packet_size;
|
|
|
|
urb->iso_frame_desc[k].actual_length = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* receive completion routine for all ISO tx fifos */
|
|
|
|
static void
|
|
|
|
rx_iso_complete(struct urb *urb)
|
|
|
|
{
|
|
|
|
struct iso_urb *context_iso_urb = (struct iso_urb *) urb->context;
|
|
|
|
struct usb_fifo *fifo = context_iso_urb->owner_fifo;
|
|
|
|
struct hfcsusb *hw = fifo->hw;
|
|
|
|
int k, len, errcode, offset, num_isoc_packets, fifon, maxlen,
|
2012-02-20 10:52:38 +07:00
|
|
|
status, iso_status, i;
|
2009-01-09 22:20:51 +07:00
|
|
|
__u8 *buf;
|
|
|
|
static __u8 eof[8];
|
|
|
|
__u8 s0_state;
|
2018-06-20 17:40:28 +07:00
|
|
|
unsigned long flags;
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
fifon = fifo->fifonum;
|
|
|
|
status = urb->status;
|
|
|
|
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_lock_irqsave(&hw->lock, flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
if (fifo->stop_gracefull) {
|
|
|
|
fifo->stop_gracefull = 0;
|
|
|
|
fifo->active = 0;
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
return;
|
|
|
|
}
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* ISO transfer only partially completed,
|
|
|
|
* look at individual frame status for details
|
|
|
|
*/
|
|
|
|
if (status == -EXDEV) {
|
|
|
|
if (debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG "%s: %s: with -EXDEV "
|
2012-02-20 10:52:38 +07:00
|
|
|
"urb->status %d, fifonum %d\n",
|
|
|
|
hw->name, __func__, status, fifon);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
/* clear status, so go on with ISO transfers */
|
|
|
|
status = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
s0_state = 0;
|
|
|
|
if (fifo->active && !status) {
|
|
|
|
num_isoc_packets = iso_packets[fifon];
|
|
|
|
maxlen = fifo->usb_packet_maxlen;
|
|
|
|
|
|
|
|
for (k = 0; k < num_isoc_packets; ++k) {
|
|
|
|
len = urb->iso_frame_desc[k].actual_length;
|
|
|
|
offset = urb->iso_frame_desc[k].offset;
|
|
|
|
buf = context_iso_urb->buffer + offset;
|
|
|
|
iso_status = urb->iso_frame_desc[k].status;
|
|
|
|
|
|
|
|
if (iso_status && (debug & DBG_HFC_FIFO_VERBOSE)) {
|
|
|
|
printk(KERN_DEBUG "%s: %s: "
|
2012-02-20 10:52:38 +07:00
|
|
|
"ISO packet %i, status: %i\n",
|
|
|
|
hw->name, __func__, k, iso_status);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* USB data log for every D ISO in */
|
|
|
|
if ((fifon == HFCUSB_D_RX) &&
|
|
|
|
(debug & DBG_HFC_USB_VERBOSE)) {
|
|
|
|
printk(KERN_DEBUG
|
2012-02-20 10:52:38 +07:00
|
|
|
"%s: %s: %d (%d/%d) len(%d) ",
|
|
|
|
hw->name, __func__, urb->start_frame,
|
|
|
|
k, num_isoc_packets - 1,
|
|
|
|
len);
|
2009-01-09 22:20:51 +07:00
|
|
|
for (i = 0; i < len; i++)
|
|
|
|
printk("%x ", buf[i]);
|
|
|
|
printk("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!iso_status) {
|
|
|
|
if (fifo->last_urblen != maxlen) {
|
|
|
|
/*
|
|
|
|
* save fifo fill-level threshold bits
|
|
|
|
* to use them later in TX ISO URB
|
|
|
|
* completions
|
|
|
|
*/
|
|
|
|
hw->threshold_mask = buf[1];
|
|
|
|
|
|
|
|
if (fifon == HFCUSB_D_RX)
|
|
|
|
s0_state = (buf[0] >> 4);
|
|
|
|
|
|
|
|
eof[fifon] = buf[0] & 1;
|
|
|
|
if (len > 2)
|
|
|
|
hfcsusb_rx_frame(fifo, buf + 2,
|
2012-02-20 10:52:38 +07:00
|
|
|
len - 2, (len < maxlen)
|
|
|
|
? eof[fifon] : 0);
|
2009-01-09 22:20:51 +07:00
|
|
|
} else
|
|
|
|
hfcsusb_rx_frame(fifo, buf, len,
|
2012-02-20 10:52:38 +07:00
|
|
|
(len < maxlen) ?
|
|
|
|
eof[fifon] : 0);
|
2009-01-09 22:20:51 +07:00
|
|
|
fifo->last_urblen = len;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* signal S0 layer1 state change */
|
|
|
|
if ((s0_state) && (hw->initdone) &&
|
|
|
|
(s0_state != hw->dch.state)) {
|
|
|
|
hw->dch.state = s0_state;
|
|
|
|
schedule_event(&hw->dch, FLG_PHCHANGE);
|
|
|
|
}
|
|
|
|
|
|
|
|
fill_isoc_urb(urb, fifo->hw->dev, fifo->pipe,
|
|
|
|
context_iso_urb->buffer, num_isoc_packets,
|
|
|
|
fifo->usb_packet_maxlen, fifo->intervall,
|
|
|
|
(usb_complete_t)rx_iso_complete, urb->context);
|
|
|
|
errcode = usb_submit_urb(urb, GFP_ATOMIC);
|
|
|
|
if (errcode < 0) {
|
|
|
|
if (debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG "%s: %s: error submitting "
|
2012-02-20 10:52:38 +07:00
|
|
|
"ISO URB: %d\n",
|
|
|
|
hw->name, __func__, errcode);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (status && (debug & DBG_HFC_URB_INFO))
|
|
|
|
printk(KERN_DEBUG "%s: %s: rx_iso_complete : "
|
2012-02-20 10:52:38 +07:00
|
|
|
"urb->status %d, fifonum %d\n",
|
|
|
|
hw->name, __func__, status, fifon);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* receive completion routine for all interrupt rx fifos */
|
|
|
|
static void
|
|
|
|
rx_int_complete(struct urb *urb)
|
|
|
|
{
|
|
|
|
int len, status, i;
|
|
|
|
__u8 *buf, maxlen, fifon;
|
|
|
|
struct usb_fifo *fifo = (struct usb_fifo *) urb->context;
|
|
|
|
struct hfcsusb *hw = fifo->hw;
|
|
|
|
static __u8 eof[8];
|
2018-06-20 17:40:28 +07:00
|
|
|
unsigned long flags;
|
2009-01-09 22:20:51 +07:00
|
|
|
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_lock_irqsave(&hw->lock, flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
if (fifo->stop_gracefull) {
|
|
|
|
fifo->stop_gracefull = 0;
|
|
|
|
fifo->active = 0;
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
return;
|
|
|
|
}
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
fifon = fifo->fifonum;
|
|
|
|
if ((!fifo->active) || (urb->status)) {
|
|
|
|
if (debug & DBG_HFC_URB_ERROR)
|
|
|
|
printk(KERN_DEBUG
|
2012-02-20 10:52:38 +07:00
|
|
|
"%s: %s: RX-Fifo %i is going down (%i)\n",
|
|
|
|
hw->name, __func__, fifon, urb->status);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
fifo->urb->interval = 0; /* cancel automatic rescheduling */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
len = urb->actual_length;
|
|
|
|
buf = fifo->buffer;
|
|
|
|
maxlen = fifo->usb_packet_maxlen;
|
|
|
|
|
|
|
|
/* USB data log for every D INT in */
|
|
|
|
if ((fifon == HFCUSB_D_RX) && (debug & DBG_HFC_USB_VERBOSE)) {
|
|
|
|
printk(KERN_DEBUG "%s: %s: D RX INT len(%d) ",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, len);
|
2009-01-09 22:20:51 +07:00
|
|
|
for (i = 0; i < len; i++)
|
|
|
|
printk("%02x ", buf[i]);
|
|
|
|
printk("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fifo->last_urblen != fifo->usb_packet_maxlen) {
|
|
|
|
/* the threshold mask is in the 2nd status byte */
|
|
|
|
hw->threshold_mask = buf[1];
|
|
|
|
|
|
|
|
/* signal S0 layer1 state change */
|
|
|
|
if (hw->initdone && ((buf[0] >> 4) != hw->dch.state)) {
|
|
|
|
hw->dch.state = (buf[0] >> 4);
|
|
|
|
schedule_event(&hw->dch, FLG_PHCHANGE);
|
|
|
|
}
|
|
|
|
|
|
|
|
eof[fifon] = buf[0] & 1;
|
|
|
|
/* if we have more than the 2 status bytes -> collect data */
|
|
|
|
if (len > 2)
|
|
|
|
hfcsusb_rx_frame(fifo, buf + 2,
|
2012-02-20 10:52:38 +07:00
|
|
|
urb->actual_length - 2,
|
|
|
|
(len < maxlen) ? eof[fifon] : 0);
|
2009-01-09 22:20:51 +07:00
|
|
|
} else {
|
|
|
|
hfcsusb_rx_frame(fifo, buf, urb->actual_length,
|
|
|
|
(len < maxlen) ? eof[fifon] : 0);
|
|
|
|
}
|
|
|
|
fifo->last_urblen = urb->actual_length;
|
|
|
|
|
|
|
|
status = usb_submit_urb(urb, GFP_ATOMIC);
|
|
|
|
if (status) {
|
|
|
|
if (debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG "%s: %s: error resubmitting USB\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* transmit completion routine for all ISO tx fifos */
|
|
|
|
static void
|
|
|
|
tx_iso_complete(struct urb *urb)
|
|
|
|
{
|
|
|
|
struct iso_urb *context_iso_urb = (struct iso_urb *) urb->context;
|
|
|
|
struct usb_fifo *fifo = context_iso_urb->owner_fifo;
|
|
|
|
struct hfcsusb *hw = fifo->hw;
|
|
|
|
struct sk_buff *tx_skb;
|
|
|
|
int k, tx_offset, num_isoc_packets, sink, remain, current_len,
|
2012-02-20 10:52:38 +07:00
|
|
|
errcode, hdlc, i;
|
2009-01-09 22:20:51 +07:00
|
|
|
int *tx_idx;
|
2012-05-16 06:51:07 +07:00
|
|
|
int frame_complete, fifon, status, fillempty = 0;
|
|
|
|
__u8 threshbit, *p;
|
2018-06-20 17:40:28 +07:00
|
|
|
unsigned long flags;
|
2009-01-09 22:20:51 +07:00
|
|
|
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_lock_irqsave(&hw->lock, flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
if (fifo->stop_gracefull) {
|
|
|
|
fifo->stop_gracefull = 0;
|
|
|
|
fifo->active = 0;
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fifo->dch) {
|
|
|
|
tx_skb = fifo->dch->tx_skb;
|
|
|
|
tx_idx = &fifo->dch->tx_idx;
|
|
|
|
hdlc = 1;
|
|
|
|
} else if (fifo->bch) {
|
|
|
|
tx_skb = fifo->bch->tx_skb;
|
|
|
|
tx_idx = &fifo->bch->tx_idx;
|
|
|
|
hdlc = test_bit(FLG_HDLC, &fifo->bch->Flags);
|
2012-05-16 06:51:07 +07:00
|
|
|
if (!tx_skb && !hdlc &&
|
|
|
|
test_bit(FLG_FILLEMPTY, &fifo->bch->Flags))
|
|
|
|
fillempty = 1;
|
2009-01-09 22:20:51 +07:00
|
|
|
} else {
|
|
|
|
printk(KERN_DEBUG "%s: %s: neither BCH nor DCH\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__);
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
fifon = fifo->fifonum;
|
|
|
|
status = urb->status;
|
|
|
|
|
|
|
|
tx_offset = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* ISO transfer only partially completed,
|
|
|
|
* look at individual frame status for details
|
|
|
|
*/
|
|
|
|
if (status == -EXDEV) {
|
|
|
|
if (debug & DBG_HFC_URB_ERROR)
|
|
|
|
printk(KERN_DEBUG "%s: %s: "
|
2012-02-20 10:52:38 +07:00
|
|
|
"-EXDEV (%i) fifon (%d)\n",
|
|
|
|
hw->name, __func__, status, fifon);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
/* clear status, so go on with ISO transfers */
|
|
|
|
status = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fifo->active && !status) {
|
|
|
|
/* is FifoFull-threshold set for our channel? */
|
|
|
|
threshbit = (hw->threshold_mask & (1 << fifon));
|
|
|
|
num_isoc_packets = iso_packets[fifon];
|
|
|
|
|
|
|
|
/* predict dataflow to avoid fifo overflow */
|
|
|
|
if (fifon >= HFCUSB_D_TX)
|
|
|
|
sink = (threshbit) ? SINK_DMIN : SINK_DMAX;
|
|
|
|
else
|
|
|
|
sink = (threshbit) ? SINK_MIN : SINK_MAX;
|
|
|
|
fill_isoc_urb(urb, fifo->hw->dev, fifo->pipe,
|
|
|
|
context_iso_urb->buffer, num_isoc_packets,
|
|
|
|
fifo->usb_packet_maxlen, fifo->intervall,
|
|
|
|
(usb_complete_t)tx_iso_complete, urb->context);
|
|
|
|
memset(context_iso_urb->buffer, 0,
|
|
|
|
sizeof(context_iso_urb->buffer));
|
|
|
|
frame_complete = 0;
|
|
|
|
|
|
|
|
for (k = 0; k < num_isoc_packets; ++k) {
|
|
|
|
/* analyze tx success of previous ISO packets */
|
|
|
|
if (debug & DBG_HFC_URB_ERROR) {
|
|
|
|
errcode = urb->iso_frame_desc[k].status;
|
|
|
|
if (errcode) {
|
|
|
|
printk(KERN_DEBUG "%s: %s: "
|
2012-02-20 10:52:38 +07:00
|
|
|
"ISO packet %i, status: %i\n",
|
|
|
|
hw->name, __func__, k, errcode);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Generate next ISO Packets */
|
|
|
|
if (tx_skb)
|
|
|
|
remain = tx_skb->len - *tx_idx;
|
2012-05-16 06:51:07 +07:00
|
|
|
else if (fillempty)
|
|
|
|
remain = 15; /* > not complete */
|
2009-01-09 22:20:51 +07:00
|
|
|
else
|
|
|
|
remain = 0;
|
|
|
|
|
|
|
|
if (remain > 0) {
|
|
|
|
fifo->bit_line -= sink;
|
|
|
|
current_len = (0 - fifo->bit_line) / 8;
|
|
|
|
if (current_len > 14)
|
|
|
|
current_len = 14;
|
|
|
|
if (current_len < 0)
|
|
|
|
current_len = 0;
|
|
|
|
if (remain < current_len)
|
|
|
|
current_len = remain;
|
|
|
|
|
|
|
|
/* how much bit do we put on the line? */
|
|
|
|
fifo->bit_line += current_len * 8;
|
|
|
|
|
|
|
|
context_iso_urb->buffer[tx_offset] = 0;
|
|
|
|
if (current_len == remain) {
|
|
|
|
if (hdlc) {
|
|
|
|
/* signal frame completion */
|
|
|
|
context_iso_urb->
|
2012-02-20 10:52:38 +07:00
|
|
|
buffer[tx_offset] = 1;
|
2009-01-09 22:20:51 +07:00
|
|
|
/* add 2 byte flags and 16bit
|
|
|
|
* CRC at end of ISDN frame */
|
|
|
|
fifo->bit_line += 32;
|
|
|
|
}
|
|
|
|
frame_complete = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy tx data to iso-urb buffer */
|
2012-05-16 06:51:07 +07:00
|
|
|
p = context_iso_urb->buffer + tx_offset + 1;
|
|
|
|
if (fillempty) {
|
|
|
|
memset(p, fifo->bch->fill[0],
|
|
|
|
current_len);
|
|
|
|
} else {
|
|
|
|
memcpy(p, (tx_skb->data + *tx_idx),
|
|
|
|
current_len);
|
|
|
|
*tx_idx += current_len;
|
|
|
|
}
|
2009-01-09 22:20:51 +07:00
|
|
|
urb->iso_frame_desc[k].offset = tx_offset;
|
|
|
|
urb->iso_frame_desc[k].length = current_len + 1;
|
|
|
|
|
|
|
|
/* USB data log for every D ISO out */
|
2012-05-16 06:51:07 +07:00
|
|
|
if ((fifon == HFCUSB_D_RX) && !fillempty &&
|
2009-01-09 22:20:51 +07:00
|
|
|
(debug & DBG_HFC_USB_VERBOSE)) {
|
|
|
|
printk(KERN_DEBUG
|
2012-02-20 10:52:38 +07:00
|
|
|
"%s: %s (%d/%d) offs(%d) len(%d) ",
|
|
|
|
hw->name, __func__,
|
|
|
|
k, num_isoc_packets - 1,
|
|
|
|
urb->iso_frame_desc[k].offset,
|
|
|
|
urb->iso_frame_desc[k].length);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
for (i = urb->iso_frame_desc[k].offset;
|
|
|
|
i < (urb->iso_frame_desc[k].offset
|
2012-02-20 10:52:38 +07:00
|
|
|
+ urb->iso_frame_desc[k].length);
|
2009-01-09 22:20:51 +07:00
|
|
|
i++)
|
|
|
|
printk("%x ",
|
2012-02-20 10:52:38 +07:00
|
|
|
context_iso_urb->buffer[i]);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
printk(" skb->len(%i) tx-idx(%d)\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
tx_skb->len, *tx_idx);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
tx_offset += (current_len + 1);
|
|
|
|
} else {
|
|
|
|
urb->iso_frame_desc[k].offset = tx_offset++;
|
|
|
|
urb->iso_frame_desc[k].length = 1;
|
|
|
|
/* we lower data margin every msec */
|
|
|
|
fifo->bit_line -= sink;
|
|
|
|
if (fifo->bit_line < BITLINE_INF)
|
|
|
|
fifo->bit_line = BITLINE_INF;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (frame_complete) {
|
|
|
|
frame_complete = 0;
|
|
|
|
|
|
|
|
if (debug & DBG_HFC_FIFO_VERBOSE) {
|
|
|
|
printk(KERN_DEBUG "%s: %s: "
|
2012-02-20 10:52:38 +07:00
|
|
|
"fifon(%i) new TX len(%i): ",
|
|
|
|
hw->name, __func__,
|
|
|
|
fifon, tx_skb->len);
|
2009-01-09 22:20:51 +07:00
|
|
|
i = 0;
|
|
|
|
while (i < tx_skb->len)
|
|
|
|
printk("%02x ",
|
2012-02-20 10:52:38 +07:00
|
|
|
tx_skb->data[i++]);
|
2009-01-09 22:20:51 +07:00
|
|
|
printk("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
dev_kfree_skb(tx_skb);
|
|
|
|
tx_skb = NULL;
|
|
|
|
if (fifo->dch && get_next_dframe(fifo->dch))
|
|
|
|
tx_skb = fifo->dch->tx_skb;
|
|
|
|
else if (fifo->bch &&
|
2012-05-16 06:51:02 +07:00
|
|
|
get_next_bframe(fifo->bch))
|
2009-01-09 22:20:51 +07:00
|
|
|
tx_skb = fifo->bch->tx_skb;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
errcode = usb_submit_urb(urb, GFP_ATOMIC);
|
|
|
|
if (errcode < 0) {
|
|
|
|
if (debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG
|
2012-02-20 10:52:38 +07:00
|
|
|
"%s: %s: error submitting ISO URB: %d \n",
|
|
|
|
hw->name, __func__, errcode);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* abuse DChannel tx iso completion to trigger NT mode state
|
|
|
|
* changes tx_iso_complete is assumed to be called every
|
|
|
|
* fifo->intervall (ms)
|
|
|
|
*/
|
|
|
|
if ((fifon == HFCUSB_D_TX) && (hw->protocol == ISDN_P_NT_S0)
|
|
|
|
&& (hw->timers & NT_ACTIVATION_TIMER)) {
|
|
|
|
if ((--hw->nt_timer) < 0)
|
|
|
|
schedule_event(&hw->dch, FLG_PHCHANGE);
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
if (status && (debug & DBG_HFC_URB_ERROR))
|
|
|
|
printk(KERN_DEBUG "%s: %s: urb->status %s (%i)"
|
2012-02-20 10:52:38 +07:00
|
|
|
"fifonum=%d\n",
|
|
|
|
hw->name, __func__,
|
|
|
|
symbolic(urb_errlist, status), status, fifon);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
2018-06-20 17:40:28 +07:00
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* allocs urbs and start isoc transfer with two pending urbs to avoid
|
|
|
|
* gaps in the transfer chain
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
start_isoc_chain(struct usb_fifo *fifo, int num_packets_per_urb,
|
|
|
|
usb_complete_t complete, int packet_size)
|
|
|
|
{
|
|
|
|
struct hfcsusb *hw = fifo->hw;
|
|
|
|
int i, k, errcode;
|
|
|
|
|
|
|
|
if (debug)
|
|
|
|
printk(KERN_DEBUG "%s: %s: fifo %i\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, fifo->fifonum);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
/* allocate Memory for Iso out Urbs */
|
|
|
|
for (i = 0; i < 2; i++) {
|
|
|
|
if (!(fifo->iso[i].urb)) {
|
|
|
|
fifo->iso[i].urb =
|
2012-02-20 10:52:38 +07:00
|
|
|
usb_alloc_urb(num_packets_per_urb, GFP_KERNEL);
|
2009-01-09 22:20:51 +07:00
|
|
|
if (!(fifo->iso[i].urb)) {
|
|
|
|
printk(KERN_DEBUG
|
2012-02-20 10:52:38 +07:00
|
|
|
"%s: %s: alloc urb for fifo %i failed",
|
|
|
|
hw->name, __func__, fifo->fifonum);
|
2019-07-26 15:27:36 +07:00
|
|
|
continue;
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
fifo->iso[i].owner_fifo = (struct usb_fifo *) fifo;
|
|
|
|
fifo->iso[i].indx = i;
|
|
|
|
|
|
|
|
/* Init the first iso */
|
|
|
|
if (ISO_BUFFER_SIZE >=
|
|
|
|
(fifo->usb_packet_maxlen *
|
|
|
|
num_packets_per_urb)) {
|
|
|
|
fill_isoc_urb(fifo->iso[i].urb,
|
2012-02-20 10:52:38 +07:00
|
|
|
fifo->hw->dev, fifo->pipe,
|
|
|
|
fifo->iso[i].buffer,
|
|
|
|
num_packets_per_urb,
|
|
|
|
fifo->usb_packet_maxlen,
|
|
|
|
fifo->intervall, complete,
|
|
|
|
&fifo->iso[i]);
|
2009-01-09 22:20:51 +07:00
|
|
|
memset(fifo->iso[i].buffer, 0,
|
|
|
|
sizeof(fifo->iso[i].buffer));
|
|
|
|
|
|
|
|
for (k = 0; k < num_packets_per_urb; k++) {
|
|
|
|
fifo->iso[i].urb->
|
2012-02-20 10:52:38 +07:00
|
|
|
iso_frame_desc[k].offset =
|
|
|
|
k * packet_size;
|
2009-01-09 22:20:51 +07:00
|
|
|
fifo->iso[i].urb->
|
2012-02-20 10:52:38 +07:00
|
|
|
iso_frame_desc[k].length =
|
|
|
|
packet_size;
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
printk(KERN_DEBUG
|
2012-02-20 10:52:38 +07:00
|
|
|
"%s: %s: ISO Buffer size to small!\n",
|
|
|
|
hw->name, __func__);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fifo->bit_line = BITLINE_INF;
|
|
|
|
|
|
|
|
errcode = usb_submit_urb(fifo->iso[i].urb, GFP_KERNEL);
|
|
|
|
fifo->active = (errcode >= 0) ? 1 : 0;
|
|
|
|
fifo->stop_gracefull = 0;
|
|
|
|
if (errcode < 0) {
|
|
|
|
printk(KERN_DEBUG "%s: %s: %s URB nr:%d\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__,
|
|
|
|
symbolic(urb_errlist, errcode), i);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return fifo->active;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
stop_iso_gracefull(struct usb_fifo *fifo)
|
|
|
|
{
|
|
|
|
struct hfcsusb *hw = fifo->hw;
|
|
|
|
int i, timeout;
|
|
|
|
u_long flags;
|
|
|
|
|
|
|
|
for (i = 0; i < 2; i++) {
|
|
|
|
spin_lock_irqsave(&hw->lock, flags);
|
|
|
|
if (debug)
|
|
|
|
printk(KERN_DEBUG "%s: %s for fifo %i.%i\n",
|
|
|
|
hw->name, __func__, fifo->fifonum, i);
|
|
|
|
fifo->stop_gracefull = 1;
|
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < 2; i++) {
|
|
|
|
timeout = 3;
|
|
|
|
while (fifo->stop_gracefull && timeout--)
|
2012-02-20 10:52:38 +07:00
|
|
|
schedule_timeout_interruptible((HZ / 1000) * 16);
|
2009-01-09 22:20:51 +07:00
|
|
|
if (debug && fifo->stop_gracefull)
|
|
|
|
printk(KERN_DEBUG "%s: ERROR %s for fifo %i.%i\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, fifo->fifonum, i);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
stop_int_gracefull(struct usb_fifo *fifo)
|
|
|
|
{
|
|
|
|
struct hfcsusb *hw = fifo->hw;
|
|
|
|
int timeout;
|
|
|
|
u_long flags;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&hw->lock, flags);
|
|
|
|
if (debug)
|
|
|
|
printk(KERN_DEBUG "%s: %s for fifo %i\n",
|
|
|
|
hw->name, __func__, fifo->fifonum);
|
|
|
|
fifo->stop_gracefull = 1;
|
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
|
|
|
|
|
|
|
timeout = 3;
|
|
|
|
while (fifo->stop_gracefull && timeout--)
|
2012-02-20 10:52:38 +07:00
|
|
|
schedule_timeout_interruptible((HZ / 1000) * 3);
|
2009-01-09 22:20:51 +07:00
|
|
|
if (debug && fifo->stop_gracefull)
|
|
|
|
printk(KERN_DEBUG "%s: ERROR %s for fifo %i\n",
|
|
|
|
hw->name, __func__, fifo->fifonum);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* start the interrupt transfer for the given fifo */
|
|
|
|
static void
|
|
|
|
start_int_fifo(struct usb_fifo *fifo)
|
|
|
|
{
|
|
|
|
struct hfcsusb *hw = fifo->hw;
|
|
|
|
int errcode;
|
|
|
|
|
|
|
|
if (debug)
|
|
|
|
printk(KERN_DEBUG "%s: %s: INT IN fifo:%d\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, fifo->fifonum);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
if (!fifo->urb) {
|
|
|
|
fifo->urb = usb_alloc_urb(0, GFP_KERNEL);
|
|
|
|
if (!fifo->urb)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
usb_fill_int_urb(fifo->urb, fifo->hw->dev, fifo->pipe,
|
2012-02-20 10:52:38 +07:00
|
|
|
fifo->buffer, fifo->usb_packet_maxlen,
|
|
|
|
(usb_complete_t)rx_int_complete, fifo, fifo->intervall);
|
2009-01-09 22:20:51 +07:00
|
|
|
fifo->active = 1;
|
|
|
|
fifo->stop_gracefull = 0;
|
|
|
|
errcode = usb_submit_urb(fifo->urb, GFP_KERNEL);
|
|
|
|
if (errcode) {
|
|
|
|
printk(KERN_DEBUG "%s: %s: submit URB: status:%i\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, errcode);
|
2009-01-09 22:20:51 +07:00
|
|
|
fifo->active = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
setPortMode(struct hfcsusb *hw)
|
|
|
|
{
|
|
|
|
if (debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG "%s: %s %s\n", hw->name, __func__,
|
2012-02-20 10:52:38 +07:00
|
|
|
(hw->protocol == ISDN_P_TE_S0) ? "TE" : "NT");
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
if (hw->protocol == ISDN_P_TE_S0) {
|
|
|
|
write_reg(hw, HFCUSB_SCTRL, 0x40);
|
|
|
|
write_reg(hw, HFCUSB_SCTRL_E, 0x00);
|
|
|
|
write_reg(hw, HFCUSB_CLKDEL, CLKDEL_TE);
|
|
|
|
write_reg(hw, HFCUSB_STATES, 3 | 0x10);
|
|
|
|
write_reg(hw, HFCUSB_STATES, 3);
|
|
|
|
} else {
|
|
|
|
write_reg(hw, HFCUSB_SCTRL, 0x44);
|
|
|
|
write_reg(hw, HFCUSB_SCTRL_E, 0x09);
|
|
|
|
write_reg(hw, HFCUSB_CLKDEL, CLKDEL_NT);
|
|
|
|
write_reg(hw, HFCUSB_STATES, 1 | 0x10);
|
|
|
|
write_reg(hw, HFCUSB_STATES, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
reset_hfcsusb(struct hfcsusb *hw)
|
|
|
|
{
|
|
|
|
struct usb_fifo *fifo;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG "%s: %s\n", hw->name, __func__);
|
|
|
|
|
|
|
|
/* do Chip reset */
|
|
|
|
write_reg(hw, HFCUSB_CIRM, 8);
|
|
|
|
|
|
|
|
/* aux = output, reset off */
|
|
|
|
write_reg(hw, HFCUSB_CIRM, 0x10);
|
|
|
|
|
|
|
|
/* set USB_SIZE to match the wMaxPacketSize for INT or BULK transfers */
|
|
|
|
write_reg(hw, HFCUSB_USB_SIZE, (hw->packet_size / 8) |
|
2012-02-20 10:52:38 +07:00
|
|
|
((hw->packet_size / 8) << 4));
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
/* set USB_SIZE_I to match the the wMaxPacketSize for ISO transfers */
|
|
|
|
write_reg(hw, HFCUSB_USB_SIZE_I, hw->iso_packet_size);
|
|
|
|
|
|
|
|
/* enable PCM/GCI master mode */
|
|
|
|
write_reg(hw, HFCUSB_MST_MODE1, 0); /* set default values */
|
|
|
|
write_reg(hw, HFCUSB_MST_MODE0, 1); /* enable master mode */
|
|
|
|
|
|
|
|
/* init the fifos */
|
|
|
|
write_reg(hw, HFCUSB_F_THRES,
|
2012-02-20 10:52:38 +07:00
|
|
|
(HFCUSB_TX_THRESHOLD / 8) | ((HFCUSB_RX_THRESHOLD / 8) << 4));
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
fifo = hw->fifos;
|
|
|
|
for (i = 0; i < HFCUSB_NUM_FIFOS; i++) {
|
|
|
|
write_reg(hw, HFCUSB_FIFO, i); /* select the desired fifo */
|
|
|
|
fifo[i].max_size =
|
2012-02-20 10:52:38 +07:00
|
|
|
(i <= HFCUSB_B2_RX) ? MAX_BCH_SIZE : MAX_DFRAME_LEN;
|
2009-01-09 22:20:51 +07:00
|
|
|
fifo[i].last_urblen = 0;
|
|
|
|
|
|
|
|
/* set 2 bit for D- & E-channel */
|
|
|
|
write_reg(hw, HFCUSB_HDLC_PAR, ((i <= HFCUSB_B2_RX) ? 0 : 2));
|
|
|
|
|
|
|
|
/* enable all fifos */
|
|
|
|
if (i == HFCUSB_D_TX)
|
|
|
|
write_reg(hw, HFCUSB_CON_HDLC,
|
2012-02-20 10:52:38 +07:00
|
|
|
(hw->protocol == ISDN_P_NT_S0) ? 0x08 : 0x09);
|
2009-01-09 22:20:51 +07:00
|
|
|
else
|
|
|
|
write_reg(hw, HFCUSB_CON_HDLC, 0x08);
|
|
|
|
write_reg(hw, HFCUSB_INC_RES_F, 2); /* reset the fifo */
|
|
|
|
}
|
|
|
|
|
|
|
|
write_reg(hw, HFCUSB_SCTRL_R, 0); /* disable both B receivers */
|
|
|
|
handle_led(hw, LED_POWER_ON);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* start USB data pipes dependand on device's endpoint configuration */
|
|
|
|
static void
|
|
|
|
hfcsusb_start_endpoint(struct hfcsusb *hw, int channel)
|
|
|
|
{
|
|
|
|
/* quick check if endpoint already running */
|
|
|
|
if ((channel == HFC_CHAN_D) && (hw->fifos[HFCUSB_D_RX].active))
|
|
|
|
return;
|
|
|
|
if ((channel == HFC_CHAN_B1) && (hw->fifos[HFCUSB_B1_RX].active))
|
|
|
|
return;
|
|
|
|
if ((channel == HFC_CHAN_B2) && (hw->fifos[HFCUSB_B2_RX].active))
|
|
|
|
return;
|
|
|
|
if ((channel == HFC_CHAN_E) && (hw->fifos[HFCUSB_PCM_RX].active))
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* start rx endpoints using USB INT IN method */
|
|
|
|
if (hw->cfg_used == CNF_3INT3ISO || hw->cfg_used == CNF_4INT3ISO)
|
2012-02-20 10:52:38 +07:00
|
|
|
start_int_fifo(hw->fifos + channel * 2 + 1);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
/* start rx endpoints using USB ISO IN method */
|
|
|
|
if (hw->cfg_used == CNF_3ISO3ISO || hw->cfg_used == CNF_4ISO3ISO) {
|
|
|
|
switch (channel) {
|
|
|
|
case HFC_CHAN_D:
|
|
|
|
start_isoc_chain(hw->fifos + HFCUSB_D_RX,
|
2012-02-20 10:52:38 +07:00
|
|
|
ISOC_PACKETS_D,
|
|
|
|
(usb_complete_t)rx_iso_complete,
|
|
|
|
16);
|
2009-01-09 22:20:51 +07:00
|
|
|
break;
|
|
|
|
case HFC_CHAN_E:
|
|
|
|
start_isoc_chain(hw->fifos + HFCUSB_PCM_RX,
|
2012-02-20 10:52:38 +07:00
|
|
|
ISOC_PACKETS_D,
|
|
|
|
(usb_complete_t)rx_iso_complete,
|
|
|
|
16);
|
2009-01-09 22:20:51 +07:00
|
|
|
break;
|
|
|
|
case HFC_CHAN_B1:
|
|
|
|
start_isoc_chain(hw->fifos + HFCUSB_B1_RX,
|
2012-02-20 10:52:38 +07:00
|
|
|
ISOC_PACKETS_B,
|
|
|
|
(usb_complete_t)rx_iso_complete,
|
|
|
|
16);
|
2009-01-09 22:20:51 +07:00
|
|
|
break;
|
|
|
|
case HFC_CHAN_B2:
|
|
|
|
start_isoc_chain(hw->fifos + HFCUSB_B2_RX,
|
2012-02-20 10:52:38 +07:00
|
|
|
ISOC_PACKETS_B,
|
|
|
|
(usb_complete_t)rx_iso_complete,
|
|
|
|
16);
|
2009-01-09 22:20:51 +07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* start tx endpoints using USB ISO OUT method */
|
|
|
|
switch (channel) {
|
|
|
|
case HFC_CHAN_D:
|
|
|
|
start_isoc_chain(hw->fifos + HFCUSB_D_TX,
|
2012-02-20 10:52:38 +07:00
|
|
|
ISOC_PACKETS_B,
|
|
|
|
(usb_complete_t)tx_iso_complete, 1);
|
2009-01-09 22:20:51 +07:00
|
|
|
break;
|
|
|
|
case HFC_CHAN_B1:
|
|
|
|
start_isoc_chain(hw->fifos + HFCUSB_B1_TX,
|
2012-02-20 10:52:38 +07:00
|
|
|
ISOC_PACKETS_D,
|
|
|
|
(usb_complete_t)tx_iso_complete, 1);
|
2009-01-09 22:20:51 +07:00
|
|
|
break;
|
|
|
|
case HFC_CHAN_B2:
|
|
|
|
start_isoc_chain(hw->fifos + HFCUSB_B2_TX,
|
2012-02-20 10:52:38 +07:00
|
|
|
ISOC_PACKETS_B,
|
|
|
|
(usb_complete_t)tx_iso_complete, 1);
|
2009-01-09 22:20:51 +07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* stop USB data pipes dependand on device's endpoint configuration */
|
|
|
|
static void
|
|
|
|
hfcsusb_stop_endpoint(struct hfcsusb *hw, int channel)
|
|
|
|
{
|
|
|
|
/* quick check if endpoint currently running */
|
|
|
|
if ((channel == HFC_CHAN_D) && (!hw->fifos[HFCUSB_D_RX].active))
|
|
|
|
return;
|
|
|
|
if ((channel == HFC_CHAN_B1) && (!hw->fifos[HFCUSB_B1_RX].active))
|
|
|
|
return;
|
|
|
|
if ((channel == HFC_CHAN_B2) && (!hw->fifos[HFCUSB_B2_RX].active))
|
|
|
|
return;
|
|
|
|
if ((channel == HFC_CHAN_E) && (!hw->fifos[HFCUSB_PCM_RX].active))
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* rx endpoints using USB INT IN method */
|
|
|
|
if (hw->cfg_used == CNF_3INT3ISO || hw->cfg_used == CNF_4INT3ISO)
|
2012-02-20 10:52:38 +07:00
|
|
|
stop_int_gracefull(hw->fifos + channel * 2 + 1);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
/* rx endpoints using USB ISO IN method */
|
|
|
|
if (hw->cfg_used == CNF_3ISO3ISO || hw->cfg_used == CNF_4ISO3ISO)
|
2012-02-20 10:52:38 +07:00
|
|
|
stop_iso_gracefull(hw->fifos + channel * 2 + 1);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
/* tx endpoints using USB ISO OUT method */
|
|
|
|
if (channel != HFC_CHAN_E)
|
2012-02-20 10:52:38 +07:00
|
|
|
stop_iso_gracefull(hw->fifos + channel * 2);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Hardware Initialization */
|
2009-02-12 16:28:40 +07:00
|
|
|
static int
|
2009-01-09 22:20:51 +07:00
|
|
|
setup_hfcsusb(struct hfcsusb *hw)
|
|
|
|
{
|
2019-07-31 20:17:23 +07:00
|
|
|
void *dmabuf = kmalloc(sizeof(u_char), GFP_KERNEL);
|
2009-01-09 22:20:51 +07:00
|
|
|
u_char b;
|
2019-07-31 20:17:23 +07:00
|
|
|
int ret;
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s\n", hw->name, __func__);
|
|
|
|
|
2019-07-31 20:17:23 +07:00
|
|
|
if (!dmabuf)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
ret = read_reg_atomic(hw, HFCUSB_CHIP_ID, dmabuf);
|
|
|
|
|
|
|
|
memcpy(&b, dmabuf, sizeof(u_char));
|
|
|
|
kfree(dmabuf);
|
|
|
|
|
2009-01-09 22:20:51 +07:00
|
|
|
/* check the chip id */
|
2019-07-31 20:17:23 +07:00
|
|
|
if (ret != 1) {
|
2009-01-09 22:20:51 +07:00
|
|
|
printk(KERN_DEBUG "%s: %s: cannot read chip id\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__);
|
2009-01-09 22:20:51 +07:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (b != HFCUSB_CHIPID) {
|
|
|
|
printk(KERN_DEBUG "%s: %s: Invalid chip id 0x%02x\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, b);
|
2009-01-09 22:20:51 +07:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* first set the needed config, interface and alternate */
|
2011-04-18 06:35:27 +07:00
|
|
|
(void) usb_set_interface(hw->dev, hw->if_used, hw->alt_used);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
hw->led_state = 0;
|
|
|
|
|
|
|
|
/* init the background machinery for control requests */
|
|
|
|
hw->ctrl_read.bRequestType = 0xc0;
|
|
|
|
hw->ctrl_read.bRequest = 1;
|
|
|
|
hw->ctrl_read.wLength = cpu_to_le16(1);
|
|
|
|
hw->ctrl_write.bRequestType = 0x40;
|
|
|
|
hw->ctrl_write.bRequest = 0;
|
|
|
|
hw->ctrl_write.wLength = 0;
|
|
|
|
usb_fill_control_urb(hw->ctrl_urb, hw->dev, hw->ctrl_out_pipe,
|
2012-02-20 10:52:38 +07:00
|
|
|
(u_char *)&hw->ctrl_write, NULL, 0,
|
|
|
|
(usb_complete_t)ctrl_complete, hw);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
reset_hfcsusb(hw);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
release_hw(struct hfcsusb *hw)
|
|
|
|
{
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s\n", hw->name, __func__);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* stop all endpoints gracefully
|
|
|
|
* TODO: mISDN_core should generate CLOSE_CHANNEL
|
|
|
|
* signals after calling mISDN_unregister_device()
|
|
|
|
*/
|
|
|
|
hfcsusb_stop_endpoint(hw, HFC_CHAN_D);
|
|
|
|
hfcsusb_stop_endpoint(hw, HFC_CHAN_B1);
|
|
|
|
hfcsusb_stop_endpoint(hw, HFC_CHAN_B2);
|
|
|
|
if (hw->fifos[HFCUSB_PCM_RX].pipe)
|
|
|
|
hfcsusb_stop_endpoint(hw, HFC_CHAN_E);
|
|
|
|
if (hw->protocol == ISDN_P_TE_S0)
|
|
|
|
l1_event(hw->dch.l1, CLOSE_CHANNEL);
|
|
|
|
|
|
|
|
mISDN_unregister_device(&hw->dch.dev);
|
|
|
|
mISDN_freebchannel(&hw->bch[1]);
|
|
|
|
mISDN_freebchannel(&hw->bch[0]);
|
|
|
|
mISDN_freedchannel(&hw->dch);
|
|
|
|
|
|
|
|
if (hw->ctrl_urb) {
|
|
|
|
usb_kill_urb(hw->ctrl_urb);
|
|
|
|
usb_free_urb(hw->ctrl_urb);
|
|
|
|
hw->ctrl_urb = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hw->intf)
|
|
|
|
usb_set_intfdata(hw->intf, NULL);
|
|
|
|
list_del(&hw->list);
|
|
|
|
kfree(hw);
|
|
|
|
hw = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
deactivate_bchannel(struct bchannel *bch)
|
|
|
|
{
|
|
|
|
struct hfcsusb *hw = bch->hw;
|
|
|
|
u_long flags;
|
|
|
|
|
|
|
|
if (bch->debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG "%s: %s: bch->nr(%i)\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
hw->name, __func__, bch->nr);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
spin_lock_irqsave(&hw->lock, flags);
|
2009-07-09 15:02:29 +07:00
|
|
|
mISDN_clear_bchannel(bch);
|
2009-01-09 22:20:51 +07:00
|
|
|
spin_unlock_irqrestore(&hw->lock, flags);
|
|
|
|
hfcsusb_setup_bch(bch, ISDN_P_NONE);
|
2012-05-16 06:51:04 +07:00
|
|
|
hfcsusb_stop_endpoint(hw, bch->nr - 1);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Layer 1 B-channel hardware access
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
hfc_bctrl(struct mISDNchannel *ch, u_int cmd, void *arg)
|
|
|
|
{
|
|
|
|
struct bchannel *bch = container_of(ch, struct bchannel, ch);
|
|
|
|
int ret = -EINVAL;
|
|
|
|
|
|
|
|
if (bch->debug & DEBUG_HW)
|
|
|
|
printk(KERN_DEBUG "%s: cmd:%x %p\n", __func__, cmd, arg);
|
|
|
|
|
|
|
|
switch (cmd) {
|
|
|
|
case HW_TESTRX_RAW:
|
|
|
|
case HW_TESTRX_HDLC:
|
|
|
|
case HW_TESTRX_OFF:
|
|
|
|
ret = -EINVAL;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CLOSE_CHANNEL:
|
|
|
|
test_and_clear_bit(FLG_OPEN, &bch->Flags);
|
2012-05-16 06:51:01 +07:00
|
|
|
deactivate_bchannel(bch);
|
2009-01-09 22:20:51 +07:00
|
|
|
ch->protocol = ISDN_P_NONE;
|
|
|
|
ch->peer = NULL;
|
|
|
|
module_put(THIS_MODULE);
|
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
case CONTROL_CHANNEL:
|
|
|
|
ret = channel_bctrl(bch, arg);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
printk(KERN_WARNING "%s: unknown prim(%x)\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
__func__, cmd);
|
2009-01-09 22:20:51 +07:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
setup_instance(struct hfcsusb *hw, struct device *parent)
|
|
|
|
{
|
|
|
|
u_long flags;
|
|
|
|
int err, i;
|
|
|
|
|
|
|
|
if (debug & DBG_HFC_CALL_TRACE)
|
|
|
|
printk(KERN_DEBUG "%s: %s\n", hw->name, __func__);
|
|
|
|
|
|
|
|
spin_lock_init(&hw->ctrl_lock);
|
|
|
|
spin_lock_init(&hw->lock);
|
|
|
|
|
|
|
|
mISDN_initdchannel(&hw->dch, MAX_DFRAME_LEN_L1, ph_state);
|
|
|
|
hw->dch.debug = debug & 0xFFFF;
|
|
|
|
hw->dch.hw = hw;
|
|
|
|
hw->dch.dev.Dprotocols = (1 << ISDN_P_TE_S0) | (1 << ISDN_P_NT_S0);
|
|
|
|
hw->dch.dev.D.send = hfcusb_l2l1D;
|
|
|
|
hw->dch.dev.D.ctrl = hfc_dctrl;
|
|
|
|
|
|
|
|
/* enable E-Channel logging */
|
|
|
|
if (hw->fifos[HFCUSB_PCM_RX].pipe)
|
|
|
|
mISDN_initdchannel(&hw->ech, MAX_DFRAME_LEN_L1, NULL);
|
|
|
|
|
|
|
|
hw->dch.dev.Bprotocols = (1 << (ISDN_P_B_RAW & ISDN_P_B_MASK)) |
|
2012-02-20 10:52:38 +07:00
|
|
|
(1 << (ISDN_P_B_HDLC & ISDN_P_B_MASK));
|
2009-01-09 22:20:51 +07:00
|
|
|
hw->dch.dev.nrbchan = 2;
|
|
|
|
for (i = 0; i < 2; i++) {
|
|
|
|
hw->bch[i].nr = i + 1;
|
|
|
|
set_channelmap(i + 1, hw->dch.dev.channelmap);
|
|
|
|
hw->bch[i].debug = debug;
|
2012-05-16 06:51:06 +07:00
|
|
|
mISDN_initbchannel(&hw->bch[i], MAX_DATA_MEM, poll >> 1);
|
2009-01-09 22:20:51 +07:00
|
|
|
hw->bch[i].hw = hw;
|
|
|
|
hw->bch[i].ch.send = hfcusb_l2l1B;
|
|
|
|
hw->bch[i].ch.ctrl = hfc_bctrl;
|
|
|
|
hw->bch[i].ch.nr = i + 1;
|
|
|
|
list_add(&hw->bch[i].ch.list, &hw->dch.dev.bchannels);
|
|
|
|
}
|
|
|
|
|
|
|
|
hw->fifos[HFCUSB_B1_TX].bch = &hw->bch[0];
|
|
|
|
hw->fifos[HFCUSB_B1_RX].bch = &hw->bch[0];
|
|
|
|
hw->fifos[HFCUSB_B2_TX].bch = &hw->bch[1];
|
|
|
|
hw->fifos[HFCUSB_B2_RX].bch = &hw->bch[1];
|
|
|
|
hw->fifos[HFCUSB_D_TX].dch = &hw->dch;
|
|
|
|
hw->fifos[HFCUSB_D_RX].dch = &hw->dch;
|
|
|
|
hw->fifos[HFCUSB_PCM_RX].ech = &hw->ech;
|
|
|
|
hw->fifos[HFCUSB_PCM_TX].ech = &hw->ech;
|
|
|
|
|
|
|
|
err = setup_hfcsusb(hw);
|
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
snprintf(hw->name, MISDN_MAX_IDLEN - 1, "%s.%d", DRIVER_NAME,
|
2012-02-20 10:52:38 +07:00
|
|
|
hfcsusb_cnt + 1);
|
2009-01-09 22:20:51 +07:00
|
|
|
printk(KERN_INFO "%s: registered as '%s'\n",
|
2012-02-20 10:52:38 +07:00
|
|
|
DRIVER_NAME, hw->name);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
err = mISDN_register_device(&hw->dch.dev, parent, hw->name);
|
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
hfcsusb_cnt++;
|
|
|
|
write_lock_irqsave(&HFClock, flags);
|
|
|
|
list_add_tail(&hw->list, &HFClist);
|
|
|
|
write_unlock_irqrestore(&HFClock, flags);
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
out:
|
|
|
|
mISDN_freebchannel(&hw->bch[1]);
|
|
|
|
mISDN_freebchannel(&hw->bch[0]);
|
|
|
|
mISDN_freedchannel(&hw->dch);
|
|
|
|
kfree(hw);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
hfcsusb_probe(struct usb_interface *intf, const struct usb_device_id *id)
|
|
|
|
{
|
|
|
|
struct hfcsusb *hw;
|
|
|
|
struct usb_device *dev = interface_to_usbdev(intf);
|
|
|
|
struct usb_host_interface *iface = intf->cur_altsetting;
|
|
|
|
struct usb_host_interface *iface_used = NULL;
|
|
|
|
struct usb_host_endpoint *ep;
|
|
|
|
struct hfcsusb_vdata *driver_info;
|
|
|
|
int ifnum = iface->desc.bInterfaceNumber, i, idx, alt_idx,
|
2012-02-20 10:52:38 +07:00
|
|
|
probe_alt_setting, vend_idx, cfg_used, *vcf, attr, cfg_found,
|
|
|
|
ep_addr, cmptbl[16], small_match, iso_packet_size, packet_size,
|
|
|
|
alt_used = 0;
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
vend_idx = 0xffff;
|
|
|
|
for (i = 0; hfcsusb_idtab[i].idVendor; i++) {
|
|
|
|
if ((le16_to_cpu(dev->descriptor.idVendor)
|
2012-02-20 10:52:38 +07:00
|
|
|
== hfcsusb_idtab[i].idVendor) &&
|
2009-01-09 22:20:51 +07:00
|
|
|
(le16_to_cpu(dev->descriptor.idProduct)
|
2012-02-20 10:52:38 +07:00
|
|
|
== hfcsusb_idtab[i].idProduct)) {
|
2009-01-09 22:20:51 +07:00
|
|
|
vend_idx = i;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printk(KERN_DEBUG
|
2012-02-20 10:52:38 +07:00
|
|
|
"%s: interface(%d) actalt(%d) minor(%d) vend_idx(%d)\n",
|
|
|
|
__func__, ifnum, iface->desc.bAlternateSetting,
|
|
|
|
intf->minor, vend_idx);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
if (vend_idx == 0xffff) {
|
|
|
|
printk(KERN_WARNING
|
2012-02-20 10:52:38 +07:00
|
|
|
"%s: no valid vendor found in USB descriptor\n",
|
|
|
|
__func__);
|
2009-01-09 22:20:51 +07:00
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
/* if vendor and product ID is OK, start probing alternate settings */
|
|
|
|
alt_idx = 0;
|
|
|
|
small_match = -1;
|
|
|
|
|
|
|
|
/* default settings */
|
|
|
|
iso_packet_size = 16;
|
|
|
|
packet_size = 64;
|
|
|
|
|
|
|
|
while (alt_idx < intf->num_altsetting) {
|
|
|
|
iface = intf->altsetting + alt_idx;
|
|
|
|
probe_alt_setting = iface->desc.bAlternateSetting;
|
|
|
|
cfg_used = 0;
|
|
|
|
|
|
|
|
while (validconf[cfg_used][0]) {
|
|
|
|
cfg_found = 1;
|
|
|
|
vcf = validconf[cfg_used];
|
|
|
|
ep = iface->endpoint;
|
|
|
|
memcpy(cmptbl, vcf, 16 * sizeof(int));
|
|
|
|
|
|
|
|
/* check for all endpoints in this alternate setting */
|
|
|
|
for (i = 0; i < iface->desc.bNumEndpoints; i++) {
|
|
|
|
ep_addr = ep->desc.bEndpointAddress;
|
|
|
|
|
|
|
|
/* get endpoint base */
|
|
|
|
idx = ((ep_addr & 0x7f) - 1) * 2;
|
2019-07-15 22:08:14 +07:00
|
|
|
if (idx > 15)
|
|
|
|
return -EIO;
|
|
|
|
|
2009-01-09 22:20:51 +07:00
|
|
|
if (ep_addr & 0x80)
|
|
|
|
idx++;
|
|
|
|
attr = ep->desc.bmAttributes;
|
|
|
|
|
|
|
|
if (cmptbl[idx] != EP_NOP) {
|
|
|
|
if (cmptbl[idx] == EP_NUL)
|
|
|
|
cfg_found = 0;
|
|
|
|
if (attr == USB_ENDPOINT_XFER_INT
|
2012-02-20 10:52:38 +07:00
|
|
|
&& cmptbl[idx] == EP_INT)
|
2009-01-09 22:20:51 +07:00
|
|
|
cmptbl[idx] = EP_NUL;
|
|
|
|
if (attr == USB_ENDPOINT_XFER_BULK
|
2012-02-20 10:52:38 +07:00
|
|
|
&& cmptbl[idx] == EP_BLK)
|
2009-01-09 22:20:51 +07:00
|
|
|
cmptbl[idx] = EP_NUL;
|
|
|
|
if (attr == USB_ENDPOINT_XFER_ISOC
|
2012-02-20 10:52:38 +07:00
|
|
|
&& cmptbl[idx] == EP_ISO)
|
2009-01-09 22:20:51 +07:00
|
|
|
cmptbl[idx] = EP_NUL;
|
|
|
|
|
|
|
|
if (attr == USB_ENDPOINT_XFER_INT &&
|
2012-02-20 10:52:38 +07:00
|
|
|
ep->desc.bInterval < vcf[17]) {
|
2009-01-09 22:20:51 +07:00
|
|
|
cfg_found = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ep++;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
if (cmptbl[i] != EP_NOP && cmptbl[i] != EP_NUL)
|
|
|
|
cfg_found = 0;
|
|
|
|
|
|
|
|
if (cfg_found) {
|
|
|
|
if (small_match < cfg_used) {
|
|
|
|
small_match = cfg_used;
|
|
|
|
alt_used = probe_alt_setting;
|
|
|
|
iface_used = iface;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
cfg_used++;
|
|
|
|
}
|
|
|
|
alt_idx++;
|
|
|
|
} /* (alt_idx < intf->num_altsetting) */
|
|
|
|
|
|
|
|
/* not found a valid USB Ta Endpoint config */
|
|
|
|
if (small_match == -1)
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
iface = iface_used;
|
|
|
|
hw = kzalloc(sizeof(struct hfcsusb), GFP_KERNEL);
|
|
|
|
if (!hw)
|
|
|
|
return -ENOMEM; /* got no mem */
|
|
|
|
snprintf(hw->name, MISDN_MAX_IDLEN - 1, "%s", DRIVER_NAME);
|
|
|
|
|
|
|
|
ep = iface->endpoint;
|
|
|
|
vcf = validconf[small_match];
|
|
|
|
|
|
|
|
for (i = 0; i < iface->desc.bNumEndpoints; i++) {
|
|
|
|
struct usb_fifo *f;
|
|
|
|
|
|
|
|
ep_addr = ep->desc.bEndpointAddress;
|
|
|
|
/* get endpoint base */
|
|
|
|
idx = ((ep_addr & 0x7f) - 1) * 2;
|
|
|
|
if (ep_addr & 0x80)
|
|
|
|
idx++;
|
|
|
|
f = &hw->fifos[idx & 7];
|
|
|
|
|
|
|
|
/* init Endpoints */
|
|
|
|
if (vcf[idx] == EP_NOP || vcf[idx] == EP_NUL) {
|
|
|
|
ep++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
switch (ep->desc.bmAttributes) {
|
|
|
|
case USB_ENDPOINT_XFER_INT:
|
|
|
|
f->pipe = usb_rcvintpipe(dev,
|
2012-02-20 10:52:38 +07:00
|
|
|
ep->desc.bEndpointAddress);
|
2009-01-09 22:20:51 +07:00
|
|
|
f->usb_transfer_mode = USB_INT;
|
|
|
|
packet_size = le16_to_cpu(ep->desc.wMaxPacketSize);
|
|
|
|
break;
|
|
|
|
case USB_ENDPOINT_XFER_BULK:
|
|
|
|
if (ep_addr & 0x80)
|
|
|
|
f->pipe = usb_rcvbulkpipe(dev,
|
2012-02-20 10:52:38 +07:00
|
|
|
ep->desc.bEndpointAddress);
|
2009-01-09 22:20:51 +07:00
|
|
|
else
|
|
|
|
f->pipe = usb_sndbulkpipe(dev,
|
2012-02-20 10:52:38 +07:00
|
|
|
ep->desc.bEndpointAddress);
|
2009-01-09 22:20:51 +07:00
|
|
|
f->usb_transfer_mode = USB_BULK;
|
|
|
|
packet_size = le16_to_cpu(ep->desc.wMaxPacketSize);
|
|
|
|
break;
|
|
|
|
case USB_ENDPOINT_XFER_ISOC:
|
|
|
|
if (ep_addr & 0x80)
|
|
|
|
f->pipe = usb_rcvisocpipe(dev,
|
2012-02-20 10:52:38 +07:00
|
|
|
ep->desc.bEndpointAddress);
|
2009-01-09 22:20:51 +07:00
|
|
|
else
|
|
|
|
f->pipe = usb_sndisocpipe(dev,
|
2012-02-20 10:52:38 +07:00
|
|
|
ep->desc.bEndpointAddress);
|
2009-01-09 22:20:51 +07:00
|
|
|
f->usb_transfer_mode = USB_ISOC;
|
|
|
|
iso_packet_size = le16_to_cpu(ep->desc.wMaxPacketSize);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
f->pipe = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (f->pipe) {
|
|
|
|
f->fifonum = idx & 7;
|
|
|
|
f->hw = hw;
|
|
|
|
f->usb_packet_maxlen =
|
2012-02-20 10:52:38 +07:00
|
|
|
le16_to_cpu(ep->desc.wMaxPacketSize);
|
2009-01-09 22:20:51 +07:00
|
|
|
f->intervall = ep->desc.bInterval;
|
|
|
|
}
|
|
|
|
ep++;
|
|
|
|
}
|
|
|
|
hw->dev = dev; /* save device */
|
|
|
|
hw->if_used = ifnum; /* save used interface */
|
|
|
|
hw->alt_used = alt_used; /* and alternate config */
|
|
|
|
hw->ctrl_paksize = dev->descriptor.bMaxPacketSize0; /* control size */
|
|
|
|
hw->cfg_used = vcf[16]; /* store used config */
|
|
|
|
hw->vend_idx = vend_idx; /* store found vendor */
|
|
|
|
hw->packet_size = packet_size;
|
|
|
|
hw->iso_packet_size = iso_packet_size;
|
|
|
|
|
|
|
|
/* create the control pipes needed for register access */
|
|
|
|
hw->ctrl_in_pipe = usb_rcvctrlpipe(hw->dev, 0);
|
|
|
|
hw->ctrl_out_pipe = usb_sndctrlpipe(hw->dev, 0);
|
2012-07-13 05:59:46 +07:00
|
|
|
|
|
|
|
driver_info = (struct hfcsusb_vdata *)
|
|
|
|
hfcsusb_idtab[vend_idx].driver_info;
|
|
|
|
|
2009-01-09 22:20:51 +07:00
|
|
|
hw->ctrl_urb = usb_alloc_urb(0, GFP_KERNEL);
|
2012-07-13 05:59:46 +07:00
|
|
|
if (!hw->ctrl_urb) {
|
|
|
|
pr_warn("%s: No memory for control urb\n",
|
|
|
|
driver_info->vend_name);
|
|
|
|
kfree(hw);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
2009-01-09 22:20:51 +07:00
|
|
|
|
2012-07-13 05:59:46 +07:00
|
|
|
pr_info("%s: %s: detected \"%s\" (%s, if=%d alt=%d)\n",
|
|
|
|
hw->name, __func__, driver_info->vend_name,
|
|
|
|
conf_str[small_match], ifnum, alt_used);
|
2009-01-09 22:20:51 +07:00
|
|
|
|
|
|
|
if (setup_instance(hw, dev->dev.parent))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
hw->intf = intf;
|
|
|
|
usb_set_intfdata(hw->intf, hw);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* function called when an active device is removed */
|
|
|
|
static void
|
|
|
|
hfcsusb_disconnect(struct usb_interface *intf)
|
|
|
|
{
|
|
|
|
struct hfcsusb *hw = usb_get_intfdata(intf);
|
|
|
|
struct hfcsusb *next;
|
|
|
|
int cnt = 0;
|
|
|
|
|
|
|
|
printk(KERN_INFO "%s: device disconnected\n", hw->name);
|
|
|
|
|
|
|
|
handle_led(hw, LED_POWER_OFF);
|
|
|
|
release_hw(hw);
|
|
|
|
|
|
|
|
list_for_each_entry_safe(hw, next, &HFClist, list)
|
|
|
|
cnt++;
|
|
|
|
if (!cnt)
|
|
|
|
hfcsusb_cnt = 0;
|
|
|
|
|
|
|
|
usb_set_intfdata(intf, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct usb_driver hfcsusb_drv = {
|
|
|
|
.name = DRIVER_NAME,
|
|
|
|
.id_table = hfcsusb_idtab,
|
|
|
|
.probe = hfcsusb_probe,
|
|
|
|
.disconnect = hfcsusb_disconnect,
|
2012-04-24 00:08:51 +07:00
|
|
|
.disable_hub_initiated_lpm = 1,
|
2009-01-09 22:20:51 +07:00
|
|
|
};
|
|
|
|
|
2011-11-19 00:52:10 +07:00
|
|
|
module_usb_driver(hfcsusb_drv);
|