cregit-Linux how code gets into the kernel

Release 4.11 drivers/isdn/gigaset/bas-gigaset.c

/*
 * USB driver for Gigaset 307x base via direct USB connection.
 *
 * Copyright (c) 2001 by Hansjoerg Lipp <hjlipp@web.de>,
 *                       Tilman Schmidt <tilman@imap.cc>,
 *                       Stefan Eilers.
 *
 * =====================================================================
 *      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.
 * =====================================================================
 */

#include "gigaset.h"
#include <linux/usb.h>
#include <linux/module.h>
#include <linux/moduleparam.h>

/* Version Information */

#define DRIVER_AUTHOR "Tilman Schmidt <tilman@imap.cc>, Hansjoerg Lipp <hjlipp@web.de>, Stefan Eilers"

#define DRIVER_DESC "USB Driver for Gigaset 307x"


/* Module parameters */


static int startmode = SM_ISDN;

static int cidmode = 1;

module_param(startmode, int, S_IRUGO);
module_param(cidmode, int, S_IRUGO);
MODULE_PARM_DESC(startmode, "start in isdn4linux mode");
MODULE_PARM_DESC(cidmode, "Call-ID mode");


#define GIGASET_MINORS     1

#define GIGASET_MINOR      16

#define GIGASET_MODULENAME "bas_gigaset"

#define GIGASET_DEVNAME    "ttyGB"

/* length limit according to Siemens 3070usb-protokoll.doc ch. 2.1 */

#define IF_WRITEBUF 264

/* interrupt pipe message size according to ibid. ch. 2.2 */

#define IP_MSGSIZE 3

/* Values for the Gigaset 307x */

#define USB_GIGA_VENDOR_ID      0x0681

#define USB_3070_PRODUCT_ID     0x0001

#define USB_3075_PRODUCT_ID     0x0002

#define USB_SX303_PRODUCT_ID    0x0021

#define USB_SX353_PRODUCT_ID    0x0022

/* table of devices that work with this driver */

static const struct usb_device_id gigaset_table[] = {
	{ USB_DEVICE(USB_GIGA_VENDOR_ID, USB_3070_PRODUCT_ID) },
	{ USB_DEVICE(USB_GIGA_VENDOR_ID, USB_3075_PRODUCT_ID) },
	{ USB_DEVICE(USB_GIGA_VENDOR_ID, USB_SX303_PRODUCT_ID) },
	{ USB_DEVICE(USB_GIGA_VENDOR_ID, USB_SX353_PRODUCT_ID) },
	{ } /* Terminating entry */
};

MODULE_DEVICE_TABLE(usb, gigaset_table);

/*======================= local function prototypes ==========================*/

/* function called if a new device belonging to this driver is connected */
static int gigaset_probe(struct usb_interface *interface,
			 const struct usb_device_id *id);

/* Function will be called if the device is unplugged */
static void gigaset_disconnect(struct usb_interface *interface);

/* functions called before/after suspend */
static int gigaset_suspend(struct usb_interface *intf, pm_message_t message);
static int gigaset_resume(struct usb_interface *intf);

/* functions called before/after device reset */
static int gigaset_pre_reset(struct usb_interface *intf);
static int gigaset_post_reset(struct usb_interface *intf);

static int atread_submit(struct cardstate *, int);
static void stopurbs(struct bas_bc_state *);
static int req_submit(struct bc_state *, int, int, int);
static int atwrite_submit(struct cardstate *, unsigned char *, int);
static int start_cbsend(struct cardstate *);

/*============================================================================*/


struct bas_cardstate {
	
struct usb_device	*udev;		/* USB device pointer */
	
struct usb_interface	*interface;	/* interface for this device */
	
unsigned char		minor;		/* starting minor number */

	
struct urb		*urb_ctrl;	/* control pipe default URB */
	
struct usb_ctrlrequest	dr_ctrl;
	
struct timer_list	timer_ctrl;	/* control request timeout */
	
int			retry_ctrl;

	
struct timer_list	timer_atrdy;	/* AT command ready timeout */
	
struct urb		*urb_cmd_out;	/* for sending AT commands */
	
struct usb_ctrlrequest	dr_cmd_out;
	
int			retry_cmd_out;

	
struct urb		*urb_cmd_in;	/* for receiving AT replies */
	
struct usb_ctrlrequest	dr_cmd_in;
	
struct timer_list	timer_cmd_in;	/* receive request timeout */
	
unsigned char		*rcvbuf;	/* AT reply receive buffer */

	
struct urb		*urb_int_in;	/* URB for interrupt pipe */
	
unsigned char		*int_in_buf;
	
struct work_struct	int_in_wq;	/* for usb_clear_halt() */
	
struct timer_list	timer_int_in;	/* int read retry delay */
	
int			retry_int_in;

	
spinlock_t		lock;		/* locks all following */
	
int			basstate;	/* bitmap (BS_*) */
	
int			pending;	/* uncompleted base request */
	
wait_queue_head_t	waitqueue;
	
int			rcvbuf_size;	/* size of AT receive buffer */
						/* 0: no receive in progress */
	
int			retry_cmd_in;	/* receive req retry count */
};

/* status of direct USB connection to 307x base (bits in basstate) */

#define BS_ATOPEN	0x001	
/* AT channel open */

#define BS_B1OPEN	0x002	
/* B channel 1 open */

#define BS_B2OPEN	0x004	
/* B channel 2 open */

#define BS_ATREADY	0x008	
/* base ready for AT command */

#define BS_INIT		0x010	
/* base has signalled INIT_OK */

#define BS_ATTIMER	0x020	
/* waiting for HD_READY_SEND_ATDATA */

#define BS_ATRDPEND	0x040	
/* urb_cmd_in in use */

#define BS_ATWRPEND	0x080	
/* urb_cmd_out in use */

#define BS_SUSPEND	0x100	
/* USB port suspended */

#define BS_RESETTING	0x200	
/* waiting for HD_RESET_INTERRUPT_PIPE_ACK */



static struct gigaset_driver *driver;

/* usb specific object needed to register this driver with the usb subsystem */

static struct usb_driver gigaset_usb_driver = {
	.name =         GIGASET_MODULENAME,
	.probe =        gigaset_probe,
	.disconnect =   gigaset_disconnect,
	.id_table =     gigaset_table,
	.suspend =	gigaset_suspend,
	.resume =	gigaset_resume,
	.reset_resume =	gigaset_post_reset,
	.pre_reset =	gigaset_pre_reset,
	.post_reset =	gigaset_post_reset,
	.disable_hub_initiated_lpm = 1,
};

/* get message text for usb_submit_urb return code
 */

static char *get_usb_rcmsg(int rc) { static char unkmsg[28]; switch (rc) { case 0: return "success"; case -ENOMEM: return "out of memory"; case -ENODEV: return "device not present"; case -ENOENT: return "endpoint not present"; case -ENXIO: return "URB type not supported"; case -EINVAL: return "invalid argument"; case -EAGAIN: return "start frame too early or too much scheduled"; case -EFBIG: return "too many isoc frames requested"; case -EPIPE: return "endpoint stalled"; case -EMSGSIZE: return "invalid packet size"; case -ENOSPC: return "would overcommit USB bandwidth"; case -ESHUTDOWN: return "device shut down"; case -EPERM: return "reject flag set"; case -EHOSTUNREACH: return "device suspended"; default: snprintf(unkmsg, sizeof(unkmsg), "unknown error %d", rc); return unkmsg; } }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt8057.97%266.67%
Hansjoerg Lipp5842.03%133.33%
Total138100.00%3100.00%

/* get message text for USB status code */
static char *get_usb_statmsg(int status) { static char unkmsg[28]; switch (status) { case 0: return "success"; case -ENOENT: return "unlinked (sync)"; case -EINPROGRESS: return "URB still pending"; case -EPROTO: return "bitstuff error, timeout, or unknown USB error"; case -EILSEQ: return "CRC mismatch, timeout, or unknown USB error"; case -ETIME: return "USB response timeout"; case -EPIPE: return "endpoint stalled"; case -ECOMM: return "IN buffer overrun"; case -ENOSR: return "OUT buffer underrun"; case -EOVERFLOW: return "endpoint babble"; case -EREMOTEIO: return "short packet"; case -ENODEV: return "device removed"; case -EXDEV: return "partial isoc transfer"; case -EINVAL: return "ISO madness"; case -ECONNRESET: return "unlinked (async)"; case -ESHUTDOWN: return "device shut down"; default: snprintf(unkmsg, sizeof(unkmsg), "unknown status %d", status); return unkmsg; } }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp7750.66%125.00%
Tilman Schmidt7448.68%250.00%
Pete Zaitcev10.66%125.00%
Total152100.00%4100.00%

/* usb_pipetype_str * retrieve string representation of USB pipe type */
static inline char *usb_pipetype_str(int pipe) { if (usb_pipeisoc(pipe)) return "Isoc"; if (usb_pipeint(pipe)) return "Int"; if (usb_pipecontrol(pipe)) return "Ctrl"; if (usb_pipebulk(pipe)) return "Bulk"; return "?"; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp54100.00%1100.00%
Total54100.00%1100.00%

/* dump_urb * write content of URB to syslog for debugging */
static inline void dump_urb(enum debuglevel level, const char *tag, struct urb *urb) { #ifdef CONFIG_GIGASET_DEBUG int i; gig_dbg(level, "%s urb(0x%08lx)->{", tag, (unsigned long) urb); if (urb) { gig_dbg(level, " dev=0x%08lx, pipe=%s:EP%d/DV%d:%s, " "hcpriv=0x%08lx, transfer_flags=0x%x,", (unsigned long) urb->dev, usb_pipetype_str(urb->pipe), usb_pipeendpoint(urb->pipe), usb_pipedevice(urb->pipe), usb_pipein(urb->pipe) ? "in" : "out", (unsigned long) urb->hcpriv, urb->transfer_flags); gig_dbg(level, " transfer_buffer=0x%08lx[%d], actual_length=%d, " "setup_packet=0x%08lx,", (unsigned long) urb->transfer_buffer, urb->transfer_buffer_length, urb->actual_length, (unsigned long) urb->setup_packet); gig_dbg(level, " start_frame=%d, number_of_packets=%d, interval=%d, " "error_count=%d,", urb->start_frame, urb->number_of_packets, urb->interval, urb->error_count); gig_dbg(level, " context=0x%08lx, complete=0x%08lx, " "iso_frame_desc[]={", (unsigned long) urb->context, (unsigned long) urb->complete); for (i = 0; i < urb->number_of_packets; i++) { struct usb_iso_packet_descriptor *pifd = &urb->iso_frame_desc[i]; gig_dbg(level, " {offset=%u, length=%u, actual_length=%u, " "status=%u}", pifd->offset, pifd->length, pifd->actual_length, pifd->status); } } gig_dbg(level, "}}"); #endif }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp24095.62%125.00%
Tilman Schmidt103.98%250.00%
Andrew Morton10.40%125.00%
Total251100.00%4100.00%

/* read/set modem control bits etc. (m10x only) */
static int gigaset_set_modem_ctrl(struct cardstate *cs, unsigned old_state, unsigned new_state) { return -EINVAL; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp21100.00%1100.00%
Total21100.00%1100.00%


static int gigaset_baud_rate(struct cardstate *cs, unsigned cflag) { return -EINVAL; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp18100.00%1100.00%
Total18100.00%1100.00%


static int gigaset_set_line_ctrl(struct cardstate *cs, unsigned cflag) { return -EINVAL; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp18100.00%1100.00%
Total18100.00%1100.00%

/* set/clear bits in base connection state, return previous state */
static inline int update_basstate(struct bas_cardstate *ucs, int set, int clear) { unsigned long flags; int state; spin_lock_irqsave(&ucs->lock, flags); state = ucs->basstate; ucs->basstate = (state & ~clear) | set; spin_unlock_irqrestore(&ucs->lock, flags); return state; }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt67100.00%1100.00%
Total67100.00%1100.00%

/* error_hangup * hang up any existing connection because of an unrecoverable error * This function may be called from any context and takes care of scheduling * the necessary actions for execution outside of interrupt context. * cs->lock must not be held. * argument: * B channel control structure */
static inline void error_hangup(struct bc_state *bcs) { struct cardstate *cs = bcs->cs; gigaset_add_event(cs, &bcs->at_state, EV_HUP, NULL, 0, NULL); gigaset_schedule_event(cs); }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp4295.45%150.00%
Tilman Schmidt24.55%150.00%
Total44100.00%2100.00%

/* error_reset * reset Gigaset device because of an unrecoverable error * This function may be called from any context, and takes care of * scheduling the necessary actions for execution outside of interrupt context. * cs->hw.bas->lock must not be held. * argument: * controller state structure */
static inline void error_reset(struct cardstate *cs) { /* reset interrupt pipe to recover (ignore errors) */ update_basstate(cs->hw.bas, BS_RESETTING, 0); if (req_submit(cs->bcs, HD_RESET_INTERRUPT_PIPE, 0, BAS_TIMEOUT)) /* submission failed, escalate to USB port reset */ usb_queue_reset_device(cs->hw.bas->interface); }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt3871.70%480.00%
Hansjoerg Lipp1528.30%120.00%
Total53100.00%5100.00%

/* check_pending * check for completion of pending control request * parameter: * ucs hardware specific controller state structure */
static void check_pending(struct bas_cardstate *ucs) { unsigned long flags; spin_lock_irqsave(&ucs->lock, flags); switch (ucs->pending) { case 0: break; case HD_OPEN_ATCHANNEL: if (ucs->basstate & BS_ATOPEN) ucs->pending = 0; break; case HD_OPEN_B1CHANNEL: if (ucs->basstate & BS_B1OPEN) ucs->pending = 0; break; case HD_OPEN_B2CHANNEL: if (ucs->basstate & BS_B2OPEN) ucs->pending = 0; break; case HD_CLOSE_ATCHANNEL: if (!(ucs->basstate & BS_ATOPEN)) ucs->pending = 0; break; case HD_CLOSE_B1CHANNEL: if (!(ucs->basstate & BS_B1OPEN)) ucs->pending = 0; break; case HD_CLOSE_B2CHANNEL: if (!(ucs->basstate & BS_B2OPEN)) ucs->pending = 0; break; case HD_DEVICE_INIT_ACK: /* no reply expected */ ucs->pending = 0; break; case HD_RESET_INTERRUPT_PIPE: if (!(ucs->basstate & BS_RESETTING)) ucs->pending = 0; break; /* * HD_READ_ATMESSAGE and HD_WRITE_ATMESSAGE are handled separately * and should never end up here */ default: dev_warn(&ucs->interface->dev, "unknown pending request 0x%02x cleared\n", ucs->pending); ucs->pending = 0; } if (!ucs->pending) del_timer(&ucs->timer_ctrl); spin_unlock_irqrestore(&ucs->lock, flags); }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp20486.81%125.00%
Tilman Schmidt3113.19%375.00%
Total235100.00%4100.00%

/* cmd_in_timeout * timeout routine for command input request * argument: * controller state structure */
static void cmd_in_timeout(unsigned long data) { struct cardstate *cs = (struct cardstate *) data; struct bas_cardstate *ucs = cs->hw.bas; int rc; if (!ucs->rcvbuf_size) { gig_dbg(DEBUG_USBREQ, "%s: no receive in progress", __func__); return; } if (ucs->retry_cmd_in++ >= BAS_RETRY) { dev_err(cs->dev, "control read: timeout, giving up after %d tries\n", ucs->retry_cmd_in); kfree(ucs->rcvbuf); ucs->rcvbuf = NULL; ucs->rcvbuf_size = 0; error_reset(cs); return; } gig_dbg(DEBUG_USBREQ, "%s: timeout, retry %d", __func__, ucs->retry_cmd_in); rc = atread_submit(cs, BAS_TIMEOUT); if (rc < 0) { kfree(ucs->rcvbuf); ucs->rcvbuf = NULL; ucs->rcvbuf_size = 0; if (rc != -ENODEV) error_reset(cs); } }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt11569.70%583.33%
Hansjoerg Lipp5030.30%116.67%
Total165100.00%6100.00%

/* read_ctrl_callback * USB completion handler for control pipe input * called by the USB subsystem in interrupt context * parameter: * urb USB request block * urb->context = inbuf structure for controller state */
static void read_ctrl_callback(struct urb *urb) { struct inbuf_t *inbuf = urb->context; struct cardstate *cs = inbuf->cs; struct bas_cardstate *ucs = cs->hw.bas; int status = urb->status; unsigned numbytes; int rc; update_basstate(ucs, 0, BS_ATRDPEND); wake_up(&ucs->waitqueue); del_timer(&ucs->timer_cmd_in); switch (status) { case 0: /* normal completion */ numbytes = urb->actual_length; if (unlikely(numbytes != ucs->rcvbuf_size)) { dev_warn(cs->dev, "control read: received %d chars, expected %d\n", numbytes, ucs->rcvbuf_size); if (numbytes > ucs->rcvbuf_size) numbytes = ucs->rcvbuf_size; } /* copy received bytes to inbuf, notify event layer */ if (gigaset_fill_inbuf(inbuf, ucs->rcvbuf, numbytes)) { gig_dbg(DEBUG_INTR, "%s-->BH", __func__); gigaset_schedule_event(cs); } break; case -ENOENT: /* cancelled */ case -ECONNRESET: /* cancelled (async) */ case -EINPROGRESS: /* pending */ case -ENODEV: /* device removed */ case -ESHUTDOWN: /* device shut down */ /* no further action necessary */ gig_dbg(DEBUG_USBREQ, "%s: %s", __func__, get_usb_statmsg(status)); break; default: /* other errors: retry */ if (ucs->retry_cmd_in++ < BAS_RETRY) { gig_dbg(DEBUG_USBREQ, "%s: %s, retry %d", __func__, get_usb_statmsg(status), ucs->retry_cmd_in); rc = atread_submit(cs, BAS_TIMEOUT); if (rc >= 0) /* successfully resubmitted, skip freeing */ return; if (rc == -ENODEV) /* disconnect, no further action necessary */ break; } dev_err(cs->dev, "control read: %s, giving up after %d tries\n", get_usb_statmsg(status), ucs->retry_cmd_in); error_reset(cs); } /* read finished, free buffer */ kfree(ucs->rcvbuf); ucs->rcvbuf = NULL; ucs->rcvbuf_size = 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt21971.10%787.50%
Hansjoerg Lipp8928.90%112.50%
Total308100.00%8100.00%

/* atread_submit * submit an HD_READ_ATMESSAGE command URB and optionally start a timeout * parameters: * cs controller state structure * timeout timeout in 1/10 sec., 0: none * return value: * 0 on success * -EBUSY if another request is pending * any URB submission error code */
static int atread_submit(struct cardstate *cs, int timeout) { struct bas_cardstate *ucs = cs->hw.bas; int basstate; int ret; gig_dbg(DEBUG_USBREQ, "-------> HD_READ_ATMESSAGE (%d)", ucs->rcvbuf_size); basstate = update_basstate(ucs, BS_ATRDPEND, 0); if (basstate & BS_ATRDPEND) { dev_err(cs->dev, "could not submit HD_READ_ATMESSAGE: URB busy\n"); return -EBUSY; } if (basstate & BS_SUSPEND) { dev_notice(cs->dev, "HD_READ_ATMESSAGE not submitted, " "suspend in progress\n"); update_basstate(ucs, 0, BS_ATRDPEND); /* treat like disconnect */ return -ENODEV; } ucs->dr_cmd_in.bRequestType = IN_VENDOR_REQ; ucs->dr_cmd_in.bRequest = HD_READ_ATMESSAGE; ucs->dr_cmd_in.wValue = 0; ucs->dr_cmd_in.wIndex = 0; ucs->dr_cmd_in.wLength = cpu_to_le16(ucs->rcvbuf_size); usb_fill_control_urb(ucs->urb_cmd_in, ucs->udev, usb_rcvctrlpipe(ucs->udev, 0), (unsigned char *) &ucs->dr_cmd_in, ucs->rcvbuf, ucs->rcvbuf_size, read_ctrl_callback, cs->inbuf); ret = usb_submit_urb(ucs->urb_cmd_in, GFP_ATOMIC); if (ret != 0) { update_basstate(ucs, 0, BS_ATRDPEND); dev_err(cs->dev, "could not submit HD_READ_ATMESSAGE: %s\n", get_usb_rcmsg(ret)); return ret; } if (timeout > 0) { gig_dbg(DEBUG_USBREQ, "setting timeout of %d/10 secs", timeout); mod_timer(&ucs->timer_cmd_in, jiffies + timeout * HZ / 10); } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt20674.64%675.00%
Hansjoerg Lipp6925.00%112.50%
Christoph Lameter10.36%112.50%
Total276100.00%8100.00%

/* int_in_work * workqueue routine to clear halt on interrupt in endpoint */
static void int_in_work(struct work_struct *work) { struct bas_cardstate *ucs = container_of(work, struct bas_cardstate, int_in_wq); struct urb *urb = ucs->urb_int_in; struct cardstate *cs = urb->context; int rc; /* clear halt condition */ rc = usb_clear_halt(ucs->udev, urb->pipe); gig_dbg(DEBUG_USBREQ, "clear_halt: %s", get_usb_rcmsg(rc)); if (rc == 0) /* success, resubmit interrupt read URB */ rc = usb_submit_urb(urb, GFP_ATOMIC); switch (rc) { case 0: /* success */ case -ENODEV: /* device gone */ case -EINVAL: /* URB already resubmitted, or terminal badness */ break; default: /* failure: try to recover by resetting the device */ dev_err(cs->dev, "clear halt failed: %s\n", get_usb_rcmsg(rc)); rc = usb_lock_device_for_reset(ucs->udev, ucs->interface); if (rc == 0) { rc = usb_reset_device(ucs->udev); usb_unlock_device(ucs->udev); } } ucs->retry_int_in = 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt169100.00%2100.00%
Total169100.00%2100.00%

/* int_in_resubmit * timer routine for interrupt read delayed resubmit * argument: * controller state structure */
static void int_in_resubmit(unsigned long data) { struct cardstate *cs = (struct cardstate *) data; struct bas_cardstate *ucs = cs->hw.bas; int rc; if (ucs->retry_int_in++ >= BAS_RETRY) { dev_err(cs->dev, "interrupt read: giving up after %d tries\n", ucs->retry_int_in); usb_queue_reset_device(ucs->interface); return; } gig_dbg(DEBUG_USBREQ, "%s: retry %d", __func__, ucs->retry_int_in); rc = usb_submit_urb(ucs->urb_int_in, GFP_ATOMIC); if (rc != 0 && rc != -ENODEV) { dev_err(cs->dev, "could not resubmit interrupt URB: %s\n", get_usb_rcmsg(rc)); usb_queue_reset_device(ucs->interface); } }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt126100.00%1100.00%
Total126100.00%1100.00%

/* read_int_callback * USB completion handler for interrupt pipe input * called by the USB subsystem in interrupt context * parameter: * urb USB request block * urb->context = controller state structure */
static void read_int_callback(struct urb *urb) { struct cardstate *cs = urb->context; struct bas_cardstate *ucs = cs->hw.bas; struct bc_state *bcs; int status = urb->status; unsigned long flags; int rc; unsigned l; int channel; switch (status) { case 0: /* success */ ucs->retry_int_in = 0; break; case -EPIPE: /* endpoint stalled */ schedule_work(&ucs->int_in_wq); /* fall through */ case -ENOENT: /* cancelled */ case -ECONNRESET: /* cancelled (async) */ case -EINPROGRESS: /* pending */ case -ENODEV: /* device removed */ case -ESHUTDOWN: /* device shut down */ /* no further action necessary */ gig_dbg(DEBUG_USBREQ, "%s: %s", __func__, get_usb_statmsg(status)); return; case -EPROTO: /* protocol error or unplug */ case -EILSEQ: case -ETIME: /* resubmit after delay */ gig_dbg(DEBUG_USBREQ, "%s: %s", __func__, get_usb_statmsg(status)); mod_timer(&ucs->timer_int_in, jiffies + HZ / 10); return; default: /* other errors: just resubmit */ dev_warn(cs->dev, "interrupt read: %s\n", get_usb_statmsg(status)); goto resubmit; } /* drop incomplete packets even if the missing bytes wouldn't matter */ if (unlikely(urb->actual_length < IP_MSGSIZE)) { dev_warn(cs->dev, "incomplete interrupt packet (%d bytes)\n", urb->actual_length); goto resubmit; } l = (unsigned) ucs->int_in_buf[1] + (((unsigned) ucs->int_in_buf[2]) << 8); gig_dbg(DEBUG_USBREQ, "<-------%d: 0x%02x (%u [0x%02x 0x%02x])", urb->actual_length, (int)ucs->int_in_buf[0], l, (int)ucs->int_in_buf[1], (int)ucs->int_in_buf[2]); channel = 0; switch (ucs->int_in_buf[0]) { case HD_DEVICE_INIT_OK: update_basstate(ucs, BS_INIT, 0); break; case HD_READY_SEND_ATDATA: del_timer(&ucs->timer_atrdy); update_basstate(ucs, BS_ATREADY, BS_ATTIMER); start_cbsend(cs); break; case HD_OPEN_B2CHANNEL_ACK: ++channel; case HD_OPEN_B1CHANNEL_ACK: bcs = cs->bcs + channel; update_basstate(ucs, BS_B1OPEN << channel, 0); gigaset_bchannel_up(bcs); break; case HD_OPEN_ATCHANNEL_ACK: update_basstate(ucs, BS_ATOPEN, 0); start_cbsend(cs); break; case HD_CLOSE_B2CHANNEL_ACK: ++channel; case HD_CLOSE_B1CHANNEL_ACK: bcs = cs->bcs + channel; update_basstate(ucs, 0, BS_B1OPEN << channel); stopurbs(bcs->hw.bas); gigaset_bchannel_down(bcs); break; case HD_CLOSE_ATCHANNEL_ACK: update_basstate(ucs, 0, BS_ATOPEN); break; case HD_B2_FLOW_CONTROL: ++channel; case HD_B1_FLOW_CONTROL: bcs = cs->bcs + channel; atomic_add((l - BAS_NORMFRAME) * BAS_CORRFRAMES, &bcs->hw.bas->corrbytes); gig_dbg(DEBUG_ISO, "Flow control (channel %d, sub %d): 0x%02x => %d", channel, bcs->hw.bas->numsub, l, atomic_read(&bcs->hw.bas->corrbytes)); break; case HD_RECEIVEATDATA_ACK: /* AT response ready to be received */ if (!l) { dev_warn(cs->dev, "HD_RECEIVEATDATA_ACK with length 0 ignored\n"); break; } spin_lock_irqsave(&cs->lock, flags); if (ucs->basstate & BS_ATRDPEND) { spin_unlock_irqrestore(&cs->lock, flags); dev_warn(cs->dev, "HD_RECEIVEATDATA_ACK(%d) during HD_READ_ATMESSAGE(%d) ignored\n", l, ucs->rcvbuf_size); break; } if (ucs->rcvbuf_size) { /* throw away previous buffer - we have no queue */ dev_err(cs->dev, "receive AT data overrun, %d bytes lost\n", ucs->rcvbuf_size); kfree(ucs->rcvbuf); ucs->rcvbuf_size = 0; } ucs->rcvbuf = kmalloc(l, GFP_ATOMIC); if (ucs->rcvbuf == NULL) { spin_unlock_irqrestore(&cs->lock, flags); dev_err(cs->dev, "out of memory receiving AT data\n"); break; } ucs->rcvbuf_size = l; ucs->retry_cmd_in = 0; rc = atread_submit(cs, BAS_TIMEOUT); if (rc < 0) { kfree(ucs->rcvbuf); ucs->rcvbuf = NULL; ucs->rcvbuf_size = 0; } spin_unlock_irqrestore(&cs->lock, flags); if (rc < 0 && rc != -ENODEV) error_reset(cs); break; case HD_RESET_INTERRUPT_PIPE_ACK: update_basstate(ucs, 0, BS_RESETTING); dev_notice(cs->dev, "interrupt pipe reset\n"); break; case HD_SUSPEND_END: gig_dbg(DEBUG_USBREQ, "HD_SUSPEND_END"); break; default: dev_warn(cs->dev, "unknown Gigaset signal 0x%02x (%u) ignored\n", (int) ucs->int_in_buf[0], l); } check_pending(ucs); wake_up(&ucs->waitqueue); resubmit: rc = usb_submit_urb(urb, GFP_ATOMIC); if (unlikely(rc != 0 && rc != -ENODEV)) { dev_err(cs->dev, "could not resubmit interrupt URB: %s\n", get_usb_rcmsg(rc)); error_reset(cs); } }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt71583.72%1083.33%
Hansjoerg Lipp13816.16%18.33%
Christoph Lameter10.12%18.33%
Total854100.00%12100.00%

/* read_iso_callback * USB completion handler for B channel isochronous input * called by the USB subsystem in interrupt context * parameter: * urb USB request block of completed request * urb->context = bc_state structure */
static void read_iso_callback(struct urb *urb) { struct bc_state *bcs; struct bas_bc_state *ubc; int status = urb->status; unsigned long flags; int i, rc; /* status codes not worth bothering the tasklet with */ if (unlikely(status == -ENOENT || status == -ECONNRESET || status == -EINPROGRESS || status == -ENODEV || status == -ESHUTDOWN)) { gig_dbg(DEBUG_ISO, "%s: %s", __func__, get_usb_statmsg(status)); return; } bcs = urb->context; ubc = bcs->hw.bas; spin_lock_irqsave(&ubc->isoinlock, flags); if (likely(ubc->isoindone == NULL)) { /* pass URB to tasklet */ ubc->isoindone = urb; ubc->isoinstatus = status; tasklet_hi_schedule(&ubc->rcvd_tasklet); } else { /* tasklet still busy, drop data and resubmit URB */ gig_dbg(DEBUG_ISO, "%s: overrun", __func__); ubc->loststatus = status; for (i = 0; i < BAS_NUMFRAMES; i++) { ubc->isoinlost += urb->iso_frame_desc[i].actual_length; if (unlikely(urb->iso_frame_desc[i].status != 0 && urb->iso_frame_desc[i].status != -EINPROGRESS)) ubc->loststatus = urb->iso_frame_desc[i].status; urb->iso_frame_desc[i].status = 0; urb->iso_frame_desc[i].actual_length = 0; } if (likely(ubc->running)) { /* urb->dev is clobbered by USB subsystem */ urb->dev = bcs->cs->hw.bas->udev; urb->transfer_flags = URB_ISO_ASAP; urb->number_of_packets = BAS_NUMFRAMES; rc = usb_submit_urb(urb, GFP_ATOMIC); if (unlikely(rc != 0 && rc != -ENODEV)) { dev_err(bcs->cs->dev, "could not resubmit isoc read URB: %s\n", get_usb_rcmsg(rc)); dump_urb(DEBUG_ISO, "isoc read", urb); error_hangup(bcs); } } } spin_unlock_irqrestore(&ubc->isoinlock, flags); }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp30685.96%114.29%
Tilman Schmidt4913.76%571.43%
Christoph Lameter10.28%114.29%
Total356100.00%7100.00%

/* write_iso_callback * USB completion handler for B channel isochronous output * called by the USB subsystem in interrupt context * parameter: * urb USB request block of completed request * urb->context = isow_urbctx_t structure */
static void write_iso_callback(struct urb *urb) { struct isow_urbctx_t *ucx; struct bas_bc_state *ubc; int status = urb->status; unsigned long flags; /* status codes not worth bothering the tasklet with */ if (unlikely(status == -ENOENT || status == -ECONNRESET || status == -EINPROGRESS || status == -ENODEV || status == -ESHUTDOWN)) { gig_dbg(DEBUG_ISO, "%s: %s", __func__, get_usb_statmsg(status)); return; } /* pass URB context to tasklet */ ucx = urb->context; ubc = ucx->bcs->hw.bas; ucx->status = status; spin_lock_irqsave(&ubc->isooutlock, flags); ubc->isooutovfl = ubc->isooutdone; ubc->isooutdone = ucx; spin_unlock_irqrestore(&ubc->isooutlock, flags); tasklet_hi_schedule(&ubc->sent_tasklet); }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp12082.76%120.00%
Tilman Schmidt2517.24%480.00%
Total145100.00%5100.00%

/* starturbs * prepare and submit USB request blocks for isochronous input and output * argument: * B channel control structure * return value: * 0 on success * < 0 on error (no URBs submitted) */
static int starturbs(struct bc_state *bcs) { struct bas_bc_state *ubc = bcs->hw.bas; struct urb *urb; int j, k; int rc; /* initialize L2 reception */ if (bcs->proto2 == L2_HDLC) bcs->inputstate |= INS_flag_hunt; /* submit all isochronous input URBs */ ubc->running = 1; for (k = 0; k < BAS_INURBS; k++) { urb = ubc->isoinurbs[k]; if (!urb) { rc = -EFAULT; goto error; } urb->dev = bcs->cs->hw.bas->udev; urb->pipe = usb_rcvisocpipe(urb->dev, 3 + 2 * bcs->channel); urb->transfer_flags = URB_ISO_ASAP; urb->transfer_buffer = ubc->isoinbuf + k * BAS_INBUFSIZE; urb->transfer_buffer_length = BAS_INBUFSIZE; urb->number_of_packets = BAS_NUMFRAMES; urb->interval = BAS_FRAMETIME; urb->complete = read_iso_callback; urb->context = bcs; for (j = 0; j < BAS_NUMFRAMES; j++) { urb->iso_frame_desc[j].offset = j * BAS_MAXFRAME; urb->iso_frame_desc[j].length = BAS_MAXFRAME; urb->iso_frame_desc[j].status = 0; urb->iso_frame_desc[j].actual_length = 0; } dump_urb(DEBUG_ISO, "Initial isoc read", urb); rc = usb_submit_urb(urb, GFP_ATOMIC); if (rc != 0) goto error; } /* initialize L2 transmission */ gigaset_isowbuf_init(ubc->isooutbuf, PPP_FLAG); /* set up isochronous output URBs for flag idling */ for (k = 0; k < BAS_OUTURBS; ++k) { urb = ubc->isoouturbs[k].urb; if (!urb) { rc = -EFAULT; goto error; } urb->dev = bcs->cs->hw.bas->udev; urb->pipe = usb_sndisocpipe(urb->dev, 4 + 2 * bcs->channel); urb->transfer_flags = URB_ISO_ASAP; urb->transfer_buffer = ubc->isooutbuf->data; urb->transfer_buffer_length = sizeof(ubc->isooutbuf->data); urb->number_of_packets = BAS_NUMFRAMES; urb->interval = BAS_FRAMETIME; urb->complete = write_iso_callback; urb->context = &ubc->isoouturbs[k]; for (j = 0; j < BAS_NUMFRAMES; ++j) { urb->iso_frame_desc[j].offset = BAS_OUTBUFSIZE; urb->iso_frame_desc[j].length = BAS_NORMFRAME; urb->iso_frame_desc[j].status = 0; urb->iso_frame_desc[j].actual_length = 0; } ubc->isoouturbs[k].limit = -1; } /* keep one URB free, submit the others */ for (k = 0; k < BAS_OUTURBS - 1; ++k) { dump_urb(DEBUG_ISO, "Initial isoc write", urb); rc = usb_submit_urb(ubc->isoouturbs[k].urb, GFP_ATOMIC); if (rc != 0) goto error; } dump_urb(DEBUG_ISO, "Initial isoc write (free)", urb); ubc->isooutfree = &ubc->isoouturbs[BAS_OUTURBS - 1]; ubc->isooutdone = ubc->isooutovfl = NULL; return 0; error: stopurbs(ubc); return rc; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp55395.67%114.29%
Tilman Schmidt233.98%571.43%
Christoph Lameter20.35%114.29%
Total578100.00%7100.00%

/* stopurbs * cancel the USB request blocks for isochronous input and output * errors are silently ignored * argument: * B channel control structure */
static void stopurbs(struct bas_bc_state *ubc) { int k, rc; ubc->running = 0; for (k = 0; k < BAS_INURBS; ++k) { rc = usb_unlink_urb(ubc->isoinurbs[k]); gig_dbg(DEBUG_ISO, "%s: isoc input URB %d unlinked, result = %s", __func__, k, get_usb_rcmsg(rc)); } for (k = 0; k < BAS_OUTURBS; ++k) { rc = usb_unlink_urb(ubc->isoouturbs[k].urb); gig_dbg(DEBUG_ISO, "%s: isoc output URB %d unlinked, result = %s", __func__, k, get_usb_rcmsg(rc)); } }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp9990.00%125.00%
Tilman Schmidt1110.00%375.00%
Total110100.00%4100.00%

/* Isochronous Write - Bottom Half */ /* =============================== */ /* submit_iso_write_urb * fill and submit the next isochronous write URB * parameters: * ucx context structure containing URB * return value: * number of frames submitted in URB * 0 if URB not submitted because no data available (isooutbuf busy) * error code < 0 on error */
static int submit_iso_write_urb(struct isow_urbctx_t *ucx) { struct urb *urb = ucx->urb; struct bas_bc_state *ubc = ucx->bcs->hw.bas; struct usb_iso_packet_descriptor *ifd; int corrbytes, nframe, rc; /* urb->dev is clobbered by USB subsystem */ urb->dev = ucx->bcs->cs->hw.bas->udev; urb->transfer_flags = URB_ISO_ASAP; urb->transfer_buffer = ubc->isooutbuf->data; urb->transfer_buffer_length = sizeof(ubc->isooutbuf->data); for (nframe = 0; nframe < BAS_NUMFRAMES; nframe++) { ifd = &urb->iso_frame_desc[nframe]; /* compute frame length according to flow control */ ifd->length = BAS_NORMFRAME; corrbytes = atomic_read(&ubc->corrbytes); if (corrbytes != 0) { gig_dbg(DEBUG_ISO, "%s: corrbytes=%d", __func__, corrbytes); if (corrbytes > BAS_HIGHFRAME - BAS_NORMFRAME) corrbytes = BAS_HIGHFRAME - BAS_NORMFRAME; else if (corrbytes < BAS_LOWFRAME - BAS_NORMFRAME) corrbytes = BAS_LOWFRAME - BAS_NORMFRAME; ifd->length += corrbytes; atomic_add(-corrbytes, &ubc->corrbytes); } /* retrieve block of data to send */ rc = gigaset_isowbuf_getbytes(ubc->isooutbuf, ifd->length); if (rc < 0) { if (rc == -EBUSY) { gig_dbg(DEBUG_ISO, "%s: buffer busy at frame %d", __func__, nframe); /* tasklet will be restarted from gigaset_isoc_send_skb() */ } else { dev_err(ucx->bcs->cs->dev, "%s: buffer error %d at frame %d\n", __func__, rc, nframe); return rc; } break; } ifd->offset = rc; ucx->limit = ubc->isooutbuf->nextread; ifd->status = 0; ifd->actual_length = 0; } if (unlikely(nframe == 0)) return 0; /* no data to send */ urb->number_of_packets = nframe; rc = usb_submit_urb(urb, GFP_ATOMIC); if (unlikely(rc)) { if (rc == -ENODEV) /* device removed - give up silently */ gig_dbg(DEBUG_ISO, "%s: disconnected", __func__); else dev_err(ucx->bcs->cs->dev, "could not submit isoc write URB: %s\n", get_usb_rcmsg(rc)); return rc; } ++ubc->numsub; return nframe; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp28775.73%110.00%
Tilman Schmidt9124.01%880.00%
Christoph Lameter10.26%110.00%
Total379100.00%10100.00%

/* write_iso_tasklet * tasklet scheduled when an isochronous output URB from the Gigaset device * has completed * parameter: * data B channel state structure */
static void write_iso_tasklet(unsigned long data) { struct bc_state *bcs = (struct bc_state *) data; struct bas_bc_state *ubc = bcs->hw.bas; struct cardstate *cs = bcs->cs; struct isow_urbctx_t *done, *next, *ovfl; struct urb *urb; int status; struct usb_iso_packet_descriptor *ifd; unsigned long flags; int i; struct sk_buff *skb; int len; int rc; /* loop while completed URBs arrive in time */ for (;;) { if (unlikely(!(ubc->running))) { gig_dbg(DEBUG_ISO, "%s: not running", __func__); return; } /* retrieve completed URBs */ spin_lock_irqsave(&ubc->isooutlock, flags); done = ubc->isooutdone; ubc->isooutdone = NULL; ovfl = ubc->isooutovfl; ubc->isooutovfl = NULL; spin_unlock_irqrestore(&ubc->isooutlock, flags); if (ovfl) { dev_err(cs->dev, "isoc write underrun\n"); error_hangup(bcs); break; } if (!done) break; /* submit free URB if available */ spin_lock_irqsave(&ubc->isooutlock, flags); next = ubc->isooutfree; ubc->isooutfree = NULL; spin_unlock_irqrestore(&ubc->isooutlock, flags); if (next) { rc = submit_iso_write_urb(next); if (unlikely(rc <= 0 && rc != -ENODEV)) { /* could not submit URB, put it back */ spin_lock_irqsave(&ubc->isooutlock, flags); if (ubc->isooutfree == NULL) { ubc->isooutfree = next; next = NULL; } spin_unlock_irqrestore(&ubc->isooutlock, flags); if (next) { /* couldn't put it back */ dev_err(cs->dev, "losing isoc write URB\n"); error_hangup(bcs); } } } /* process completed URB */ urb = done->urb; status = done->status; switch (status) { case -EXDEV: /* partial completion */ gig_dbg(DEBUG_ISO, "%s: URB partially completed", __func__); /* fall through - what's the difference anyway? */ case 0: /* normal completion */ /* inspect individual frames * assumptions (for lack of documentation): * - actual_length bytes of first frame in error are * successfully sent * - all following frames are not sent at all */ for (i = 0; i < BAS_NUMFRAMES; i++) { ifd = &urb->iso_frame_desc[i]; if (ifd->status || ifd->actual_length != ifd->length) { dev_warn(cs->dev, "isoc write: frame %d[%d/%d]: %s\n", i, ifd->actual_length, ifd->length, get_usb_statmsg(ifd->status)); break; } } break; case -EPIPE: /* stall - probably underrun */ dev_err(cs->dev, "isoc write: stalled\n"); error_hangup(bcs); break; default: /* other errors */ dev_warn(cs->dev, "isoc write: %s\n", get_usb_statmsg(status)); } /* mark the write buffer area covered by this URB as free */ if (done->limit >= 0) ubc->isooutbuf->read = done->limit; /* mark URB as free */ spin_lock_irqsave(&ubc->isooutlock, flags); next = ubc->isooutfree; ubc->isooutfree = done; spin_unlock_irqrestore(&ubc->isooutlock, flags); if (next) { /* only one URB still active - resubmit one */ rc = submit_iso_write_urb(next); if (unlikely(rc <= 0 && rc != -ENODEV)) { /* couldn't submit */ error_hangup(bcs); } } } /* process queued SKBs */ while ((skb = skb_dequeue(&bcs->squeue))) { /* copy to output buffer, doing L2 encapsulation */ len = skb->len; if (gigaset_isoc_buildframe(bcs, skb->data, len) == -EAGAIN) { /* insufficient buffer space, push back onto queue */ skb_queue_head(&bcs->squeue, skb); gig_dbg(DEBUG_ISO, "%s: skb requeued, qlen=%d", __func__, skb_queue_len(&bcs->squeue)); break; } skb_pull(skb, len); gigaset_skb_sent(bcs, skb); dev_kfree_skb_any(skb); } }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp52883.28%114.29%
Tilman Schmidt10616.72%685.71%
Total634100.00%7100.00%

/* Isochronous Read - Bottom Half */ /* ============================== */ /* read_iso_tasklet * tasklet scheduled when an isochronous input URB from the Gigaset device * has completed * parameter: * data B channel state structure */
static void read_iso_tasklet(unsigned long data) { struct bc_state *bcs = (struct bc_state *) data; struct bas_bc_state *ubc = bcs->hw.bas; struct cardstate *cs = bcs->cs; struct urb *urb; int status; struct usb_iso_packet_descriptor *ifd; char *rcvbuf; unsigned long flags; int totleft, numbytes, offset, frame, rc; /* loop while more completed URBs arrive in the meantime */ for (;;) { /* retrieve URB */ spin_lock_irqsave(&ubc->isoinlock, flags); urb = ubc->isoindone; if (!urb) { spin_unlock_irqrestore(&ubc->isoinlock, flags); return; } status = ubc->isoinstatus; ubc->isoindone = NULL; if (unlikely(ubc->loststatus != -EINPROGRESS)) { dev_warn(cs->dev, "isoc read overrun, URB dropped (status: %s, %d bytes)\n", get_usb_statmsg(ubc->loststatus), ubc->isoinlost); ubc->loststatus = -EINPROGRESS; } spin_unlock_irqrestore(&ubc->isoinlock, flags); if (unlikely(!(ubc->running))) { gig_dbg(DEBUG_ISO, "%s: channel not running, " "dropped URB with status: %s", __func__, get_usb_statmsg(status)); return; } switch (status) { case 0: /* normal completion */ break; case -EXDEV: /* inspect individual frames (we do that anyway) */ gig_dbg(DEBUG_ISO, "%s: URB partially completed", __func__); break; case -ENOENT: case -ECONNRESET: case -EINPROGRESS: gig_dbg(DEBUG_ISO, "%s: %s", __func__, get_usb_statmsg(status)); continue; /* -> skip */ case -EPIPE: dev_err(cs->dev, "isoc read: stalled\n"); error_hangup(bcs); continue; /* -> skip */ default: /* other error */ dev_warn(cs->dev, "isoc read: %s\n", get_usb_statmsg(status)); goto error; } rcvbuf = urb->transfer_buffer; totleft = urb->actual_length; for (frame = 0; totleft > 0 && frame < BAS_NUMFRAMES; frame++) { ifd = &urb->iso_frame_desc[frame]; numbytes = ifd->actual_length; switch (ifd->status) { case 0: /* success */ break; case -EPROTO: /* protocol error or unplug */ case -EILSEQ: case -ETIME: /* probably just disconnected, ignore */ gig_dbg(DEBUG_ISO, "isoc read: frame %d[%d]: %s\n", frame, numbytes, get_usb_statmsg(ifd->status)); break; default: /* other error */ /* report, assume transferred bytes are ok */ dev_warn(cs->dev, "isoc read: frame %d[%d]: %s\n", frame, numbytes, get_usb_statmsg(ifd->status)); } if (unlikely(numbytes > BAS_MAXFRAME)) dev_warn(cs->dev, "isoc read: frame %d[%d]: %s\n", frame, numbytes, "exceeds max frame size"); if (unlikely(numbytes > totleft)) { dev_warn(cs->dev, "isoc read: frame %d[%d]: %s\n", frame, numbytes, "exceeds total transfer length"); numbytes = totleft; } offset = ifd->offset; if (unlikely(offset + numbytes > BAS_INBUFSIZE)) { dev_warn(cs->dev, "isoc read: frame %d[%d]: %s\n", frame, numbytes, "exceeds end of buffer"); numbytes = BAS_INBUFSIZE - offset; } gigaset_isoc_receive(rcvbuf + offset, numbytes, bcs); totleft -= numbytes; } if (unlikely(totleft > 0)) dev_warn(cs->dev, "isoc read: %d data bytes missing\n", totleft); error: /* URB processed, resubmit */ for (frame = 0; frame < BAS_NUMFRAMES; frame++) { urb->iso_frame_desc[frame].status = 0; urb->iso_frame_desc[frame].actual_length = 0; } /* urb->dev is clobbered by USB subsystem */ urb->dev = bcs->cs->hw.bas->udev; urb->transfer_flags = URB_ISO_ASAP; urb->number_of_packets = BAS_NUMFRAMES; rc = usb_submit_urb(urb, GFP_ATOMIC); if (unlikely(rc != 0 && rc != -ENODEV)) { dev_err(cs->dev, "could not resubmit isoc read URB: %s\n", get_usb_rcmsg(rc)); dump_urb(DEBUG_ISO, "resubmit isoc read", urb); error_hangup(bcs); } } }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp46870.06%19.09%
Tilman Schmidt19929.79%981.82%
Christoph Lameter10.15%19.09%
Total668100.00%11100.00%

/* Channel Operations */ /* ================== */ /* req_timeout * timeout routine for control output request * argument: * controller state structure */
static void req_timeout(unsigned long data) { struct cardstate *cs = (struct cardstate *) data; struct bas_cardstate *ucs = cs->hw.bas; int pending; unsigned long flags; check_pending(ucs); spin_lock_irqsave(&ucs->lock, flags); pending = ucs->pending; ucs->pending = 0; spin_unlock_irqrestore(&ucs->lock, flags); switch (pending) { case 0: /* no pending request */ gig_dbg(DEBUG_USBREQ, "%s: no request pending", __func__); break; case HD_OPEN_ATCHANNEL: dev_err(cs->dev, "timeout opening AT channel\n"); error_reset(cs); break; case HD_OPEN_B1CHANNEL: dev_err(cs->dev, "timeout opening channel 1\n"); error_hangup(&cs->bcs[0]); break; case HD_OPEN_B2CHANNEL: dev_err(cs->dev, "timeout opening channel 2\n"); error_hangup(&cs->bcs[1]); break; case HD_CLOSE_ATCHANNEL: dev_err(cs->dev, "timeout closing AT channel\n"); error_reset(cs); break; case HD_CLOSE_B1CHANNEL: dev_err(cs->dev, "timeout closing channel 1\n"); error_reset(cs); break; case HD_CLOSE_B2CHANNEL: dev_err(cs->dev, "timeout closing channel 2\n"); error_reset(cs); break; case HD_RESET_INTERRUPT_PIPE: /* error recovery escalation */ dev_err(cs->dev, "reset interrupt pipe timeout, attempting USB reset\n"); usb_queue_reset_device(ucs->interface); break; default: dev_warn(cs->dev, "request 0x%02x timed out, clearing\n", pending); } wake_up(&ucs->waitqueue); }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp13351.55%114.29%
Tilman Schmidt12548.45%685.71%
Total258100.00%7100.00%

/* write_ctrl_callback * USB completion handler for control pipe output * called by the USB subsystem in interrupt context * parameter: * urb USB request block of completed request * urb->context = hardware specific controller state structure */
static void write_ctrl_callback(struct urb *urb) { struct bas_cardstate *ucs = urb->context; int status = urb->status; int rc; unsigned long flags; /* check status */ switch (status) { case 0: /* normal completion */ spin_lock_irqsave(&ucs->lock, flags); switch (ucs->pending) { case HD_DEVICE_INIT_ACK: /* no reply expected */ del_timer(&ucs->timer_ctrl); ucs->pending = 0; break; } spin_unlock_irqrestore(&ucs->lock, flags); return; case -ENOENT: /* cancelled */ case -ECONNRESET: /* cancelled (async) */ case -EINPROGRESS: /* pending */ case -ENODEV: /* device removed */ case -ESHUTDOWN: /* device shut down */ /* ignore silently */ gig_dbg(DEBUG_USBREQ, "%s: %s", __func__, get_usb_statmsg(status)); break; default: /* any failure */ /* don't retry if suspend requested */ if (++ucs->retry_ctrl > BAS_RETRY || (ucs->basstate & BS_SUSPEND)) { dev_err(&ucs->interface->dev, "control request 0x%02x failed: %s\n", ucs->dr_ctrl.bRequest, get_usb_statmsg(status)); break; /* give up */ } dev_notice(&ucs->interface->dev, "control request 0x%02x: %s, retry %d\n", ucs->dr_ctrl.bRequest, get_usb_statmsg(status), ucs->retry_ctrl); /* urb->dev is clobbered by USB subsystem */ urb->dev = ucs->udev; rc = usb_submit_urb(urb, GFP_ATOMIC); if (unlikely(rc)) { dev_err(&ucs->interface->dev, "could not resubmit request 0x%02x: %s\n", ucs->dr_ctrl.bRequest, get_usb_rcmsg(rc)); break; } /* resubmitted */ return; } /* failed, clear pending request */ spin_lock_irqsave(&ucs->lock, flags); del_timer(&ucs->timer_ctrl); ucs->pending = 0; spin_unlock_irqrestore(&ucs->lock, flags); wake_up(&ucs->waitqueue); }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt23476.97%571.43%
Hansjoerg Lipp6922.70%114.29%
Christoph Lameter10.33%114.29%
Total304100.00%7100.00%

/* req_submit * submit a control output request without message buffer to the Gigaset base * and optionally start a timeout * parameters: * bcs B channel control structure * req control request code (HD_*) * val control request parameter value (set to 0 if unused) * timeout timeout in seconds (0: no timeout) * return value: * 0 on success * -EBUSY if another request is pending * any URB submission error code */
static int req_submit(struct bc_state *bcs, int req, int val, int timeout) { struct bas_cardstate *ucs = bcs->cs->hw.bas; int ret; unsigned long flags; gig_dbg(DEBUG_USBREQ, "-------> 0x%02x (%d)", req, val); spin_lock_irqsave(&ucs->lock, flags); if (ucs->pending) { spin_unlock_irqrestore(&ucs->lock, flags); dev_err(bcs->cs->dev, "submission of request 0x%02x failed: " "request 0x%02x still pending\n", req, ucs->pending); return -EBUSY; } ucs->dr_ctrl.bRequestType = OUT_VENDOR_REQ; ucs->dr_ctrl.bRequest = req; ucs->dr_ctrl.wValue = cpu_to_le16(val); ucs->dr_ctrl.wIndex = 0; ucs->dr_ctrl.wLength = 0; usb_fill_control_urb(ucs->urb_ctrl, ucs->udev, usb_sndctrlpipe(ucs->udev, 0), (unsigned char *) &ucs->dr_ctrl, NULL, 0, write_ctrl_callback, ucs); ucs->retry_ctrl = 0; ret = usb_submit_urb(ucs->urb_ctrl, GFP_ATOMIC); if (unlikely(ret)) { dev_err(bcs->cs->dev, "could not submit request 0x%02x: %s\n", req, get_usb_rcmsg(ret)); spin_unlock_irqrestore(&ucs->lock, flags); return ret; } ucs->pending = req; if (timeout > 0) { gig_dbg(DEBUG_USBREQ, "setting timeout of %d/10 secs", timeout); mod_timer(&ucs->timer_ctrl, jiffies + timeout * HZ / 10); } spin_unlock_irqrestore(&ucs->lock, flags); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp24584.19%116.67%
Tilman Schmidt4515.46%466.67%
Christoph Lameter10.34%116.67%
Total291100.00%6100.00%

/* gigaset_init_bchannel * called by common.c to connect a B channel * initialize isochronous I/O and tell the Gigaset base to open the channel * argument: * B channel control structure * return value: * 0 on success, error code < 0 on error */
static int gigaset_init_bchannel(struct bc_state *bcs) { struct cardstate *cs = bcs->cs; int req, ret; unsigned long flags; spin_lock_irqsave(&cs->lock, flags); if (unlikely(!cs->connected)) { gig_dbg(DEBUG_USBREQ, "%s: not connected", __func__); spin_unlock_irqrestore(&cs->lock, flags); return -ENODEV; } if (cs->hw.bas->basstate & BS_SUSPEND) { dev_notice(cs->dev, "not starting isoc I/O, suspend in progress\n"); spin_unlock_irqrestore(&cs->lock, flags); return -EHOSTUNREACH; } ret = starturbs(bcs); if (ret < 0) { spin_unlock_irqrestore(&cs->lock, flags); dev_err(cs->dev, "could not start isoc I/O for channel B%d: %s\n", bcs->channel + 1, ret == -EFAULT ? "null URB" : get_usb_rcmsg(ret)); if (ret != -ENODEV) error_hangup(bcs); return ret; } req = bcs->channel ? HD_OPEN_B2CHANNEL : HD_OPEN_B1CHANNEL; ret = req_submit(bcs, req, 0, BAS_TIMEOUT); if (ret < 0) { dev_err(cs->dev, "could not open channel B%d\n", bcs->channel + 1); stopurbs(bcs->hw.bas); } spin_unlock_irqrestore(&cs->lock, flags); if (ret < 0 && ret != -ENODEV) error_hangup(bcs); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt16562.98%787.50%
Hansjoerg Lipp9737.02%112.50%
Total262100.00%8100.00%

/* gigaset_close_bchannel * called by common.c to disconnect a B channel * tell the Gigaset base to close the channel * stopping isochronous I/O and LL notification will be done when the * acknowledgement for the close arrives * argument: * B channel control structure * return value: * 0 on success, error code < 0 on error */
static int gigaset_close_bchannel(struct bc_state *bcs) { struct cardstate *cs = bcs->cs; int req, ret; unsigned long flags; spin_lock_irqsave(&cs->lock, flags); if (unlikely(!cs->connected)) { spin_unlock_irqrestore(&cs->lock, flags); gig_dbg(DEBUG_USBREQ, "%s: not connected", __func__); return -ENODEV; } if (!(cs->hw.bas->basstate & (bcs->channel ? BS_B2OPEN : BS_B1OPEN))) { /* channel not running: just signal common.c */ spin_unlock_irqrestore(&cs->lock, flags); gigaset_bchannel_down(bcs); return 0; } /* channel running: tell device to close it */ req = bcs->channel ? HD_CLOSE_B2CHANNEL : HD_CLOSE_B1CHANNEL; ret = req_submit(bcs, req, 0, BAS_TIMEOUT); if (ret < 0) dev_err(cs->dev, "closing channel B%d failed\n", bcs->channel + 1); spin_unlock_irqrestore(&cs->lock, flags); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt9352.84%480.00%
Hansjoerg Lipp8347.16%120.00%
Total176100.00%5100.00%

/* Device Operations */ /* ================= */ /* complete_cb * unqueue first command buffer from queue, waking any sleepers * must be called with cs->cmdlock held * parameter: * cs controller state structure */
static void complete_cb(struct cardstate *cs) { struct cmdbuf_t *cb = cs->cmdbuf; /* unqueue completed buffer */ cs->cmdbytes -= cs->curlen; gig_dbg(DEBUG_OUTPUT, "write_command: sent %u bytes, %u left", cs->curlen, cs->cmdbytes); if (cb->next != NULL) { cs->cmdbuf = cb->next; cs->cmdbuf->prev = NULL; cs->curlen = cs->cmdbuf->len; } else { cs->cmdbuf = NULL; cs->lastcmdbuf = NULL; cs->curlen = 0; } if (cb->wake_tasklet) tasklet_schedule(cb->wake_tasklet); kfree(cb); }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp10184.87%120.00%
Tilman Schmidt1815.13%480.00%
Total119100.00%5100.00%

/* write_command_callback * USB completion handler for AT command transmission * called by the USB subsystem in interrupt context * parameter: * urb USB request block of completed request * urb->context = controller state structure */
static void write_command_callback(struct urb *urb) { struct cardstate *cs = urb->context; struct bas_cardstate *ucs = cs->hw.bas; int status = urb->status; unsigned long flags; update_basstate(ucs, 0, BS_ATWRPEND); wake_up(&ucs->waitqueue); /* check status */ switch (status) { case 0: /* normal completion */ break; case -ENOENT: /* cancelled */ case -ECONNRESET: /* cancelled (async) */ case -EINPROGRESS: /* pending */ case -ENODEV: /* device removed */ case -ESHUTDOWN: /* device shut down */ /* ignore silently */ gig_dbg(DEBUG_USBREQ, "%s: %s", __func__, get_usb_statmsg(status)); return; default: /* any failure */ if (++ucs->retry_cmd_out > BAS_RETRY) { dev_warn(cs->dev, "command write: %s, " "giving up after %d retries\n", get_usb_statmsg(status), ucs->retry_cmd_out); break; } if (ucs->basstate & BS_SUSPEND) { dev_warn(cs->dev, "command write: %s, " "won't retry - suspend requested\n", get_usb_statmsg(status)); break; } if (cs->cmdbuf == NULL) { dev_warn(cs->dev, "command write: %s, " "cannot retry - cmdbuf gone\n", get_usb_statmsg(status)); break; } dev_notice(cs->dev, "command write: %s, retry %d\n", get_usb_statmsg(status), ucs->retry_cmd_out); if (atwrite_submit(cs, cs->cmdbuf->buf, cs->cmdbuf->len) >= 0) /* resubmitted - bypass regular exit block */ return; /* command send failed, assume base still waiting */ update_basstate(ucs, BS_ATREADY, 0); } spin_lock_irqsave(&cs->cmdlock, flags); if (cs->cmdbuf != NULL) complete_cb(cs); spin_unlock_irqrestore(&cs->cmdlock, flags); }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp18866.90%116.67%
Tilman Schmidt9333.10%583.33%
Total281100.00%6100.00%

/* atrdy_timeout * timeout routine for AT command transmission * argument: * controller state structure */
static void atrdy_timeout(unsigned long data) { struct cardstate *cs = (struct cardstate *) data; struct bas_cardstate *ucs = cs->hw.bas; dev_warn(cs->dev, "timeout waiting for HD_READY_SEND_ATDATA\n"); /* fake the missing signal - what else can I do? */ update_basstate(ucs, BS_ATREADY, BS_ATTIMER); start_cbsend(cs); }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp4985.96%133.33%
Tilman Schmidt814.04%266.67%
Total57100.00%3100.00%

/* atwrite_submit * submit an HD_WRITE_ATMESSAGE command URB * parameters: * cs controller state structure * buf buffer containing command to send * len length of command to send * return value: * 0 on success * -EBUSY if another request is pending * any URB submission error code */
static int atwrite_submit(struct cardstate *cs, unsigned char *buf, int len) { struct bas_cardstate *ucs = cs->hw.bas; int rc; gig_dbg(DEBUG_USBREQ, "-------> HD_WRITE_ATMESSAGE (%d)", len); if (update_basstate(ucs, BS_ATWRPEND, 0) & BS_ATWRPEND) { dev_err(cs->dev, "could not submit HD_WRITE_ATMESSAGE: URB busy\n"); return -EBUSY; } ucs->dr_cmd_out.bRequestType = OUT_VENDOR_REQ; ucs->dr_cmd_out.bRequest = HD_WRITE_ATMESSAGE; ucs->dr_cmd_out.wValue = 0; ucs->dr_cmd_out.wIndex = 0; ucs->dr_cmd_out.wLength = cpu_to_le16(len); usb_fill_control_urb(ucs->urb_cmd_out, ucs->udev, usb_sndctrlpipe(ucs->udev, 0), (unsigned char *) &ucs->dr_cmd_out, buf, len, write_command_callback, cs); rc = usb_submit_urb(ucs->urb_cmd_out, GFP_ATOMIC); if (unlikely(rc)) { update_basstate(ucs, 0, BS_ATWRPEND); dev_err(cs->dev, "could not submit HD_WRITE_ATMESSAGE: %s\n", get_usb_rcmsg(rc)); return rc; } /* submitted successfully, start timeout if necessary */ if (!(update_basstate(ucs, BS_ATTIMER, BS_ATREADY) & BS_ATTIMER)) { gig_dbg(DEBUG_OUTPUT, "setting ATREADY timeout of %d/10 secs", ATRDY_TIMEOUT); mod_timer(&ucs->timer_atrdy, jiffies + ATRDY_TIMEOUT * HZ / 10); } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp18575.82%114.29%
Tilman Schmidt5823.77%571.43%
Christoph Lameter10.41%114.29%
Total244100.00%7100.00%

/* start_cbsend * start transmission of AT command queue if necessary * parameter: * cs controller state structure * return value: * 0 on success * error code < 0 on error */
static int start_cbsend(struct cardstate *cs) { struct cmdbuf_t *cb; struct bas_cardstate *ucs = cs->hw.bas; unsigned long flags; int rc; int retval = 0; /* check if suspend requested */ if (ucs->basstate & BS_SUSPEND) { gig_dbg(DEBUG_OUTPUT, "suspending"); return -EHOSTUNREACH; } /* check if AT channel is open */ if (!(ucs->basstate & BS_ATOPEN)) { gig_dbg(DEBUG_OUTPUT, "AT channel not open"); rc = req_submit(cs->bcs, HD_OPEN_ATCHANNEL, 0, BAS_TIMEOUT); if (rc < 0) { /* flush command queue */ spin_lock_irqsave(&cs->cmdlock, flags); while (cs->cmdbuf != NULL) complete_cb(cs); spin_unlock_irqrestore(&cs->cmdlock, flags); } return rc; } /* try to send first command in queue */ spin_lock_irqsave(&cs->cmdlock, flags); while ((cb = cs->cmdbuf) != NULL && (ucs->basstate & BS_ATREADY)) { ucs->retry_cmd_out = 0; rc = atwrite_submit(cs, cb->buf, cb->len); if (unlikely(rc)) { retval = rc; complete_cb(cs); } } spin_unlock_irqrestore(&cs->cmdlock, flags); return retval; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp19585.90%116.67%
Tilman Schmidt3214.10%583.33%
Total227100.00%6100.00%

/* gigaset_write_cmd * This function is called by the device independent part of the driver * to transmit an AT command string to the Gigaset device. * It encapsulates the device specific method for transmission over the * direct USB connection to the base. * The command string is added to the queue of commands to send, and * USB transmission is started if necessary. * parameters: * cs controller state structure * cb command buffer structure * return value: * number of bytes queued on success * error code < 0 on error */
static int gigaset_write_cmd(struct cardstate *cs, struct cmdbuf_t *cb) { unsigned long flags; int rc; gigaset_dbg_buffer(cs->mstate != MS_LOCKED ? DEBUG_TRANSCMD : DEBUG_LOCKCMD, "CMD Transmit", cb->len, cb->buf); /* translate "+++" escape sequence sent as a single separate command * into "close AT channel" command for error recovery * The next command will reopen the AT channel automatically. */ if (cb->len == 3 && !memcmp(cb->buf, "+++", 3)) { /* If an HD_RECEIVEATDATA_ACK message remains unhandled * because of an error, the base never sends another one. * The response channel is thus effectively blocked. * Closing and reopening the AT channel does *not* clear * this condition. * As a stopgap measure, submit a zero-length AT read * before closing the AT channel. This has the undocumented * effect of triggering a new HD_RECEIVEATDATA_ACK message * from the base if necessary. * The subsequent AT channel close then discards any pending * messages. */ spin_lock_irqsave(&cs->lock, flags); if (!(cs->hw.bas->basstate & BS_ATRDPEND)) { kfree(cs->hw.bas->rcvbuf); cs->hw.bas->rcvbuf = NULL; cs->hw.bas->rcvbuf_size = 0; cs->hw.bas->retry_cmd_in = 0; atread_submit(cs, 0); } spin_unlock_irqrestore(&cs->lock, flags); rc = req_submit(cs->bcs, HD_CLOSE_ATCHANNEL, 0, BAS_TIMEOUT); if (cb->wake_tasklet) tasklet_schedule(cb->wake_tasklet); if (!rc) rc = cb->len; kfree(cb); return rc; } spin_lock_irqsave(&cs->cmdlock, flags); cb->prev = cs->lastcmdbuf; if (cs->lastcmdbuf) cs->lastcmdbuf->next = cb; else { cs->cmdbuf = cb; cs->curlen = cb->len; } cs->cmdbytes += cb->len; cs->lastcmdbuf = cb; spin_unlock_irqrestore(&cs->cmdlock, flags); spin_lock_irqsave(&cs->lock, flags); if (unlikely(!cs->connected)) { spin_unlock_irqrestore(&cs->lock, flags); gig_dbg(DEBUG_USBREQ, "%s: not connected", __func__); /* flush command queue */ spin_lock_irqsave(&cs->cmdlock, flags); while (cs->cmdbuf != NULL) complete_cb(cs); spin_unlock_irqrestore(&cs->cmdlock, flags); return -ENODEV; } rc = start_cbsend(cs); spin_unlock_irqrestore(&cs->lock, flags); return rc < 0 ? rc : cb->len; }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt24363.61%675.00%
Hansjoerg Lipp12632.98%112.50%
Dan Carpenter133.40%112.50%
Total382100.00%8100.00%

/* gigaset_write_room * tty_driver.write_room interface routine * return number of characters the driver will accept to be written via * gigaset_write_cmd * parameter: * controller state structure * return value: * number of characters */
static int gigaset_write_room(struct cardstate *cs) { return IF_WRITEBUF; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp14100.00%1100.00%
Total14100.00%1100.00%

/* gigaset_chars_in_buffer * tty_driver.chars_in_buffer interface routine * return number of characters waiting to be sent * parameter: * controller state structure * return value: * number of characters */
static int gigaset_chars_in_buffer(struct cardstate *cs) { return cs->cmdbytes; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp1593.75%150.00%
Tilman Schmidt16.25%150.00%
Total16100.00%2100.00%

/* gigaset_brkchars * implementation of ioctl(GIGASET_BRKCHARS) * parameter: * controller state structure * return value: * -EINVAL (unimplemented function) */
static int gigaset_brkchars(struct cardstate *cs, const unsigned char buf[6]) { return -EINVAL; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp23100.00%1100.00%
Total23100.00%1100.00%

/* Device Initialization/Shutdown */ /* ============================== */ /* Free hardware dependent part of the B channel structure * parameter: * bcs B channel structure */
static void gigaset_freebcshw(struct bc_state *bcs) { struct bas_bc_state *ubc = bcs->hw.bas; int i; if (!ubc) return; /* kill URBs and tasklets before freeing - better safe than sorry */ ubc->running = 0; gig_dbg(DEBUG_INIT, "%s: killing isoc URBs", __func__); for (i = 0; i < BAS_OUTURBS; ++i) { usb_kill_urb(ubc->isoouturbs[i].urb); usb_free_urb(ubc->isoouturbs[i].urb); } for (i = 0; i < BAS_INURBS; ++i) { usb_kill_urb(ubc->isoinurbs[i]); usb_free_urb(ubc->isoinurbs[i]); } tasklet_kill(&ubc->sent_tasklet); tasklet_kill(&ubc->rcvd_tasklet); kfree(ubc->isooutbuf); kfree(ubc); bcs->hw.bas = NULL; }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt12378.34%583.33%
Hansjoerg Lipp3421.66%116.67%
Total157100.00%6100.00%

/* Initialize hardware dependent part of the B channel structure * parameter: * bcs B channel structure * return value: * 0 on success, error code < 0 on failure */
static int gigaset_initbcshw(struct bc_state *bcs) { int i; struct bas_bc_state *ubc; bcs->hw.bas = ubc = kmalloc(sizeof(struct bas_bc_state), GFP_KERNEL); if (!ubc) { pr_err("out of memory\n"); return -ENOMEM; } ubc->running = 0; atomic_set(&ubc->corrbytes, 0); spin_lock_init(&ubc->isooutlock); for (i = 0; i < BAS_OUTURBS; ++i) { ubc->isoouturbs[i].urb = NULL; ubc->isoouturbs[i].bcs = bcs; } ubc->isooutdone = ubc->isooutfree = ubc->isooutovfl = NULL; ubc->numsub = 0; ubc->isooutbuf = kmalloc(sizeof(struct isowbuf_t), GFP_KERNEL); if (!ubc->isooutbuf) { pr_err("out of memory\n"); kfree(ubc); bcs->hw.bas = NULL; return -ENOMEM; } tasklet_init(&ubc->sent_tasklet, write_iso_tasklet, (unsigned long) bcs); spin_lock_init(&ubc->isoinlock); for (i = 0; i < BAS_INURBS; ++i) ubc->isoinurbs[i] = NULL; ubc->isoindone = NULL; ubc->loststatus = -EINPROGRESS; ubc->isoinlost = 0; ubc->seqlen = 0; ubc->inbyte = 0; ubc->inbits = 0; ubc->goodbytes = 0; ubc->alignerrs = 0; ubc->fcserrs = 0; ubc->frameerrs = 0; ubc->giants = 0; ubc->runts = 0; ubc->aborts = 0; ubc->shared0s = 0; ubc->stolen0s = 0; tasklet_init(&ubc->rcvd_tasklet, read_iso_tasklet, (unsigned long) bcs); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp32094.96%120.00%
Tilman Schmidt175.04%480.00%
Total337100.00%5100.00%


static void gigaset_reinitbcshw(struct bc_state *bcs) { struct bas_bc_state *ubc = bcs->hw.bas; bcs->hw.bas->running = 0; atomic_set(&bcs->hw.bas->corrbytes, 0); bcs->hw.bas->numsub = 0; spin_lock_init(&ubc->isooutlock); spin_lock_init(&ubc->isoinlock); ubc->loststatus = -EINPROGRESS; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp7898.73%150.00%
Tilman Schmidt11.27%150.00%
Total79100.00%2100.00%


static void gigaset_freecshw(struct cardstate *cs) { /* timers, URBs and rcvbuf are disposed of in disconnect */ kfree(cs->hw.bas->int_in_buf); kfree(cs->hw.bas); cs->hw.bas = NULL; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp2460.00%133.33%
Tilman Schmidt1640.00%266.67%
Total40100.00%3100.00%

/* Initialize hardware dependent part of the cardstate structure * parameter: * cs cardstate structure * return value: * 0 on success, error code < 0 on failure */
static int gigaset_initcshw(struct cardstate *cs) { struct bas_cardstate *ucs; cs->hw.bas = ucs = kmalloc(sizeof *ucs, GFP_KERNEL); if (!ucs) { pr_err("out of memory\n"); return -ENOMEM; } ucs->int_in_buf = kmalloc(IP_MSGSIZE, GFP_KERNEL); if (!ucs->int_in_buf) { kfree(ucs); pr_err("out of memory\n"); return -ENOMEM; } ucs->urb_cmd_in = NULL; ucs->urb_cmd_out = NULL; ucs->rcvbuf = NULL; ucs->rcvbuf_size = 0; spin_lock_init(&ucs->lock); ucs->pending = 0; ucs->basstate = 0; setup_timer(&ucs->timer_ctrl, req_timeout, (unsigned long) cs); setup_timer(&ucs->timer_atrdy, atrdy_timeout, (unsigned long) cs); setup_timer(&ucs->timer_cmd_in, cmd_in_timeout, (unsigned long) cs); setup_timer(&ucs->timer_int_in, int_in_resubmit, (unsigned long) cs); init_waitqueue_head(&ucs->waitqueue); INIT_WORK(&ucs->int_in_wq, int_in_work); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp10650.00%112.50%
Tilman Schmidt10650.00%787.50%
Total212100.00%8100.00%

/* freeurbs * unlink and deallocate all URBs unconditionally * caller must make sure that no commands are still in progress * parameter: * cs controller state structure */
static void freeurbs(struct cardstate *cs) { struct bas_cardstate *ucs = cs->hw.bas; struct bas_bc_state *ubc; int i, j; gig_dbg(DEBUG_INIT, "%s: killing URBs", __func__); for (j = 0; j < BAS_CHANNELS; ++j) { ubc = cs->bcs[j].hw.bas; for (i = 0; i < BAS_OUTURBS; ++i) { usb_kill_urb(ubc->isoouturbs[i].urb); usb_free_urb(ubc->isoouturbs[i].urb); ubc->isoouturbs[i].urb = NULL; } for (i = 0; i < BAS_INURBS; ++i) { usb_kill_urb(ubc->isoinurbs[i]); usb_free_urb(ubc->isoinurbs[i]); ubc->isoinurbs[i] = NULL; } } usb_kill_urb(ucs->urb_int_in); usb_free_urb(ucs->urb_int_in); ucs->urb_int_in = NULL; usb_kill_urb(ucs->urb_cmd_out); usb_free_urb(ucs->urb_cmd_out); ucs->urb_cmd_out = NULL; usb_kill_urb(ucs->urb_cmd_in); usb_free_urb(ucs->urb_cmd_in); ucs->urb_cmd_in = NULL; usb_kill_urb(ucs->urb_ctrl); usb_free_urb(ucs->urb_ctrl); ucs->urb_ctrl = NULL; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp22793.42%125.00%
Tilman Schmidt166.58%375.00%
Total243100.00%4100.00%

/* gigaset_probe * This function is called when a new USB device is connected. * It checks whether the new device is handled by this driver. */
static int gigaset_probe(struct usb_interface *interface, const struct usb_device_id *id) { struct usb_host_interface *hostif; struct usb_device *udev = interface_to_usbdev(interface); struct cardstate *cs = NULL; struct bas_cardstate *ucs = NULL; struct bas_bc_state *ubc; struct usb_endpoint_descriptor *endpoint; int i, j; int rc; gig_dbg(DEBUG_INIT, "%s: Check if device matches .. (Vendor: 0x%x, Product: 0x%x)", __func__, le16_to_cpu(udev->descriptor.idVendor), le16_to_cpu(udev->descriptor.idProduct)); /* set required alternate setting */ hostif = interface->cur_altsetting; if (hostif->desc.bAlternateSetting != 3) { gig_dbg(DEBUG_INIT, "%s: wrong alternate setting %d - trying to switch", __func__, hostif->desc.bAlternateSetting); if (usb_set_interface(udev, hostif->desc.bInterfaceNumber, 3) < 0) { dev_warn(&udev->dev, "usb_set_interface failed, " "device %d interface %d altsetting %d\n", udev->devnum, hostif->desc.bInterfaceNumber, hostif->desc.bAlternateSetting); return -ENODEV; } hostif = interface->cur_altsetting; } /* Reject application specific interfaces */ if (hostif->desc.bInterfaceClass != 255) { dev_warn(&udev->dev, "%s: bInterfaceClass == %d\n", __func__, hostif->desc.bInterfaceClass); return -ENODEV; } if (hostif->desc.bNumEndpoints < 1) return -ENODEV; dev_info(&udev->dev, "%s: Device matched (Vendor: 0x%x, Product: 0x%x)\n", __func__, le16_to_cpu(udev->descriptor.idVendor), le16_to_cpu(udev->descriptor.idProduct)); /* allocate memory for our device state and initialize it */ cs = gigaset_initcs(driver, BAS_CHANNELS, 0, 0, cidmode, GIGASET_MODULENAME); if (!cs) return -ENODEV; ucs = cs->hw.bas; /* save off device structure ptrs for later use */ usb_get_dev(udev); ucs->udev = udev; ucs->interface = interface; cs->dev = &interface->dev; /* allocate URBs: * - one for the interrupt pipe * - three for the different uses of the default control pipe * - three for each isochronous pipe */ if (!(ucs->urb_int_in = usb_alloc_urb(0, GFP_KERNEL)) || !(ucs->urb_cmd_in = usb_alloc_urb(0, GFP_KERNEL)) || !(ucs->urb_cmd_out = usb_alloc_urb(0, GFP_KERNEL)) || !(ucs->urb_ctrl = usb_alloc_urb(0, GFP_KERNEL))) goto allocerr; for (j = 0; j < BAS_CHANNELS; ++j) { ubc = cs->bcs[j].hw.bas; for (i = 0; i < BAS_OUTURBS; ++i) if (!(ubc->isoouturbs[i].urb = usb_alloc_urb(BAS_NUMFRAMES, GFP_KERNEL))) goto allocerr; for (i = 0; i < BAS_INURBS; ++i) if (!(ubc->isoinurbs[i] = usb_alloc_urb(BAS_NUMFRAMES, GFP_KERNEL))) goto allocerr; } ucs->rcvbuf = NULL; ucs->rcvbuf_size = 0; /* Fill the interrupt urb and send it to the core */ endpoint = &hostif->endpoint[0].desc; usb_fill_int_urb(ucs->urb_int_in, udev, usb_rcvintpipe(udev, usb_endpoint_num(endpoint)), ucs->int_in_buf, IP_MSGSIZE, read_int_callback, cs, endpoint->bInterval); rc = usb_submit_urb(ucs->urb_int_in, GFP_KERNEL); if (rc != 0) { dev_err(cs->dev, "could not submit interrupt URB: %s\n", get_usb_rcmsg(rc)); goto error; } ucs->retry_int_in = 0; /* tell the device that the driver is ready */ rc = req_submit(cs->bcs, HD_DEVICE_INIT_ACK, 0, 0); if (rc != 0) goto error; /* tell common part that the device is ready */ if (startmode == SM_LOCKED) cs->mstate = MS_LOCKED; /* save address of controller structure */ usb_set_intfdata(interface, cs); rc = gigaset_start(cs); if (rc < 0) goto error; return 0; allocerr: dev_err(cs->dev, "could not allocate URBs\n"); rc = -ENOMEM; error: freeurbs(cs); usb_set_intfdata(interface, NULL); usb_put_dev(udev); gigaset_freecs(cs); return rc; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp51273.99%15.56%
Tilman Schmidt15021.68%1266.67%
Johan Hovold142.02%15.56%
Christoph Lameter71.01%15.56%
Alexey Khoroshilov50.72%15.56%
Himangi Saraogi30.43%15.56%
Uwe Kleine-König10.14%15.56%
Total692100.00%18100.00%

/* gigaset_disconnect * This function is called when the Gigaset base is unplugged. */
static void gigaset_disconnect(struct usb_interface *interface) { struct cardstate *cs; struct bas_cardstate *ucs; int j; cs = usb_get_intfdata(interface); ucs = cs->hw.bas; dev_info(cs->dev, "disconnecting Gigaset base\n"); /* mark base as not ready, all channels disconnected */ ucs->basstate = 0; /* tell LL all channels are down */ for (j = 0; j < BAS_CHANNELS; ++j) gigaset_bchannel_down(cs->bcs + j); /* stop driver (common part) */ gigaset_stop(cs); /* stop delayed work and URBs, free ressources */ del_timer_sync(&ucs->timer_ctrl); del_timer_sync(&ucs->timer_atrdy); del_timer_sync(&ucs->timer_cmd_in); del_timer_sync(&ucs->timer_int_in); cancel_work_sync(&ucs->int_in_wq); freeurbs(cs); usb_set_intfdata(interface, NULL); kfree(ucs->rcvbuf); ucs->rcvbuf = NULL; ucs->rcvbuf_size = 0; usb_put_dev(ucs->udev); ucs->interface = NULL; ucs->udev = NULL; cs->dev = NULL; gigaset_freecs(cs); }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt11461.29%787.50%
Hansjoerg Lipp7238.71%112.50%
Total186100.00%8100.00%

/* gigaset_suspend * This function is called before the USB connection is suspended * or before the USB device is reset. * In the latter case, message == PMSG_ON. */
static int gigaset_suspend(struct usb_interface *intf, pm_message_t message) { struct cardstate *cs = usb_get_intfdata(intf); struct bas_cardstate *ucs = cs->hw.bas; int rc; /* set suspend flag; this stops AT command/response traffic */ if (update_basstate(ucs, BS_SUSPEND, 0) & BS_SUSPEND) { gig_dbg(DEBUG_SUSPEND, "already suspended"); return 0; } /* wait a bit for blocking conditions to go away */ rc = wait_event_timeout(ucs->waitqueue, !(ucs->basstate & (BS_B1OPEN | BS_B2OPEN | BS_ATRDPEND | BS_ATWRPEND)), BAS_TIMEOUT * HZ / 10); gig_dbg(DEBUG_SUSPEND, "wait_event_timeout() -> %d", rc); /* check for conditions preventing suspend */ if (ucs->basstate & (BS_B1OPEN | BS_B2OPEN | BS_ATRDPEND | BS_ATWRPEND)) { dev_warn(cs->dev, "cannot suspend:\n"); if (ucs->basstate & BS_B1OPEN) dev_warn(cs->dev, " B channel 1 open\n"); if (ucs->basstate & BS_B2OPEN) dev_warn(cs->dev, " B channel 2 open\n"); if (ucs->basstate & BS_ATRDPEND) dev_warn(cs->dev, " receiving AT reply\n"); if (ucs->basstate & BS_ATWRPEND) dev_warn(cs->dev, " sending AT command\n"); update_basstate(ucs, 0, BS_SUSPEND); return -EBUSY; } /* close AT channel if open */ if (ucs->basstate & BS_ATOPEN) { gig_dbg(DEBUG_SUSPEND, "closing AT channel"); rc = req_submit(cs->bcs, HD_CLOSE_ATCHANNEL, 0, 0); if (rc) { update_basstate(ucs, 0, BS_SUSPEND); return rc; } wait_event_timeout(ucs->waitqueue, !ucs->pending, BAS_TIMEOUT * HZ / 10); /* in case of timeout, proceed anyway */ } /* kill all URBs and delayed work that might still be pending */ usb_kill_urb(ucs->urb_ctrl); usb_kill_urb(ucs->urb_int_in); del_timer_sync(&ucs->timer_ctrl); del_timer_sync(&ucs->timer_atrdy); del_timer_sync(&ucs->timer_cmd_in); del_timer_sync(&ucs->timer_int_in); /* don't try to cancel int_in_wq from within reset as it * might be the one requesting the reset */ if (message.event != PM_EVENT_ON) cancel_work_sync(&ucs->int_in_wq); gig_dbg(DEBUG_SUSPEND, "suspend complete"); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt359100.00%5100.00%
Total359100.00%5100.00%

/* gigaset_resume * This function is called after the USB connection has been resumed. */
static int gigaset_resume(struct usb_interface *intf) { struct cardstate *cs = usb_get_intfdata(intf); struct bas_cardstate *ucs = cs->hw.bas; int rc; /* resubmit interrupt URB for spontaneous messages from base */ rc = usb_submit_urb(ucs->urb_int_in, GFP_KERNEL); if (rc) { dev_err(cs->dev, "could not resubmit interrupt URB: %s\n", get_usb_rcmsg(rc)); return rc; } ucs->retry_int_in = 0; /* clear suspend flag to reallow activity */ update_basstate(ucs, 0, BS_SUSPEND); gig_dbg(DEBUG_SUSPEND, "resume complete"); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt96100.00%2100.00%
Total96100.00%2100.00%

/* gigaset_pre_reset * This function is called before the USB connection is reset. */
static int gigaset_pre_reset(struct usb_interface *intf) { /* handle just like suspend */ return gigaset_suspend(intf, PMSG_ON); }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt20100.00%1100.00%
Total20100.00%1100.00%

/* gigaset_post_reset * This function is called after the USB connection has been reset. */
static int gigaset_post_reset(struct usb_interface *intf) { /* FIXME: send HD_DEVICE_INIT_ACK? */ /* resume operations */ return gigaset_resume(intf); }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt19100.00%1100.00%
Total19100.00%1100.00%

static const struct gigaset_ops gigops = { .write_cmd = gigaset_write_cmd, .write_room = gigaset_write_room, .chars_in_buffer = gigaset_chars_in_buffer, .brkchars = gigaset_brkchars, .init_bchannel = gigaset_init_bchannel, .close_bchannel = gigaset_close_bchannel, .initbcshw = gigaset_initbcshw, .freebcshw = gigaset_freebcshw, .reinitbcshw = gigaset_reinitbcshw, .initcshw = gigaset_initcshw, .freecshw = gigaset_freecshw, .set_modem_ctrl = gigaset_set_modem_ctrl, .baud_rate = gigaset_baud_rate, .set_line_ctrl = gigaset_set_line_ctrl, .send_skb = gigaset_isoc_send_skb, .handle_input = gigaset_isoc_input, }; /* bas_gigaset_init * This function is called after the kernel module is loaded. */
static int __init bas_gigaset_init(void) { int result; /* allocate memory for our driver state and initialize it */ driver = gigaset_initdriver(GIGASET_MINOR, GIGASET_MINORS, GIGASET_MODULENAME, GIGASET_DEVNAME, &gigops, THIS_MODULE); if (driver == NULL) goto error; /* register this driver with the USB subsystem */ result = usb_register(&gigaset_usb_driver); if (result < 0) { pr_err("error %d registering USB driver\n", -result); goto error; } pr_info(DRIVER_DESC "\n"); return 0; error: if (driver) gigaset_freedriver(driver); driver = NULL; return -1; }

Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp8588.54%120.00%
Tilman Schmidt1010.42%360.00%
Uwe Kleine-König11.04%120.00%
Total96100.00%5100.00%

/* bas_gigaset_exit * This function is called before the kernel module is unloaded. */
static void __exit bas_gigaset_exit(void) { struct bas_cardstate *ucs; int i; gigaset_blockdriver(driver); /* => probe will fail * => no gigaset_start any more */ /* stop all connected devices */ for (i = 0; i < driver->minors; i++) { if (gigaset_shutdown(driver->cs + i) < 0) continue; /* no device */ /* from now on, no isdn callback should be possible */ /* close all still open channels */ ucs = driver->cs[i].hw.bas; if (ucs->basstate & BS_B1OPEN) { gig_dbg(DEBUG_INIT, "closing B1 channel"); usb_control_msg(ucs->udev, usb_sndctrlpipe(ucs->udev, 0), HD_CLOSE_B1CHANNEL, OUT_VENDOR_REQ, 0, 0, NULL, 0, BAS_TIMEOUT); } if (ucs->basstate & BS_B2OPEN) { gig_dbg(DEBUG_INIT, "closing B2 channel"); usb_control_msg(ucs->udev, usb_sndctrlpipe(ucs->udev, 0), HD_CLOSE_B2CHANNEL, OUT_VENDOR_REQ, 0, 0, NULL, 0, BAS_TIMEOUT); } if (ucs->basstate & BS_ATOPEN) { gig_dbg(DEBUG_INIT, "closing AT channel"); usb_control_msg(ucs->udev, usb_sndctrlpipe(ucs->udev, 0), HD_CLOSE_ATCHANNEL, OUT_VENDOR_REQ, 0, 0, NULL, 0, BAS_TIMEOUT); } ucs->basstate = 0; } /* deregister this driver with the USB subsystem */ usb_deregister(&gigaset_usb_driver); /* this will call the disconnect-callback */ /* from now on, no disconnect/probe callback should be running */ gigaset_freedriver(driver); driver = NULL; }

Contributors

PersonTokensPropCommitsCommitProp
Tilman Schmidt17775.00%480.00%
Hansjoerg Lipp5925.00%120.00%
Total236100.00%5100.00%

module_init(bas_gigaset_init); module_exit(bas_gigaset_exit); MODULE_AUTHOR(DRIVER_AUTHOR); MODULE_DESCRIPTION(DRIVER_DESC); MODULE_LICENSE("GPL");

Overall Contributors

PersonTokensPropCommitsCommitProp
Hansjoerg Lipp694058.02%12.04%
Tilman Schmidt491341.07%3775.51%
Kees Cook480.40%12.04%
Christoph Lameter170.14%24.08%
Johan Hovold140.12%12.04%
Dan Carpenter130.11%12.04%
Alexey Khoroshilov50.04%12.04%
Sarah Sharp50.04%12.04%
Himangi Saraogi30.03%12.04%
Uwe Kleine-König20.02%12.04%
Pete Zaitcev10.01%12.04%
Andrew Morton10.01%12.04%
Total11962100.00%49100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.