forked from Minki/linux
[PATCH] USB ATM: new usbatm core
Rework the core usbatm code: minidrivers (i.e. drivers for particular modems) now register themselves with the usbatm core, supplying methods for binding/unbinding etc. The design was inspired by usb-serial and usbnet. At the same time, more common code from the speedtch and cxacru (patch 3/5) drivers was generalized and moved into the core. The transmission and reception parts have been unified and simplified. Since this is a major change and I don't like underscores in file names, usb_atm.[ch] has been renamed usbatm.[ch]. Many thanks to Roman Kagan, who did a lot of the coding. Signed-off-by: Duncan Sands <baldrick@free.fr> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
This commit is contained in:
parent
d49d431744
commit
c59bba75fa
File diff suppressed because it is too large
Load Diff
@ -1,176 +0,0 @@
|
||||
/******************************************************************************
|
||||
* usb_atm.h - Generic USB xDSL driver core
|
||||
*
|
||||
* Copyright (C) 2001, Alcatel
|
||||
* Copyright (C) 2003, Duncan Sands, SolNegro, Josep Comas
|
||||
* Copyright (C) 2004, David Woodhouse
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify it
|
||||
* under the terms of the GNU General Public License as published by the Free
|
||||
* Software Foundation; either version 2 of the License, or (at your option)
|
||||
* any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||
* more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License along with
|
||||
* this program; if not, write to the Free Software Foundation, Inc., 59
|
||||
* Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
||||
*
|
||||
******************************************************************************/
|
||||
|
||||
#include <linux/config.h>
|
||||
#include <linux/list.h>
|
||||
#include <linux/kref.h>
|
||||
#include <linux/atm.h>
|
||||
#include <linux/atmdev.h>
|
||||
#include <asm/semaphore.h>
|
||||
|
||||
/*
|
||||
#define DEBUG
|
||||
#define VERBOSE_DEBUG
|
||||
*/
|
||||
|
||||
#if !defined (DEBUG) && defined (CONFIG_USB_DEBUG)
|
||||
# define DEBUG
|
||||
#endif
|
||||
|
||||
#include <linux/usb.h>
|
||||
|
||||
#ifdef DEBUG
|
||||
#define UDSL_ASSERT(x) BUG_ON(!(x))
|
||||
#else
|
||||
#define UDSL_ASSERT(x) do { if (!(x)) warn("failed assertion '" #x "' at line %d", __LINE__); } while(0)
|
||||
#endif
|
||||
|
||||
#define UDSL_MAX_RCV_URBS 4
|
||||
#define UDSL_MAX_SND_URBS 4
|
||||
#define UDSL_MAX_RCV_BUFS 8
|
||||
#define UDSL_MAX_SND_BUFS 8
|
||||
#define UDSL_MAX_RCV_BUF_SIZE 1024 /* ATM cells */
|
||||
#define UDSL_MAX_SND_BUF_SIZE 1024 /* ATM cells */
|
||||
#define UDSL_DEFAULT_RCV_URBS 2
|
||||
#define UDSL_DEFAULT_SND_URBS 2
|
||||
#define UDSL_DEFAULT_RCV_BUFS 4
|
||||
#define UDSL_DEFAULT_SND_BUFS 4
|
||||
#define UDSL_DEFAULT_RCV_BUF_SIZE 64 /* ATM cells */
|
||||
#define UDSL_DEFAULT_SND_BUF_SIZE 64 /* ATM cells */
|
||||
|
||||
#define ATM_CELL_HEADER (ATM_CELL_SIZE - ATM_CELL_PAYLOAD)
|
||||
#define UDSL_NUM_CELLS(x) (((x) + ATM_AAL5_TRAILER + ATM_CELL_PAYLOAD - 1) / ATM_CELL_PAYLOAD)
|
||||
|
||||
/* receive */
|
||||
|
||||
struct udsl_receive_buffer {
|
||||
struct list_head list;
|
||||
unsigned char *base;
|
||||
unsigned int filled_cells;
|
||||
};
|
||||
|
||||
struct udsl_receiver {
|
||||
struct list_head list;
|
||||
struct udsl_receive_buffer *buffer;
|
||||
struct urb *urb;
|
||||
struct udsl_instance_data *instance;
|
||||
};
|
||||
|
||||
struct udsl_vcc_data {
|
||||
/* vpi/vci lookup */
|
||||
struct list_head list;
|
||||
short vpi;
|
||||
int vci;
|
||||
struct atm_vcc *vcc;
|
||||
|
||||
/* raw cell reassembly */
|
||||
struct sk_buff *sarb;
|
||||
};
|
||||
|
||||
/* send */
|
||||
|
||||
struct udsl_send_buffer {
|
||||
struct list_head list;
|
||||
unsigned char *base;
|
||||
unsigned char *free_start;
|
||||
unsigned int free_cells;
|
||||
};
|
||||
|
||||
struct udsl_sender {
|
||||
struct list_head list;
|
||||
struct udsl_send_buffer *buffer;
|
||||
struct urb *urb;
|
||||
struct udsl_instance_data *instance;
|
||||
};
|
||||
|
||||
struct udsl_control {
|
||||
struct atm_skb_data atm_data;
|
||||
unsigned int num_cells;
|
||||
unsigned int num_entire;
|
||||
unsigned int pdu_padding;
|
||||
unsigned char aal5_trailer[ATM_AAL5_TRAILER];
|
||||
};
|
||||
|
||||
#define UDSL_SKB(x) ((struct udsl_control *)(x)->cb)
|
||||
|
||||
/* main driver data */
|
||||
|
||||
enum udsl_status {
|
||||
UDSL_NO_FIRMWARE,
|
||||
UDSL_LOADING_FIRMWARE,
|
||||
UDSL_LOADED_FIRMWARE
|
||||
};
|
||||
|
||||
struct udsl_instance_data {
|
||||
struct kref refcount;
|
||||
struct semaphore serialize;
|
||||
|
||||
/* USB device part */
|
||||
struct usb_device *usb_dev;
|
||||
char description[64];
|
||||
int data_endpoint;
|
||||
int snd_padding;
|
||||
int rcv_padding;
|
||||
const char *driver_name;
|
||||
|
||||
/* ATM device part */
|
||||
struct atm_dev *atm_dev;
|
||||
struct list_head vcc_list;
|
||||
|
||||
/* firmware */
|
||||
int (*firmware_wait) (struct udsl_instance_data *);
|
||||
enum udsl_status status;
|
||||
wait_queue_head_t firmware_waiters;
|
||||
|
||||
/* receive */
|
||||
struct udsl_receiver receivers[UDSL_MAX_RCV_URBS];
|
||||
struct udsl_receive_buffer receive_buffers[UDSL_MAX_RCV_BUFS];
|
||||
|
||||
spinlock_t receive_lock;
|
||||
struct list_head spare_receivers;
|
||||
struct list_head filled_receive_buffers;
|
||||
|
||||
struct tasklet_struct receive_tasklet;
|
||||
struct list_head spare_receive_buffers;
|
||||
|
||||
/* send */
|
||||
struct udsl_sender senders[UDSL_MAX_SND_URBS];
|
||||
struct udsl_send_buffer send_buffers[UDSL_MAX_SND_BUFS];
|
||||
|
||||
struct sk_buff_head sndqueue;
|
||||
|
||||
spinlock_t send_lock;
|
||||
struct list_head spare_senders;
|
||||
struct list_head spare_send_buffers;
|
||||
|
||||
struct tasklet_struct send_tasklet;
|
||||
struct sk_buff *current_skb; /* being emptied */
|
||||
struct udsl_send_buffer *current_buffer; /* being filled */
|
||||
struct list_head filled_send_buffers;
|
||||
};
|
||||
|
||||
extern int udsl_instance_setup(struct usb_device *dev,
|
||||
struct udsl_instance_data *instance);
|
||||
extern void udsl_instance_disconnect(struct udsl_instance_data *instance);
|
||||
extern void udsl_get_instance(struct udsl_instance_data *instance);
|
||||
extern void udsl_put_instance(struct udsl_instance_data *instance);
|
1231
drivers/usb/atm/usbatm.c
Normal file
1231
drivers/usb/atm/usbatm.c
Normal file
File diff suppressed because it is too large
Load Diff
183
drivers/usb/atm/usbatm.h
Normal file
183
drivers/usb/atm/usbatm.h
Normal file
@ -0,0 +1,183 @@
|
||||
/******************************************************************************
|
||||
* usbatm.h - Generic USB xDSL driver core
|
||||
*
|
||||
* Copyright (C) 2001, Alcatel
|
||||
* Copyright (C) 2003, Duncan Sands, SolNegro, Josep Comas
|
||||
* Copyright (C) 2004, David Woodhouse
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify it
|
||||
* under the terms of the GNU General Public License as published by the Free
|
||||
* Software Foundation; either version 2 of the License, or (at your option)
|
||||
* any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||
* more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License along with
|
||||
* this program; if not, write to the Free Software Foundation, Inc., 59
|
||||
* Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
||||
*
|
||||
******************************************************************************/
|
||||
|
||||
#ifndef _USBATM_H_
|
||||
#define _USBATM_H_
|
||||
|
||||
#include <linux/config.h>
|
||||
|
||||
/*
|
||||
#define DEBUG
|
||||
#define VERBOSE_DEBUG
|
||||
*/
|
||||
|
||||
#if !defined (DEBUG) && defined (CONFIG_USB_DEBUG)
|
||||
# define DEBUG
|
||||
#endif
|
||||
|
||||
#include <asm/semaphore.h>
|
||||
#include <linux/atm.h>
|
||||
#include <linux/atmdev.h>
|
||||
#include <linux/completion.h>
|
||||
#include <linux/device.h>
|
||||
#include <linux/kref.h>
|
||||
#include <linux/list.h>
|
||||
#include <linux/stringify.h>
|
||||
#include <linux/usb.h>
|
||||
|
||||
#ifdef DEBUG
|
||||
#define UDSL_ASSERT(x) BUG_ON(!(x))
|
||||
#else
|
||||
#define UDSL_ASSERT(x) do { if (!(x)) warn("failed assertion '%s' at line %d", __stringify(x), __LINE__); } while(0)
|
||||
#endif
|
||||
|
||||
#define usb_err(instance, format, arg...) \
|
||||
dev_err(&(instance)->usb_intf->dev , format , ## arg)
|
||||
#define usb_info(instance, format, arg...) \
|
||||
dev_info(&(instance)->usb_intf->dev , format , ## arg)
|
||||
#define usb_warn(instance, format, arg...) \
|
||||
dev_warn(&(instance)->usb_intf->dev , format , ## arg)
|
||||
#define usb_dbg(instance, format, arg...) \
|
||||
dev_dbg(&(instance)->usb_intf->dev , format , ## arg)
|
||||
|
||||
/* FIXME: move to dev_* once ATM is driver model aware */
|
||||
#define atm_printk(level, instance, format, arg...) \
|
||||
printk(level "ATM dev %d: " format , (instance)->atm_dev->number, ## arg)
|
||||
|
||||
#define atm_err(instance, format, arg...) \
|
||||
atm_printk(KERN_ERR, instance , format , ## arg)
|
||||
#define atm_info(instance, format, arg...) \
|
||||
atm_printk(KERN_INFO, instance , format , ## arg)
|
||||
#define atm_warn(instance, format, arg...) \
|
||||
atm_printk(KERN_WARNING, instance , format , ## arg)
|
||||
#ifdef DEBUG
|
||||
#define atm_dbg(instance, format, arg...) \
|
||||
atm_printk(KERN_DEBUG, instance , format , ## arg)
|
||||
#else
|
||||
#define atm_dbg(instance, format, arg...) \
|
||||
do {} while (0)
|
||||
#endif
|
||||
|
||||
|
||||
/* mini driver */
|
||||
|
||||
struct usbatm_data;
|
||||
|
||||
/*
|
||||
* Assuming all methods exist and succeed, they are called in this order:
|
||||
*
|
||||
* bind, heavy_init, atm_start, ..., atm_stop, unbind
|
||||
*/
|
||||
|
||||
struct usbatm_driver {
|
||||
struct module *owner;
|
||||
|
||||
const char *driver_name;
|
||||
|
||||
/*
|
||||
* init device ... can sleep, or cause probe() failure. Drivers with a heavy_init
|
||||
* method can avoid having it called by setting need_heavy_init to zero.
|
||||
*/
|
||||
int (*bind) (struct usbatm_data *, struct usb_interface *,
|
||||
const struct usb_device_id *id, int *need_heavy_init);
|
||||
|
||||
/* additional device initialization that is too slow to be done in probe() */
|
||||
int (*heavy_init) (struct usbatm_data *, struct usb_interface *);
|
||||
|
||||
/* cleanup device ... can sleep, but can't fail */
|
||||
void (*unbind) (struct usbatm_data *, struct usb_interface *);
|
||||
|
||||
/* init ATM device ... can sleep, or cause ATM initialization failure */
|
||||
int (*atm_start) (struct usbatm_data *, struct atm_dev *);
|
||||
|
||||
/* cleanup ATM device ... can sleep, but can't fail */
|
||||
void (*atm_stop) (struct usbatm_data *, struct atm_dev *);
|
||||
|
||||
int in; /* rx endpoint */
|
||||
int out; /* tx endpoint */
|
||||
|
||||
unsigned rx_padding;
|
||||
unsigned tx_padding;
|
||||
};
|
||||
|
||||
extern int usbatm_usb_probe(struct usb_interface *intf, const struct usb_device_id *id,
|
||||
struct usbatm_driver *driver);
|
||||
extern void usbatm_usb_disconnect(struct usb_interface *intf);
|
||||
|
||||
|
||||
struct usbatm_channel {
|
||||
int endpoint; /* usb pipe */
|
||||
unsigned int stride; /* ATM cell size + padding */
|
||||
unsigned int buf_size; /* urb buffer size */
|
||||
spinlock_t lock;
|
||||
struct list_head list;
|
||||
struct tasklet_struct tasklet;
|
||||
struct timer_list delay;
|
||||
struct usbatm_data *usbatm;
|
||||
};
|
||||
|
||||
/* main driver data */
|
||||
|
||||
struct usbatm_data {
|
||||
/******************
|
||||
* public fields *
|
||||
******************/
|
||||
|
||||
/* mini driver */
|
||||
struct usbatm_driver *driver;
|
||||
void *driver_data;
|
||||
char driver_name[16];
|
||||
|
||||
/* USB device */
|
||||
struct usb_device *usb_dev;
|
||||
struct usb_interface *usb_intf;
|
||||
char description[64];
|
||||
|
||||
/* ATM device */
|
||||
struct atm_dev *atm_dev;
|
||||
|
||||
/********************************
|
||||
* private fields - do not use *
|
||||
********************************/
|
||||
|
||||
struct kref refcount;
|
||||
struct semaphore serialize;
|
||||
|
||||
/* heavy init */
|
||||
int thread_pid;
|
||||
struct completion thread_started;
|
||||
struct completion thread_exited;
|
||||
|
||||
/* ATM device */
|
||||
struct list_head vcc_list;
|
||||
|
||||
struct usbatm_channel rx_channel;
|
||||
struct usbatm_channel tx_channel;
|
||||
|
||||
struct sk_buff_head sndqueue;
|
||||
struct sk_buff *current_skb; /* being emptied */
|
||||
|
||||
struct urb *urbs[0];
|
||||
};
|
||||
|
||||
#endif /* _USBATM_H_ */
|
Loading…
Reference in New Issue
Block a user