| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332 |
- /**************************************************************************/
- /*!
- @file cdc.c
- @author hathach (tinyusb.org)
- @section LICENSE
- Software License Agreement (BSD License)
- Copyright (c) 2013, hathach (tinyusb.org)
- All rights reserved.
- Redistribution and use in source and binary forms, with or without
- modification, are permitted provided that the following conditions are met:
- 1. Redistributions of source code must retain the above copyright
- notice, this list of conditions and the following disclaimer.
- 2. Redistributions in binary form must reproduce the above copyright
- notice, this list of conditions and the following disclaimer in the
- documentation and/or other materials provided with the distribution.
- 3. Neither the name of the copyright holders nor the
- names of its contributors may be used to endorse or promote products
- derived from this software without specific prior written permission.
- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ''AS IS'' AND ANY
- EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
- DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY
- DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
- INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
- LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION HOWEVER CAUSED AND
- ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- INCLUDING NEGLIGENCE OR OTHERWISE ARISING IN ANY WAY OUT OF THE USE OF THIS
- SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- This file is part of the tinyusb stack.
- */
- /**************************************************************************/
- #include "cdc.h"
- #include "common/fifo.h"
- #if defined TUSB_CFG_DEVICE_CDC && defined TUSB_CFG_DEVICE
- static USBD_HANDLE_T g_hCdc;
- static CDC_LINE_CODING line_coding;
- static uint8_t qBuffer[2][CDC_BUFFER_SIZE]; /* TX and RX buffers */
- static fifo_t ffTX, ffRX;
- void usb_cdc_recv_isr(void) ATTR_WEAK ATTR_ALIAS(usb_cdc_recv_isr_stub);
- /**************************************************************************/
- /*!
- @brief Stub for the optional CDC receive ISR that can be used
- to perform some action when data arrives via USB CDC
- */
- /**************************************************************************/
- void usb_cdc_recv_isr_stub (void)
- {
- return;
- }
- /**************************************************************************/
- /*!
- @brief Adds a single byte to the transmit buffer for USB CDC
- @param[in] c
- The byte to send
- @returns TRUE if the byte was successfully added to the TX buffer
- @note See example for tusb_cdc_getc
- */
- /**************************************************************************/
- bool tusb_cdc_putc (uint8_t c)
- {
- return fifo_write(&ffTX, c);
- }
- /**************************************************************************/
- /*!
- @brief Reads a single byte from the USB CDC buffer
- @param[in] c
- Pointer to the location where the byte should be written
- @returns TRUE if a byte was read from the buffer
- @section EXAMPLE
- @code
- // Convert incoming characters to upper case and send back via CDC
- if (usb_isConfigured())
- {
- uint8_t cdc_char;
- if( tusb_cdc_getc(&cdc_char) )
- {
- switch (cdc_char)
- {
- default :
- cdc_char = toupper(cdc_char);
- tusb_cdc_putc(cdc_char);
- break;
- }
- }
- }
- @endcode
- */
- /**************************************************************************/
- bool tusb_cdc_getc(uint8_t *c)
- {
- ASSERT(c, false); // not empty pointer
- return fifo_read(&ffRX, c);
- }
- /**************************************************************************/
- /*!
- @brief Writes the supplied buffer to the USB CDC device
- @param[in] buffer
- Pointer to the buffer that should be written via USB CDC
- @param[in] count
- The number of bytes to write
- @returns The actual number of bytes sent out via USB CDC
- @section EXAMPLE
- @code
- // Capture printf output (in Red Suite) and send it to USB CDC
- // (Note: With newlib this function should be renamed to _write)
- int __sys_write(int file, char *ptr, int len)
- {
- #ifdef CFG_PRINTF_USBCDC
- // Handle USB CDC output
- if (usb_isConfigured())
- {
- int length = len;
- while(length > 0)
- {
- uint16_t transferredCount;
- transferredCount = tusb_cdc_send( (uint8_t*) ptr, length);
- ptr += transferredCount;
- length -= transferredCount;
- }
- }
- #endif
- return len;
- }
- @endcode
- */
- /**************************************************************************/
- uint16_t tusb_cdc_send(uint8_t* buffer, uint16_t count)
- {
- uint16_t i=0;
- ASSERT(buffer && count, 0);
- while (i < count && fifo_write(&ffTX, buffer[i]) )
- {
- i++;
- }
- return i;
- }
- /**************************************************************************/
- /*!
- @brief Reads the incoming CDC buffer up to a maximum number of bytes
- @param[in] buffer
- Pointer to the buffer where data should be written
- @param[in] max
- The maximum number of bytes to read
- @returns The actual number of bytes received
- */
- /**************************************************************************/
- uint16_t tusb_cdc_recv(uint8_t* buffer, uint16_t max)
- {
- ASSERT(buffer && max, 0);
- return fifo_read_n(&ffRX, buffer, max);
- }
- #if 0
- // ROM driver bug: cannot hook this to CIC_GetRequest
- // Need this to implement GetLineCode & detect
- ErrorCode_t CDC_Control_GetRequest(USBD_HANDLE_T hUsb, USB_SETUP_PACKET *pSetup, uint8_t **pBuffer, uint16_t *length)
- {
- return LPC_OK;
- }
- #endif
- /**************************************************************************/
- /*!
- @brief TODO Add description
- */
- /**************************************************************************/
- ErrorCode_t CDC_SetLineCoding(USBD_HANDLE_T hUsb, CDC_LINE_CODING *lineCoding)
- {
- ASSERT(lineCoding, ERR_FAILED);
- memcpy(&line_coding, lineCoding, sizeof(CDC_LINE_CODING));
- return LPC_OK;
- }
- /**************************************************************************/
- /*!
- @brief TODO Add description
- */
- /**************************************************************************/
- ErrorCode_t CDC_SendBreak(USBD_HANDLE_T hCDC, uint16_t mstime)
- {
- return LPC_OK;
- }
- /**************************************************************************/
- /*!
- @brief Bulk Out handler for the USB ROM drivers (UART TX)
- */
- /**************************************************************************/
- ErrorCode_t CDC_BulkIn_Hdlr(USBD_HANDLE_T hUsb, void* data, uint32_t event)
- {
- if (USB_EVT_IN == event)
- {
- uint8_t buffer[CDC_DATA_EP_MAXPACKET_SIZE];
- uint16_t count;
- count = fifo_read_n(&ffTX, buffer, CDC_DATA_EP_MAXPACKET_SIZE);
- USBD_API->hw->WriteEP(hUsb, CDC_DATA_EP_IN, buffer, count); // write data to EP
- }
- return LPC_OK;
- }
- /**************************************************************************/
- /*!
- @brief Bulk Out handler for the USB ROM drivers (UART RX)
- */
- /**************************************************************************/
- ErrorCode_t CDC_BulkOut_Hdlr(USBD_HANDLE_T hUsb, void* data, uint32_t event)
- {
- if (USB_EVT_OUT == event)
- {
- uint16_t count, i;
- uint8_t buffer[CDC_DATA_EP_MAXPACKET_SIZE];
- count = USBD_API->hw->ReadEP(hUsb, CDC_DATA_EP_OUT, buffer);
- for (i=0; i<count; i++)
- {
- fifo_write(&ffRX, buffer[i]);
- }
- usb_cdc_recv_isr();
- }
- return LPC_OK;
- }
- /**************************************************************************/
- /*!
- @brief Initialises USB CDC using the ROM driver
- */
- /**************************************************************************/
- tusb_error_t tusb_cdc_init(USBD_HANDLE_T hUsb, USB_INTERFACE_DESCRIPTOR const *const pControlIntfDesc, USB_INTERFACE_DESCRIPTOR const *const pDataIntfDesc, uint32_t* mem_base, uint32_t* mem_size)
- {
- USBD_CDC_INIT_PARAM_T cdc_param =
- {
- .mem_base = *mem_base,
- .mem_size = *mem_size,
- .cif_intf_desc = (uint8_t*) pControlIntfDesc,
- .dif_intf_desc = (uint8_t*) pDataIntfDesc,
- .SetLineCode = CDC_SetLineCoding,
- .SendBreak = CDC_SendBreak,
- // .CIC_GetRequest = CDC_Control_GetRequest, // bug from romdrive cannot hook to this handler
- // Bug from ROM driver: can not hook bulk in & out handler here, must use USBD API register instead
- // .CDC_BulkIN_Hdlr = CDC_BulkIn_Hdlr,
- // .CDC_BulkOUT_Hdlr = CDC_BulkOut_Hdlr,
- };
- ASSERT (pControlIntfDesc && pDataIntfDesc, ERR_FAILED);
- /* register Bulk IN & OUT endpoint interrupt handler */
- ASSERT ( LPC_OK == USBD_API->core->RegisterEpHandler (hUsb , ((CDC_DATA_EP_IN & 0x0F) << 1) +1 , CDC_BulkIn_Hdlr , NULL), TUSB_ERROR_FAILED );
- ASSERT ( LPC_OK == USBD_API->core->RegisterEpHandler (hUsb , (CDC_DATA_EP_OUT & 0x0F) << 1 , CDC_BulkOut_Hdlr , NULL), TUSB_ERROR_FAILED );
- ASSERT ( LPC_OK == USBD_API->cdc->init(hUsb, &cdc_param, &g_hCdc), TUSB_ERROR_FAILED);
- /* update memory variables */
- *mem_base = cdc_param.mem_base;
- *mem_size = cdc_param.mem_size;
- return TUSB_ERROR_NONE;
- }
- /**************************************************************************/
- /*!
- @brief TODO Add description
- */
- /**************************************************************************/
- tusb_error_t tusb_cdc_configured(USBD_HANDLE_T hUsb)
- {
- uint8_t dummy=0;
- USBD_API->hw->WriteEP(hUsb, CDC_DATA_EP_IN, &dummy, 1); // initial packet for IN endpoint, will not work if omitted
- // FIXME abstract to hal
- #if MCU == MCU_LPC11UXX
- fifo_init (&ffTX, qBuffer[0], CDC_BUFFER_SIZE, false, USB_IRQn); // TX is non-overwritable
- fifo_init (&ffRX, qBuffer[1], CDC_BUFFER_SIZE, true, USB_IRQn); // RX is overwritable
- #elif MCU == MCU_LPC13UXX
- fifo_init (&ffTX, qBuffer[0], CDC_BUFFER_SIZE, false, USB_IRQ_IRQn); // TX is non-overwritable
- fifo_init (&ffRX, qBuffer[1], CDC_BUFFER_SIZE, true, USB_IRQ_IRQn); // RX is overwritable
- #elif MCU == MCU_LPC43XX
- fifo_init (&ffTX, qBuffer[0], CDC_BUFFER_SIZE, false, USB0_IRQn); // TODO USB1 TX is non-overwritable
- fifo_init (&ffRX, qBuffer[1], CDC_BUFFER_SIZE, true, USB0_IRQn); // RX is overwritable
- #else
- #error No MCU defined // TODO asbtract MCU
- #endif
- return TUSB_ERROR_NONE;
- }
- #endif
|