1 //============================================================================
 
   2 //  Copyright (c) 1996-2002 Winbond Electronic Corporation
 
   8 //    Processing the Tx message and put into down layer
 
  10 //============================================================================
 
  11 #include <linux/usb.h>
 
  18 Wb35Tx_get_tx_buffer(struct hw_data * pHwData, u8 **pBuffer)
 
  20         struct wb35_tx *pWb35Tx = &pHwData->Wb35Tx;
 
  22         *pBuffer = pWb35Tx->TxBuffer[0];
 
  26 static void Wb35Tx(struct wbsoft_priv *adapter);
 
  28 static void Wb35Tx_complete(struct urb * pUrb)
 
  30         struct wbsoft_priv *adapter = pUrb->context;
 
  31         struct hw_data *        pHwData = &adapter->sHwData;
 
  32         struct wb35_tx *pWb35Tx = &pHwData->Wb35Tx;
 
  33         PMDS            pMds = &adapter->Mds;
 
  35         printk("wb35: tx complete\n");
 
  37         pWb35Tx->EP4vm_state = VM_COMPLETED;
 
  38         pWb35Tx->EP4VM_status = pUrb->status; //Store the last result of Irp
 
  39         pMds->TxOwner[ pWb35Tx->TxSendIndex ] = 0;// Set the owner. Free the owner bit always.
 
  40         pWb35Tx->TxSendIndex++;
 
  41         pWb35Tx->TxSendIndex %= MAX_USB_TX_BUFFER_NUMBER;
 
  43         if (pHwData->SurpriseRemove || pHwData->HwStop) // Let WbWlanHalt to handle surprise remove
 
  49         // The URB is completed, check the result
 
  50         if (pWb35Tx->EP4VM_status != 0) {
 
  51                 printk("URB submission failed\n");
 
  52                 pWb35Tx->EP4vm_state = VM_STOP;
 
  61         atomic_dec(&pWb35Tx->TxFireCounter);
 
  62         pWb35Tx->EP4vm_state = VM_STOP;
 
  65 static void Wb35Tx(struct wbsoft_priv *adapter)
 
  67         struct hw_data *        pHwData = &adapter->sHwData;
 
  68         struct wb35_tx *pWb35Tx = &pHwData->Wb35Tx;
 
  70         PMDS            pMds = &adapter->Mds;
 
  71         struct urb *    pUrb = (struct urb *)pWb35Tx->Tx4Urb;
 
  76         if (pHwData->SurpriseRemove || pHwData->HwStop)
 
  83         SendIndex = pWb35Tx->TxSendIndex;
 
  84         if (!pMds->TxOwner[SendIndex]) //No more data need to be sent, return immediately
 
  87         pTxBufferAddress = pWb35Tx->TxBuffer[SendIndex];
 
  91         usb_fill_bulk_urb(pUrb, pHwData->WbUsb.udev,
 
  92                           usb_sndbulkpipe(pHwData->WbUsb.udev, 4),
 
  93                           pTxBufferAddress, pMds->TxBufferSize[ SendIndex ],
 
  94                           Wb35Tx_complete, adapter);
 
  96         pWb35Tx->EP4vm_state = VM_RUNNING;
 
  97         retv = usb_submit_urb(pUrb, GFP_ATOMIC);
 
  99                 printk("EP4 Tx Irp sending error\n");
 
 103         // Check if driver needs issue Irp for EP2
 
 104         pWb35Tx->TxFillCount += pMds->TxCountInBuffer[SendIndex];
 
 105         if (pWb35Tx->TxFillCount > 12)
 
 106                 Wb35Tx_EP2VM_start(adapter);
 
 108         pWb35Tx->ByteTransfer += pMds->TxBufferSize[SendIndex];
 
 112         pWb35Tx->EP4vm_state = VM_STOP;
 
 113         atomic_dec(&pWb35Tx->TxFireCounter);
 
 116 void Wb35Tx_start(struct wbsoft_priv *adapter)
 
 118         struct hw_data * pHwData = &adapter->sHwData;
 
 119         struct wb35_tx *pWb35Tx = &pHwData->Wb35Tx;
 
 121         // Allow only one thread to run into function
 
 122         if (atomic_inc_return(&pWb35Tx->TxFireCounter) == 1) {
 
 123                 pWb35Tx->EP4vm_state = VM_RUNNING;
 
 126                 atomic_dec(&pWb35Tx->TxFireCounter);
 
 129 unsigned char Wb35Tx_initial(struct hw_data * pHwData)
 
 131         struct wb35_tx *pWb35Tx = &pHwData->Wb35Tx;
 
 133         pWb35Tx->Tx4Urb = usb_alloc_urb(0, GFP_ATOMIC);
 
 134         if (!pWb35Tx->Tx4Urb)
 
 137         pWb35Tx->Tx2Urb = usb_alloc_urb(0, GFP_ATOMIC);
 
 138         if (!pWb35Tx->Tx2Urb)
 
 140                 usb_free_urb( pWb35Tx->Tx4Urb );
 
 147 //======================================================
 
 148 void Wb35Tx_stop(struct hw_data * pHwData)
 
 150         struct wb35_tx *pWb35Tx = &pHwData->Wb35Tx;
 
 152         // Trying to canceling the Trp of EP2
 
 153         if (pWb35Tx->EP2vm_state == VM_RUNNING)
 
 154                 usb_unlink_urb( pWb35Tx->Tx2Urb ); // Only use unlink, let Wb35Tx_destrot to free them
 
 156         printk("EP2 Tx stop\n");
 
 159         // Trying to canceling the Irp of EP4
 
 160         if (pWb35Tx->EP4vm_state == VM_RUNNING)
 
 161                 usb_unlink_urb( pWb35Tx->Tx4Urb ); // Only use unlink, let Wb35Tx_destrot to free them
 
 163         printk("EP4 Tx stop\n");
 
 167 //======================================================
 
 168 void Wb35Tx_destroy(struct hw_data * pHwData)
 
 170         struct wb35_tx *pWb35Tx = &pHwData->Wb35Tx;
 
 174                 msleep(10);  // Delay for waiting function enter 940623.1.a
 
 175         } while( (pWb35Tx->EP2vm_state != VM_STOP) && (pWb35Tx->EP4vm_state != VM_STOP) );
 
 176         msleep(10);  // Delay for waiting function enter 940623.1.b
 
 179                 usb_free_urb( pWb35Tx->Tx4Urb );
 
 182                 usb_free_urb( pWb35Tx->Tx2Urb );
 
 185         printk("Wb35Tx_destroy OK\n");
 
 189 void Wb35Tx_CurrentTime(struct wbsoft_priv *adapter, u32 TimeCount)
 
 191         struct hw_data * pHwData = &adapter->sHwData;
 
 192         struct wb35_tx *pWb35Tx = &pHwData->Wb35Tx;
 
 193         unsigned char Trigger = false;
 
 195         if (pWb35Tx->TxTimer > TimeCount)
 
 197         else if (TimeCount > (pWb35Tx->TxTimer+500))
 
 201                 pWb35Tx->TxTimer = TimeCount;
 
 202                 Wb35Tx_EP2VM_start(adapter);
 
 206 static void Wb35Tx_EP2VM(struct wbsoft_priv *adapter);
 
 208 static void Wb35Tx_EP2VM_complete(struct urb * pUrb)
 
 210         struct wbsoft_priv *adapter = pUrb->context;
 
 211         struct hw_data *        pHwData = &adapter->sHwData;
 
 212         T02_DESCRIPTOR  T02, TSTATUS;
 
 213         struct wb35_tx *pWb35Tx = &pHwData->Wb35Tx;
 
 214         u32 *           pltmp = (u32 *)pWb35Tx->EP2_buf;
 
 216         u16             InterruptInLength;
 
 220         pWb35Tx->EP2vm_state = VM_COMPLETED;
 
 221         pWb35Tx->EP2VM_status = pUrb->status;
 
 223         // For Linux 2.4. Interrupt will always trigger
 
 224         if (pHwData->SurpriseRemove || pHwData->HwStop) // Let WbWlanHalt to handle surprise remove
 
 227         if (pWb35Tx->tx_halt)
 
 230         //The Urb is completed, check the result
 
 231         if (pWb35Tx->EP2VM_status != 0) {
 
 232                 printk("EP2 IoCompleteRoutine return error\n");
 
 233                 pWb35Tx->EP2vm_state= VM_STOP;
 
 237         // Update the Tx result
 
 238         InterruptInLength = pUrb->actual_length;
 
 239         // Modify for minimum memory access and DWORD alignment.
 
 240         T02.value = cpu_to_le32(pltmp[0]) >> 8; // [31:8] -> [24:0]
 
 241         InterruptInLength -= 1;// 20051221.1.c Modify the follow for more stable
 
 242         InterruptInLength >>= 2; // InterruptInLength/4
 
 243         for (i = 1; i <= InterruptInLength; i++) {
 
 244                 T02.value |= ((cpu_to_le32(pltmp[i]) & 0xff) << 24);
 
 246                 TSTATUS.value = T02.value;  //20061009 anson's endian
 
 247                 Mds_SendComplete( adapter, &TSTATUS );
 
 248                 T02.value = cpu_to_le32(pltmp[i]) >> 8;
 
 253         atomic_dec(&pWb35Tx->TxResultCount);
 
 254         pWb35Tx->EP2vm_state = VM_STOP;
 
 257 static void Wb35Tx_EP2VM(struct wbsoft_priv *adapter)
 
 259         struct hw_data *        pHwData = &adapter->sHwData;
 
 260         struct wb35_tx *pWb35Tx = &pHwData->Wb35Tx;
 
 261         struct urb *    pUrb = (struct urb *)pWb35Tx->Tx2Urb;
 
 262         u32 *   pltmp = (u32 *)pWb35Tx->EP2_buf;
 
 265         if (pHwData->SurpriseRemove || pHwData->HwStop)
 
 268         if (pWb35Tx->tx_halt)
 
 274         usb_fill_int_urb( pUrb, pHwData->WbUsb.udev, usb_rcvintpipe(pHwData->WbUsb.udev,2),
 
 275                           pltmp, MAX_INTERRUPT_LENGTH, Wb35Tx_EP2VM_complete, adapter, 32);
 
 277         pWb35Tx->EP2vm_state = VM_RUNNING;
 
 278         retv = usb_submit_urb(pUrb, GFP_ATOMIC);
 
 282                 printk("EP2 Tx Irp sending error\n");
 
 289         pWb35Tx->EP2vm_state = VM_STOP;
 
 290         atomic_dec(&pWb35Tx->TxResultCount);
 
 293 void Wb35Tx_EP2VM_start(struct wbsoft_priv *adapter)
 
 295         struct hw_data * pHwData = &adapter->sHwData;
 
 296         struct wb35_tx *pWb35Tx = &pHwData->Wb35Tx;
 
 298         // Allow only one thread to run into function
 
 299         if (atomic_inc_return(&pWb35Tx->TxResultCount) == 1) {
 
 300                 pWb35Tx->EP2vm_state = VM_RUNNING;
 
 301                 Wb35Tx_EP2VM(adapter);
 
 304                 atomic_dec(&pWb35Tx->TxResultCount);