📄 speedtch.c
字号:
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 + -