⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 speedtch.c

📁 优龙2410linux2.6.8内核源代码
💻 C
📖 第 1 页 / 共 3 页
字号:
	else		ctrl->pdu_padding = zero_padding;	ctrl->aal5_trailer [0] = 0; /* UU = 0 */	ctrl->aal5_trailer [1] = 0; /* CPI = 0 */	ctrl->aal5_trailer [2] = skb->len >> 8;	ctrl->aal5_trailer [3] = skb->len;	crc = crc32_be (~0, skb->data, skb->len);	crc = crc32_be (crc, zeros, zero_padding);	crc = crc32_be (crc, ctrl->aal5_trailer, 4);	crc = ~crc;	ctrl->aal5_trailer [4] = crc >> 24;	ctrl->aal5_trailer [5] = crc >> 16;	ctrl->aal5_trailer [6] = crc >> 8;	ctrl->aal5_trailer [7] = crc;}static unsigned int udsl_write_cells (unsigned int howmany, struct sk_buff *skb, unsigned char **target_p){	struct udsl_control *ctrl = UDSL_SKB (skb);	unsigned char *target = *target_p;	unsigned int nc, ne, i;	vdbg ("udsl_write_cells: howmany=%u, skb->len=%d, num_cells=%u, num_entire=%u, pdu_padding=%u", howmany, skb->len, ctrl->num_cells, ctrl->num_entire, ctrl->pdu_padding);	nc = ctrl->num_cells;	ne = min (howmany, ctrl->num_entire);	for (i = 0; i < ne; i++) {		memcpy (target, ctrl->cell_header, ATM_CELL_HEADER);		target += ATM_CELL_HEADER;		memcpy (target, skb->data, ATM_CELL_PAYLOAD);		target += ATM_CELL_PAYLOAD;		__skb_pull (skb, ATM_CELL_PAYLOAD);	}	ctrl->num_entire -= ne;	if (!(ctrl->num_cells -= ne) || !(howmany -= ne))		goto out;	memcpy (target, ctrl->cell_header, ATM_CELL_HEADER);	target += ATM_CELL_HEADER;	memcpy (target, skb->data, skb->len);	target += skb->len;	__skb_pull (skb, skb->len);	memset (target, 0, ctrl->pdu_padding);	target += ctrl->pdu_padding;	if (--ctrl->num_cells) {		if (!--howmany) {			ctrl->pdu_padding = ATM_CELL_PAYLOAD - ATM_AAL5_TRAILER;			goto out;		}		memcpy (target, ctrl->cell_header, ATM_CELL_HEADER);		target += ATM_CELL_HEADER;		memset (target, 0, ATM_CELL_PAYLOAD - ATM_AAL5_TRAILER);		target += ATM_CELL_PAYLOAD - ATM_AAL5_TRAILER;		DEBUG_ON (--ctrl->num_cells);	}	memcpy (target, ctrl->aal5_trailer, ATM_AAL5_TRAILER);	target += ATM_AAL5_TRAILER;	/* set pti bit in last cell */	*(target + 3 - ATM_CELL_SIZE) |= 0x2;out:	*target_p = target;	return nc - ctrl->num_cells;}/****************  receive  ****************/static void udsl_complete_receive (struct urb *urb, struct pt_regs *regs){	struct udsl_receive_buffer *buf;	struct udsl_instance_data *instance;	struct udsl_receiver *rcv;	unsigned long flags;	if (!urb || !(rcv = urb->context)) {		dbg ("udsl_complete_receive: bad urb!");		return;	}	instance = rcv->instance;	buf = rcv->buffer;	buf->filled_cells = urb->actual_length / ATM_CELL_SIZE;	vdbg ("udsl_complete_receive: urb 0x%p, status %d, actual_length %d, filled_cells %u, rcv 0x%p, buf 0x%p", urb, urb->status, urb->actual_length, buf->filled_cells, rcv, buf);	DEBUG_ON (buf->filled_cells > rcv_buf_size);	/* may not be in_interrupt() */	spin_lock_irqsave (&instance->receive_lock, flags);	list_add (&rcv->list, &instance->spare_receivers);	list_add_tail (&buf->list, &instance->filled_receive_buffers);	if (likely (!urb->status))		tasklet_schedule (&instance->receive_tasklet);	spin_unlock_irqrestore (&instance->receive_lock, flags);}static void udsl_process_receive (unsigned long data){	struct udsl_receive_buffer *buf;	struct udsl_instance_data *instance = (struct udsl_instance_data *) data;	struct udsl_receiver *rcv;	int err;made_progress:	while (!list_empty (&instance->spare_receive_buffers)) {		spin_lock_irq (&instance->receive_lock);		if (list_empty (&instance->spare_receivers)) {			spin_unlock_irq (&instance->receive_lock);			break;		}		rcv = list_entry (instance->spare_receivers.next, struct udsl_receiver, list);		list_del (&rcv->list);		spin_unlock_irq (&instance->receive_lock);		buf = list_entry (instance->spare_receive_buffers.next, struct udsl_receive_buffer, list);		list_del (&buf->list);		rcv->buffer = buf;		usb_fill_bulk_urb (rcv->urb,				   instance->usb_dev,				   usb_rcvbulkpipe (instance->usb_dev, UDSL_ENDPOINT_DATA_IN),				   buf->base,				   rcv_buf_size * ATM_CELL_SIZE,				   udsl_complete_receive,				   rcv);		vdbg ("udsl_process_receive: sending urb 0x%p, rcv 0x%p, buf 0x%p", rcv->urb, rcv, buf);		if ((err = usb_submit_urb(rcv->urb, GFP_ATOMIC)) < 0) {			dbg ("udsl_process_receive: urb submission failed (%d)!", err);			list_add (&buf->list, &instance->spare_receive_buffers);			spin_lock_irq (&instance->receive_lock);			list_add (&rcv->list, &instance->spare_receivers);			spin_unlock_irq (&instance->receive_lock);			break;		}	}	spin_lock_irq (&instance->receive_lock);	if (list_empty (&instance->filled_receive_buffers)) {		spin_unlock_irq (&instance->receive_lock);		return; /* done - no more buffers */	}	buf = list_entry (instance->filled_receive_buffers.next, struct udsl_receive_buffer, list);	list_del (&buf->list);	spin_unlock_irq (&instance->receive_lock);	vdbg ("udsl_process_receive: processing buf 0x%p", buf);	udsl_extract_cells (instance, buf->base, buf->filled_cells);	list_add (&buf->list, &instance->spare_receive_buffers);	goto made_progress;}/*************  send  *************/static void udsl_complete_send (struct urb *urb, struct pt_regs *regs){	struct udsl_instance_data *instance;	struct udsl_sender *snd;	unsigned long flags;	if (!urb || !(snd = urb->context) || !(instance = snd->instance)) {		dbg ("udsl_complete_send: bad urb!");		return;	}	vdbg ("udsl_complete_send: urb 0x%p, status %d, snd 0x%p, buf 0x%p", urb, urb->status, snd, snd->buffer);	/* may not be in_interrupt() */	spin_lock_irqsave (&instance->send_lock, flags);	list_add (&snd->list, &instance->spare_senders);	list_add (&snd->buffer->list, &instance->spare_send_buffers);	tasklet_schedule (&instance->send_tasklet);	spin_unlock_irqrestore (&instance->send_lock, flags);}static void udsl_process_send (unsigned long data){	struct udsl_send_buffer *buf;	struct udsl_instance_data *instance = (struct udsl_instance_data *) data;	struct sk_buff *skb;	struct udsl_sender *snd;	int err;	unsigned int num_written;made_progress:	spin_lock_irq (&instance->send_lock);	while (!list_empty (&instance->spare_senders)) {		if (!list_empty (&instance->filled_send_buffers)) {			buf = list_entry (instance->filled_send_buffers.next, struct udsl_send_buffer, list);			list_del (&buf->list);		} else if ((buf = instance->current_buffer)) {			instance->current_buffer = NULL;		} else /* all buffers empty */			break;		snd = list_entry (instance->spare_senders.next, struct udsl_sender, list);		list_del (&snd->list);		spin_unlock_irq (&instance->send_lock);		snd->buffer = buf;	        usb_fill_bulk_urb (snd->urb,				   instance->usb_dev,				   usb_sndbulkpipe (instance->usb_dev, UDSL_ENDPOINT_DATA_OUT),				   buf->base,				   (snd_buf_size - buf->free_cells) * ATM_CELL_SIZE,				   udsl_complete_send,				   snd);		vdbg ("udsl_process_send: submitting urb 0x%p (%d cells), snd 0x%p, buf 0x%p", snd->urb, snd_buf_size - buf->free_cells, snd, buf);		if ((err = usb_submit_urb(snd->urb, GFP_ATOMIC)) < 0) {			dbg ("udsl_process_send: urb submission failed (%d)!", err);			spin_lock_irq (&instance->send_lock);			list_add (&snd->list, &instance->spare_senders);			spin_unlock_irq (&instance->send_lock);			list_add (&buf->list, &instance->filled_send_buffers);			return; /* bail out */		}		spin_lock_irq (&instance->send_lock);	} /* while */	spin_unlock_irq (&instance->send_lock);	if (!instance->current_skb && !(instance->current_skb = skb_dequeue (&instance->sndqueue)))		return; /* done - no more skbs */	skb = instance->current_skb;	if (!(buf = instance->current_buffer)) {		spin_lock_irq (&instance->send_lock);		if (list_empty (&instance->spare_send_buffers)) {			instance->current_buffer = NULL;			spin_unlock_irq (&instance->send_lock);			return; /* done - no more buffers */		}		buf = list_entry (instance->spare_send_buffers.next, struct udsl_send_buffer, list);		list_del (&buf->list);		spin_unlock_irq (&instance->send_lock);		buf->free_start = buf->base;		buf->free_cells = snd_buf_size;		instance->current_buffer = buf;	}	num_written = udsl_write_cells (buf->free_cells, skb, &buf->free_start);	vdbg ("udsl_process_send: wrote %u cells from skb 0x%p to buffer 0x%p", num_written, skb, buf);	if (!(buf->free_cells -= num_written)) {		list_add_tail (&buf->list, &instance->filled_send_buffers);		instance->current_buffer = NULL;	}	vdbg ("udsl_process_send: buffer contains %d cells, %d left", snd_buf_size - buf->free_cells, buf->free_cells);	if (!UDSL_SKB (skb)->num_cells) {		struct atm_vcc *vcc = UDSL_SKB (skb)->atm_data.vcc;		udsl_pop (vcc, skb);		instance->current_skb = NULL;		atomic_inc (&vcc->stats->tx);	}	goto made_progress;}static void udsl_cancel_send (struct udsl_instance_data *instance, struct atm_vcc *vcc){	struct sk_buff *skb, *n;	dbg ("udsl_cancel_send entered");	spin_lock_irq (&instance->sndqueue.lock);	for (skb = instance->sndqueue.next, n = skb->next; skb != (struct sk_buff *)&instance->sndqueue; skb = n, n = skb->next)		if (UDSL_SKB (skb)->atm_data.vcc == vcc) {			dbg ("udsl_cancel_send: popping skb 0x%p", skb);			__skb_unlink (skb, &instance->sndqueue);			udsl_pop (vcc, skb);		}	spin_unlock_irq (&instance->sndqueue.lock);	tasklet_disable (&instance->send_tasklet);	if ((skb = instance->current_skb) && (UDSL_SKB (skb)->atm_data.vcc == vcc)) {		dbg ("udsl_cancel_send: popping current skb (0x%p)", skb);		instance->current_skb = NULL;		udsl_pop (vcc, skb);	}	tasklet_enable (&instance->send_tasklet);	dbg ("udsl_cancel_send done");}static int udsl_atm_send (struct atm_vcc *vcc, struct sk_buff *skb){	struct udsl_instance_data *instance = vcc->dev->dev_data;	int err;	vdbg ("udsl_atm_send called (skb 0x%p, len %u)", skb, skb->len);	if (!instance || !instance->usb_dev) {		dbg ("udsl_atm_send: NULL data!");		err = -ENODEV;		goto fail;	}	if (vcc->qos.aal != ATM_AAL5) {		dbg ("udsl_atm_send: unsupported ATM type %d!", vcc->qos.aal);		err = -EINVAL;		goto fail;	}	if (skb->len > ATM_MAX_AAL5_PDU) {		dbg ("udsl_atm_send: packet too long (%d vs %d)!", skb->len, ATM_MAX_AAL5_PDU);		err = -EINVAL;		goto fail;	}	PACKETDEBUG (skb->data, skb->len);	udsl_groom_skb (vcc, skb);	skb_queue_tail (&instance->sndqueue, skb);	tasklet_schedule (&instance->send_tasklet);	return 0;fail:	udsl_pop (vcc, skb);	return err;}/************  ATM  ************/static void udsl_atm_dev_close (struct atm_dev *dev){	struct udsl_instance_data *instance = dev->dev_data;	if (!instance) {		dbg ("udsl_atm_dev_close: NULL instance!");		return;	}	dbg ("udsl_atm_dev_close: queue has %u elements", instance->sndqueue.qlen);	tasklet_kill (&instance->receive_tasklet);	tasklet_kill (&instance->send_tasklet);	kfree (instance);	dev->dev_data = NULL;}static int udsl_atm_proc_read (struct atm_dev *atm_dev, loff_t *pos, char *page){	struct udsl_instance_data *instance = atm_dev->dev_data;	int left = *pos;	if (!instance) {		dbg ("udsl_atm_proc_read: NULL instance!");		return -ENODEV;	}	if (!left--)		return sprintf (page, "%s\n", instance->description);	if (!left--)		return sprintf (page, "MAC: %02x:%02x:%02x:%02x:%02x:%02x\n",				atm_dev->esi [0], atm_dev->esi [1], atm_dev->esi [2],				atm_dev->esi [3], atm_dev->esi [4], atm_dev->esi [5]);	if (!left--)		return sprintf (page, "AAL5: tx %d ( %d err ), rx %d ( %d err, %d drop )\n",				atomic_read (&atm_dev->stats.aal5.tx),				atomic_read (&atm_dev->stats.aal5.tx_err),				atomic_read (&atm_dev->stats.aal5.rx),				atomic_read (&atm_dev->stats.aal5.rx_err),				atomic_read (&atm_dev->stats.aal5.rx_drop));	if (!left--) {		switch (atm_dev->signal) {		case ATM_PHY_SIG_FOUND:			sprintf (page, "Line up");			break;		case ATM_PHY_SIG_LOST:			sprintf (page, "Line down");			break;		default:			sprintf (page, "Line state unknown");			break;		}		if (instance->usb_dev) {			if (!instance->firmware_loaded)				strcat (page, ", no firmware\n");			else				strcat (page, ", firmware loaded\n");		} else			strcat (page, ", disconnected\n");		return strlen (page);	}	return 0;}static int udsl_atm_open (struct atm_vcc *vcc){	struct udsl_instance_data *instance = vcc->dev->dev_data;	struct udsl_vcc_data *new;	unsigned int max_pdu;	int vci = vcc->vci;	short vpi = vcc->vpi;	dbg ("udsl_atm_open: vpi %hd, vci %d", vpi, vci);	if (!instance || !instance->usb_dev) {		dbg ("udsl_atm_open: NULL data!");		return -ENODEV;	}	/* only support AAL5 */	if ((vcc->qos.aal != ATM_AAL5) || (vcc->qos.rxtp.max_sdu < 0) || (vcc->qos.rxtp.max_sdu > ATM_MAX_AAL5_PDU)) {		dbg ("udsl_atm_open: unsupported ATM type %d!", vcc->qos.aal);		return -EINVAL;	}	if (!instance->firmware_loaded) {		dbg ("udsl_atm_open: firmware not loaded!");		return -EAGAIN;	}	down (&instance->serialize); /* vs self, udsl_atm_close */	if (udsl_find_vcc (instance, vpi, vci)) {		dbg ("udsl_atm_open: %hd/%d already in use!", vpi, vci);		up (&instance->serialize);		return -EADDRINUSE;	}	if (!(new = kmalloc (sizeof (struct udsl_vcc_data), GFP_KERNEL))) {

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -