host_snd.c 3.03 KB
/*HEADER******************************************************************
**************************************************************************
*** 
*** Copyright (c) 2001-2002 ARC International.
*** All rights reserved                                          
***                                                              
*** This software embodies materials and concepts which are      
*** confidential to ARC International and is made
*** available solely pursuant to the terms of a written license   
*** agreement with ARC International             
***
*** File: hostapi.c
***
*** Comments:      
***   This file contains the USB Host API specific functions.
***                                                               
**************************************************************************
*END*********************************************************************/

#include "os_bb.h"
#include "osint.h"

/*FUNCTION*-------------------------------------------------------------
*
*  Function Name  : _usb_host_send_data
*  Returned Value : error or status of the transfer
*  Comments       :
* The Send Data routine is non-blocking routine that causes a block of data 
* to be made available for transmission to the USB host.
*
*END*-----------------------------------------------------------------*/
uint_8 _usb_host_send_data
   (
      /* [IN] the USB Host state structure */
      _usb_host_handle  handle,
      
      /* Index into the Pipe Descriptor Array */
      int_16            pipeid,

      /* Address of the data to send */
      uchar_ptr         buff_ptr,

      /* Number of bytes to Send */
      uint_32           length
   )
{
   PIPE_DESCRIPTOR_STRUCT_PTR pipe_descr_ptr;
   USB_HOST_STATE_STRUCT_PTR usb_host_ptr;
   
   usb_host_ptr = (USB_HOST_STATE_STRUCT_PTR)handle;

   /* Check if pipe id is valid */
   if (pipeid >= usb_host_ptr->MAX_PIPES) {
      return USBERR_INVALID_PIPE_ID;
   }

   pipe_descr_ptr = &usb_host_ptr->PIPE_DESCRIPTOR_BASE_PTR[pipeid];
   USB_lock();

   /* Check if a previously queued transfer is still in progress */
   if (pipe_descr_ptr->STATUS != USB_STATUS_IDLE) {
      USB_unlock();
      return USB_STATUS_TRANSFER_IN_PROGRESS;
   }
  
   USB_unlock();
   /* Initialize the pipe descriptor with the new send request */
   pipe_descr_ptr->SOFAR = 0;
   pipe_descr_ptr->TX1_PTR = buff_ptr;
   if ((pipe_descr_ptr->PIPETYPE == USB_ISOCHRONOUS_PIPE) && 
      (pipe_descr_ptr->TODO1 > pipe_descr_ptr->MAX_PKT_SIZE)) {
      /*
      ** For isochronous pipes, we transfer a mzximum of MAX_PKT_SIZE 
      ** packet 
      */
      pipe_descr_ptr->TODO1 = pipe_descr_ptr->MAX_PKT_SIZE;
   } else {
      /* For all others max of 64KB */
      pipe_descr_ptr->TODO1 = length;
   }

   USB_lock();
   
   /* return successful transfer initiation status */
   pipe_descr_ptr->STATUS = USB_STATUS_TRANSFER_QUEUED;

   _usb_hci_vusb11_send_data(handle, pipe_descr_ptr);

   /* Indicate that a packet is pending */      
   pipe_descr_ptr->PACKETPENDING = TRUE;

   USB_unlock();
   
   return USB_STATUS_TRANSFER_QUEUED;
}