531 lines
14 KiB
C
531 lines
14 KiB
C
/*
|
|
* Driver for ST5481 USB ISDN modem
|
|
*
|
|
* Author Frode Isaksen
|
|
* Copyright 2001 by Frode Isaksen <fisaksen@bewan.com>
|
|
* 2001 by Kai Germaschewski <kai.germaschewski@gmx.de>
|
|
*
|
|
* This software may be used and distributed according to the terms
|
|
* of the GNU General Public License, incorporated herein by reference.
|
|
*
|
|
*/
|
|
|
|
#ifndef _ST5481_H_
|
|
#define _ST5481_H_
|
|
|
|
|
|
// USB IDs, the Product Id is in the range 0x4810-0x481F
|
|
|
|
#define ST_VENDOR_ID 0x0483
|
|
#define ST5481_PRODUCT_ID 0x4810
|
|
#define ST5481_PRODUCT_ID_MASK 0xFFF0
|
|
|
|
// ST5481 endpoints when using alternative setting 3 (2B+D).
|
|
// To get the endpoint address, OR with 0x80 for IN endpoints.
|
|
|
|
#define EP_CTRL 0x00U /* Control endpoint */
|
|
#define EP_INT 0x01U /* Interrupt endpoint */
|
|
#define EP_B1_OUT 0x02U /* B1 channel out */
|
|
#define EP_B1_IN 0x03U /* B1 channel in */
|
|
#define EP_B2_OUT 0x04U /* B2 channel out */
|
|
#define EP_B2_IN 0x05U /* B2 channel in */
|
|
#define EP_D_OUT 0x06U /* D channel out */
|
|
#define EP_D_IN 0x07U /* D channel in */
|
|
|
|
// Number of isochronous packets. With 20 packets we get
|
|
// 50 interrupts/sec for each endpoint.
|
|
|
|
#define NUM_ISO_PACKETS_D 20
|
|
#define NUM_ISO_PACKETS_B 20
|
|
|
|
// Size of each isochronous packet.
|
|
// In outgoing direction we need to match ISDN data rates:
|
|
// D: 2 bytes / msec -> 16 kbit / s
|
|
// B: 16 bytes / msec -> 64 kbit / s
|
|
#define SIZE_ISO_PACKETS_D_IN 16
|
|
#define SIZE_ISO_PACKETS_D_OUT 2
|
|
#define SIZE_ISO_PACKETS_B_IN 32
|
|
#define SIZE_ISO_PACKETS_B_OUT 8
|
|
|
|
// If we overrun/underrun, we send one packet with +/- 2 bytes
|
|
#define B_FLOW_ADJUST 2
|
|
|
|
// Registers that are written using vendor specific device request
|
|
// on endpoint 0.
|
|
|
|
#define LBA 0x02 /* S loopback */
|
|
#define SET_DEFAULT 0x06 /* Soft reset */
|
|
#define LBB 0x1D /* S maintenance loopback */
|
|
#define STT 0x1e /* S force transmission signals */
|
|
#define SDA_MIN 0x20 /* SDA-sin minimal value */
|
|
#define SDA_MAX 0x21 /* SDA-sin maximal value */
|
|
#define SDELAY_VALUE 0x22 /* Delay between Tx and Rx clock */
|
|
#define IN_D_COUNTER 0x36 /* D receive channel fifo counter */
|
|
#define OUT_D_COUNTER 0x37 /* D transmit channel fifo counter */
|
|
#define IN_B1_COUNTER 0x38 /* B1 receive channel fifo counter */
|
|
#define OUT_B1_COUNTER 0x39 /* B1 transmit channel fifo counter */
|
|
#define IN_B2_COUNTER 0x3a /* B2 receive channel fifo counter */
|
|
#define OUT_B2_COUNTER 0x3b /* B2 transmit channel fifo counter */
|
|
#define FFCTRL_IN_D 0x3C /* D receive channel fifo threshold low */
|
|
#define FFCTRH_IN_D 0x3D /* D receive channel fifo threshold high */
|
|
#define FFCTRL_OUT_D 0x3E /* D transmit channel fifo threshold low */
|
|
#define FFCTRH_OUT_D 0x3F /* D transmit channel fifo threshold high */
|
|
#define FFCTRL_IN_B1 0x40 /* B1 receive channel fifo threshold low */
|
|
#define FFCTRH_IN_B1 0x41 /* B1 receive channel fifo threshold high */
|
|
#define FFCTRL_OUT_B1 0x42 /* B1 transmit channel fifo threshold low */
|
|
#define FFCTRH_OUT_B1 0x43 /* B1 transmit channel fifo threshold high */
|
|
#define FFCTRL_IN_B2 0x44 /* B2 receive channel fifo threshold low */
|
|
#define FFCTRH_IN_B2 0x45 /* B2 receive channel fifo threshold high */
|
|
#define FFCTRL_OUT_B2 0x46 /* B2 transmit channel fifo threshold low */
|
|
#define FFCTRH_OUT_B2 0x47 /* B2 transmit channel fifo threshold high */
|
|
#define MPMSK 0x4A /* Multi purpose interrupt MASK register */
|
|
#define FFMSK_D 0x4c /* D fifo interrupt MASK register */
|
|
#define FFMSK_B1 0x4e /* B1 fifo interrupt MASK register */
|
|
#define FFMSK_B2 0x50 /* B2 fifo interrupt MASK register */
|
|
#define GPIO_DIR 0x52 /* GPIO pins direction registers */
|
|
#define GPIO_OUT 0x53 /* GPIO pins output register */
|
|
#define GPIO_IN 0x54 /* GPIO pins input register */
|
|
#define TXCI 0x56 /* CI command to be transmitted */
|
|
|
|
|
|
// Format of the interrupt packet received on endpoint 1:
|
|
//
|
|
// +--------+--------+--------+--------+--------+--------+
|
|
// !MPINT !FFINT_D !FFINT_B1!FFINT_B2!CCIST !GPIO_INT!
|
|
// +--------+--------+--------+--------+--------+--------+
|
|
|
|
// Offsets in the interrupt packet
|
|
|
|
#define MPINT 0
|
|
#define FFINT_D 1
|
|
#define FFINT_B1 2
|
|
#define FFINT_B2 3
|
|
#define CCIST 4
|
|
#define GPIO_INT 5
|
|
#define INT_PKT_SIZE 6
|
|
|
|
// MPINT
|
|
#define LSD_INT 0x80 /* S line activity detected */
|
|
#define RXCI_INT 0x40 /* Indicate primitive arrived */
|
|
#define DEN_INT 0x20 /* Signal enabling data out of D Tx fifo */
|
|
#define DCOLL_INT 0x10 /* D channel collision */
|
|
#define AMIVN_INT 0x04 /* AMI violation number reached 2 */
|
|
#define INFOI_INT 0x04 /* INFOi changed */
|
|
#define DRXON_INT 0x02 /* Reception channel active */
|
|
#define GPCHG_INT 0x01 /* GPIO pin value changed */
|
|
|
|
// FFINT_x
|
|
#define IN_OVERRUN 0x80 /* In fifo overrun */
|
|
#define OUT_UNDERRUN 0x40 /* Out fifo underrun */
|
|
#define IN_UP 0x20 /* In fifo thresholdh up-crossed */
|
|
#define IN_DOWN 0x10 /* In fifo thresholdl down-crossed */
|
|
#define OUT_UP 0x08 /* Out fifo thresholdh up-crossed */
|
|
#define OUT_DOWN 0x04 /* Out fifo thresholdl down-crossed */
|
|
#define IN_COUNTER_ZEROED 0x02 /* In down-counter reached 0 */
|
|
#define OUT_COUNTER_ZEROED 0x01 /* Out down-counter reached 0 */
|
|
|
|
#define ANY_REC_INT (IN_OVERRUN+IN_UP+IN_DOWN+IN_COUNTER_ZEROED)
|
|
#define ANY_XMIT_INT (OUT_UNDERRUN+OUT_UP+OUT_DOWN+OUT_COUNTER_ZEROED)
|
|
|
|
|
|
// Level 1 commands that are sent using the TXCI device request
|
|
#define ST5481_CMD_DR 0x0 /* Deactivation Request */
|
|
#define ST5481_CMD_RES 0x1 /* state machine RESet */
|
|
#define ST5481_CMD_TM1 0x2 /* Test Mode 1 */
|
|
#define ST5481_CMD_TM2 0x3 /* Test Mode 2 */
|
|
#define ST5481_CMD_PUP 0x7 /* Power UP */
|
|
#define ST5481_CMD_AR8 0x8 /* Activation Request class 1 */
|
|
#define ST5481_CMD_AR10 0x9 /* Activation Request class 2 */
|
|
#define ST5481_CMD_ARL 0xA /* Activation Request Loopback */
|
|
#define ST5481_CMD_PDN 0xF /* Power DoWn */
|
|
|
|
// Turn on/off the LEDs using the GPIO device request.
|
|
// To use the B LEDs, number_of_leds must be set to 4
|
|
#define B1_LED 0x10U
|
|
#define B2_LED 0x20U
|
|
#define GREEN_LED 0x40U
|
|
#define RED_LED 0x80U
|
|
|
|
// D channel out states
|
|
enum {
|
|
ST_DOUT_NONE,
|
|
|
|
ST_DOUT_SHORT_INIT,
|
|
ST_DOUT_SHORT_WAIT_DEN,
|
|
|
|
ST_DOUT_LONG_INIT,
|
|
ST_DOUT_LONG_WAIT_DEN,
|
|
ST_DOUT_NORMAL,
|
|
|
|
ST_DOUT_WAIT_FOR_UNDERRUN,
|
|
ST_DOUT_WAIT_FOR_NOT_BUSY,
|
|
ST_DOUT_WAIT_FOR_STOP,
|
|
ST_DOUT_WAIT_FOR_RESET,
|
|
};
|
|
|
|
#define DOUT_STATE_COUNT (ST_DOUT_WAIT_FOR_RESET + 1)
|
|
|
|
// D channel out events
|
|
enum {
|
|
EV_DOUT_START_XMIT,
|
|
EV_DOUT_COMPLETE,
|
|
EV_DOUT_DEN,
|
|
EV_DOUT_RESETED,
|
|
EV_DOUT_STOPPED,
|
|
EV_DOUT_COLL,
|
|
EV_DOUT_UNDERRUN,
|
|
};
|
|
|
|
#define DOUT_EVENT_COUNT (EV_DOUT_UNDERRUN + 1)
|
|
|
|
// ----------------------------------------------------------------------
|
|
|
|
enum {
|
|
ST_L1_F3,
|
|
ST_L1_F4,
|
|
ST_L1_F6,
|
|
ST_L1_F7,
|
|
ST_L1_F8,
|
|
};
|
|
|
|
#define L1_STATE_COUNT (ST_L1_F8+1)
|
|
|
|
// The first 16 entries match the Level 1 indications that
|
|
// are found at offset 4 (CCIST) in the interrupt packet
|
|
|
|
enum {
|
|
EV_IND_DP, // 0000 Deactivation Pending
|
|
EV_IND_1, // 0001
|
|
EV_IND_2, // 0010
|
|
EV_IND_3, // 0011
|
|
EV_IND_RSY, // 0100 ReSYnchronizing
|
|
EV_IND_5, // 0101
|
|
EV_IND_6, // 0110
|
|
EV_IND_7, // 0111
|
|
EV_IND_AP, // 1000 Activation Pending
|
|
EV_IND_9, // 1001
|
|
EV_IND_10, // 1010
|
|
EV_IND_11, // 1011
|
|
EV_IND_AI8, // 1100 Activation Indication class 8
|
|
EV_IND_AI10,// 1101 Activation Indication class 10
|
|
EV_IND_AIL, // 1110 Activation Indication Loopback
|
|
EV_IND_DI, // 1111 Deactivation Indication
|
|
EV_PH_ACTIVATE_REQ,
|
|
EV_PH_DEACTIVATE_REQ,
|
|
EV_TIMER3,
|
|
};
|
|
|
|
#define L1_EVENT_COUNT (EV_TIMER3 + 1)
|
|
|
|
#define ERR(format, arg...) \
|
|
printk(KERN_ERR "%s:%s: " format "\n" , __FILE__, __FUNCTION__ , ## arg)
|
|
|
|
#define WARN(format, arg...) \
|
|
printk(KERN_WARNING "%s:%s: " format "\n" , __FILE__, __FUNCTION__ , ## arg)
|
|
|
|
#define INFO(format, arg...) \
|
|
printk(KERN_INFO "%s:%s: " format "\n" , __FILE__, __FUNCTION__ , ## arg)
|
|
|
|
#include "isdnhdlc.h"
|
|
#include "fsm.h"
|
|
#include "hisax_if.h"
|
|
#include <linux/skbuff.h>
|
|
|
|
/* ======================================================================
|
|
* FIFO handling
|
|
*/
|
|
|
|
/* Generic FIFO structure */
|
|
struct fifo {
|
|
u_char r,w,count,size;
|
|
spinlock_t lock;
|
|
};
|
|
|
|
/*
|
|
* Init an FIFO
|
|
*/
|
|
static inline void fifo_init(struct fifo *fifo, int size)
|
|
{
|
|
fifo->r = fifo->w = fifo->count = 0;
|
|
fifo->size = size;
|
|
spin_lock_init(&fifo->lock);
|
|
}
|
|
|
|
/*
|
|
* Add an entry to the FIFO
|
|
*/
|
|
static inline int fifo_add(struct fifo *fifo)
|
|
{
|
|
unsigned long flags;
|
|
int index;
|
|
|
|
if (!fifo) {
|
|
return -1;
|
|
}
|
|
|
|
spin_lock_irqsave(&fifo->lock, flags);
|
|
if (fifo->count == fifo->size) {
|
|
// FIFO full
|
|
index = -1;
|
|
} else {
|
|
// Return index where to get the next data to add to the FIFO
|
|
index = fifo->w++ & (fifo->size-1);
|
|
fifo->count++;
|
|
}
|
|
spin_unlock_irqrestore(&fifo->lock, flags);
|
|
return index;
|
|
}
|
|
|
|
/*
|
|
* Remove an entry from the FIFO with the index returned.
|
|
*/
|
|
static inline int fifo_remove(struct fifo *fifo)
|
|
{
|
|
unsigned long flags;
|
|
int index;
|
|
|
|
if (!fifo) {
|
|
return -1;
|
|
}
|
|
|
|
spin_lock_irqsave(&fifo->lock, flags);
|
|
if (!fifo->count) {
|
|
// FIFO empty
|
|
index = -1;
|
|
} else {
|
|
// Return index where to get the next data from the FIFO
|
|
index = fifo->r++ & (fifo->size-1);
|
|
fifo->count--;
|
|
}
|
|
spin_unlock_irqrestore(&fifo->lock, flags);
|
|
|
|
return index;
|
|
}
|
|
|
|
/* ======================================================================
|
|
* control pipe
|
|
*/
|
|
typedef void (*ctrl_complete_t)(void *);
|
|
|
|
typedef struct ctrl_msg {
|
|
struct usb_ctrlrequest dr;
|
|
ctrl_complete_t complete;
|
|
void *context;
|
|
} ctrl_msg;
|
|
|
|
/* FIFO of ctrl messages waiting to be sent */
|
|
#define MAX_EP0_MSG 16
|
|
struct ctrl_msg_fifo {
|
|
struct fifo f;
|
|
struct ctrl_msg data[MAX_EP0_MSG];
|
|
};
|
|
|
|
#define MAX_DFRAME_LEN_L1 300
|
|
#define HSCX_BUFMAX 4096
|
|
|
|
struct st5481_ctrl {
|
|
struct ctrl_msg_fifo msg_fifo;
|
|
unsigned long busy;
|
|
struct urb *urb;
|
|
};
|
|
|
|
struct st5481_intr {
|
|
// struct evt_fifo evt_fifo;
|
|
struct urb *urb;
|
|
};
|
|
|
|
struct st5481_d_out {
|
|
struct isdnhdlc_vars hdlc_state;
|
|
struct urb *urb[2]; /* double buffering */
|
|
unsigned long busy;
|
|
struct sk_buff *tx_skb;
|
|
struct FsmInst fsm;
|
|
};
|
|
|
|
struct st5481_b_out {
|
|
struct isdnhdlc_vars hdlc_state;
|
|
struct urb *urb[2]; /* double buffering */
|
|
u_char flow_event;
|
|
u_long busy;
|
|
struct sk_buff *tx_skb;
|
|
};
|
|
|
|
struct st5481_in {
|
|
struct isdnhdlc_vars hdlc_state;
|
|
struct urb *urb[2]; /* double buffering */
|
|
int mode;
|
|
int bufsize;
|
|
unsigned int num_packets;
|
|
unsigned int packet_size;
|
|
unsigned char ep, counter;
|
|
unsigned char *rcvbuf;
|
|
struct st5481_adapter *adapter;
|
|
struct hisax_if *hisax_if;
|
|
};
|
|
|
|
int st5481_setup_in(struct st5481_in *in);
|
|
void st5481_release_in(struct st5481_in *in);
|
|
void st5481_in_mode(struct st5481_in *in, int mode);
|
|
|
|
struct st5481_bcs {
|
|
struct hisax_b_if b_if;
|
|
struct st5481_adapter *adapter;
|
|
struct st5481_in b_in;
|
|
struct st5481_b_out b_out;
|
|
int channel;
|
|
int mode;
|
|
};
|
|
|
|
struct st5481_adapter {
|
|
struct list_head list;
|
|
int number_of_leds;
|
|
struct usb_device *usb_dev;
|
|
struct hisax_d_if hisax_d_if;
|
|
|
|
struct st5481_ctrl ctrl;
|
|
struct st5481_intr intr;
|
|
struct st5481_in d_in;
|
|
struct st5481_d_out d_out;
|
|
|
|
unsigned char leds;
|
|
unsigned int led_counter;
|
|
|
|
unsigned long event;
|
|
|
|
struct FsmInst l1m;
|
|
struct FsmTimer timer;
|
|
|
|
struct st5481_bcs bcs[2];
|
|
};
|
|
|
|
#define TIMER3_VALUE 7000
|
|
|
|
/* ======================================================================
|
|
*
|
|
*/
|
|
|
|
/*
|
|
* Submit an URB with error reporting. This is a macro so
|
|
* the __FUNCTION__ returns the caller function name.
|
|
*/
|
|
#define SUBMIT_URB(urb, mem_flags) \
|
|
({ \
|
|
int status; \
|
|
if ((status = usb_submit_urb(urb, mem_flags)) < 0) { \
|
|
WARN("usb_submit_urb failed,status=%d", status); \
|
|
} \
|
|
status; \
|
|
})
|
|
|
|
/*
|
|
* USB double buffering, return the URB index (0 or 1).
|
|
*/
|
|
static inline int get_buf_nr(struct urb *urbs[], struct urb *urb)
|
|
{
|
|
return (urbs[0]==urb ? 0 : 1);
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
/* B Channel */
|
|
|
|
int st5481_setup_b(struct st5481_bcs *bcs);
|
|
void st5481_release_b(struct st5481_bcs *bcs);
|
|
void st5481_d_l2l1(struct hisax_if *hisax_d_if, int pr, void *arg);
|
|
|
|
/* D Channel */
|
|
|
|
int st5481_setup_d(struct st5481_adapter *adapter);
|
|
void st5481_release_d(struct st5481_adapter *adapter);
|
|
void st5481_b_l2l1(struct hisax_if *b_if, int pr, void *arg);
|
|
int st5481_d_init(void);
|
|
void st5481_d_exit(void);
|
|
|
|
/* USB */
|
|
void st5481_ph_command(struct st5481_adapter *adapter, unsigned int command);
|
|
int st5481_setup_isocpipes(struct urb* urb[2], struct usb_device *dev,
|
|
unsigned int pipe, int num_packets,
|
|
int packet_size, int buf_size,
|
|
usb_complete_t complete, void *context);
|
|
void st5481_release_isocpipes(struct urb* urb[2]);
|
|
|
|
void st5481_usb_pipe_reset(struct st5481_adapter *adapter,
|
|
u_char pipe, ctrl_complete_t complete, void *context);
|
|
void st5481_usb_device_ctrl_msg(struct st5481_adapter *adapter,
|
|
u8 request, u16 value,
|
|
ctrl_complete_t complete, void *context);
|
|
int st5481_setup_usb(struct st5481_adapter *adapter);
|
|
void st5481_release_usb(struct st5481_adapter *adapter);
|
|
void st5481_start(struct st5481_adapter *adapter);
|
|
void st5481_stop(struct st5481_adapter *adapter);
|
|
|
|
// ----------------------------------------------------------------------
|
|
// debugging macros
|
|
|
|
#define __debug_variable st5481_debug
|
|
#include "hisax_debug.h"
|
|
|
|
extern int st5481_debug;
|
|
|
|
#ifdef CONFIG_HISAX_DEBUG
|
|
|
|
#define DBG_ISO_PACKET(level,urb) \
|
|
if (level & __debug_variable) dump_iso_packet(__FUNCTION__,urb)
|
|
|
|
static void __attribute__((unused))
|
|
dump_iso_packet(const char *name, struct urb *urb)
|
|
{
|
|
int i,j;
|
|
int len,ofs;
|
|
u_char *data;
|
|
|
|
printk(KERN_DEBUG "%s: packets=%d,errors=%d\n",
|
|
name,urb->number_of_packets,urb->error_count);
|
|
for (i = 0; i < urb->number_of_packets; ++i) {
|
|
if (urb->pipe & USB_DIR_IN) {
|
|
len = urb->iso_frame_desc[i].actual_length;
|
|
} else {
|
|
len = urb->iso_frame_desc[i].length;
|
|
}
|
|
ofs = urb->iso_frame_desc[i].offset;
|
|
printk(KERN_DEBUG "len=%.2d,ofs=%.3d ",len,ofs);
|
|
if (len) {
|
|
data = urb->transfer_buffer+ofs;
|
|
for (j=0; j < len; j++) {
|
|
printk ("%.2x", data[j]);
|
|
}
|
|
}
|
|
printk("\n");
|
|
}
|
|
}
|
|
|
|
static inline const char *ST5481_CMD_string(int evt)
|
|
{
|
|
static char s[16];
|
|
|
|
switch (evt) {
|
|
case ST5481_CMD_DR: return "DR";
|
|
case ST5481_CMD_RES: return "RES";
|
|
case ST5481_CMD_TM1: return "TM1";
|
|
case ST5481_CMD_TM2: return "TM2";
|
|
case ST5481_CMD_PUP: return "PUP";
|
|
case ST5481_CMD_AR8: return "AR8";
|
|
case ST5481_CMD_AR10: return "AR10";
|
|
case ST5481_CMD_ARL: return "ARL";
|
|
case ST5481_CMD_PDN: return "PDN";
|
|
};
|
|
|
|
sprintf(s,"0x%x",evt);
|
|
return s;
|
|
}
|
|
|
|
#else
|
|
|
|
#define DBG_ISO_PACKET(level,urb) do {} while (0)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif
|