2013-05-25 16:03:40 +07:00
|
|
|
/**************************************************************************/
|
|
|
|
/*!
|
2013-05-28 15:24:27 +07:00
|
|
|
@file dcd_lpc175x_6x.c
|
2013-05-25 16:03:40 +07:00
|
|
|
@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 "tusb_option.h"
|
|
|
|
|
2013-05-28 15:24:27 +07:00
|
|
|
#if MODE_DEVICE_SUPPORTED && (MCU == MCU_LPC175X_6X)
|
2013-05-25 16:03:40 +07:00
|
|
|
|
|
|
|
#define _TINY_USB_SOURCE_FILE_
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// INCLUDE
|
|
|
|
//--------------------------------------------------------------------+
|
2013-05-28 15:24:27 +07:00
|
|
|
#include "dcd.h"
|
|
|
|
#include "dcd_lpc175x_6x.h"
|
2013-06-08 02:50:10 +07:00
|
|
|
#include "usbd_dcd.h"
|
2013-05-25 16:03:40 +07:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// MACRO CONSTANT TYPEDEF
|
|
|
|
//--------------------------------------------------------------------+
|
2013-06-05 11:23:41 +07:00
|
|
|
STATIC_ dcd_dma_descriptor_t* dcd_udca[32] ATTR_ALIGNED(128) TUSB_CFG_ATTR_USBRAM;
|
2013-06-12 21:06:43 +07:00
|
|
|
STATIC_ dcd_dma_descriptor_t dcd_dd[DCD_MAX_DD];
|
2013-05-25 16:03:40 +07:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// INTERNAL OBJECT & FUNCTION DECLARATION
|
|
|
|
//--------------------------------------------------------------------+
|
2013-06-08 02:50:10 +07:00
|
|
|
static inline void endpoint_set_max_packet_size(uint8_t endpoint_idx, uint16_t max_packet_size) ATTR_ALWAYS_INLINE;
|
|
|
|
static inline void endpoint_set_max_packet_size(uint8_t endpoint_idx, uint16_t max_packet_size)
|
|
|
|
{
|
2013-06-16 15:32:08 +07:00
|
|
|
LPC_USB->USBReEp |= BIT_(endpoint_idx);
|
|
|
|
|
2013-06-08 02:50:10 +07:00
|
|
|
LPC_USB->USBEpInd = endpoint_idx; // select index before setting packet size
|
|
|
|
LPC_USB->USBMaxPSize = max_packet_size;
|
2013-06-16 15:32:08 +07:00
|
|
|
|
|
|
|
#ifndef _TEST_
|
|
|
|
if( endpoint_idx > 2) // endpoint control is always realized
|
|
|
|
{
|
|
|
|
while ((LPC_USB->USBDevIntSt & DEV_INT_ENDPOINT_REALIZED_MASK) == 0) {} // TODO can be omitted, or move to set max packet size
|
|
|
|
LPC_USB->USBDevIntClr = DEV_INT_ENDPOINT_REALIZED_MASK;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-06-08 02:50:10 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void sie_commamd_code (uint8_t phase, uint8_t code_data) ATTR_ALWAYS_INLINE;
|
|
|
|
static inline void sie_commamd_code (uint8_t phase, uint8_t code_data)
|
|
|
|
{
|
|
|
|
LPC_USB->USBDevIntClr = (DEV_INT_COMMAND_CODE_EMPTY_MASK | DEV_INT_COMMAND_DATA_FULL_MASK);
|
|
|
|
LPC_USB->USBCmdCode = (phase << 8) | (code_data << 16);
|
|
|
|
|
|
|
|
uint32_t const wait_flag = (phase == SIE_CMDPHASE_READ) ? DEV_INT_COMMAND_DATA_FULL_MASK : DEV_INT_COMMAND_CODE_EMPTY_MASK;
|
2013-06-12 21:06:43 +07:00
|
|
|
#ifndef _TEST_
|
2013-06-08 02:50:10 +07:00
|
|
|
while ((LPC_USB->USBDevIntSt & wait_flag) == 0); // TODO blocking forever potential
|
2013-06-12 21:06:43 +07:00
|
|
|
#endif
|
2013-06-08 02:50:10 +07:00
|
|
|
LPC_USB->USBDevIntClr = wait_flag;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void sie_command_write (uint8_t cmd_code, uint8_t data_len, uint8_t data) ATTR_ALWAYS_INLINE;
|
|
|
|
static inline void sie_command_write (uint8_t cmd_code, uint8_t data_len, uint8_t data)
|
|
|
|
{
|
|
|
|
sie_commamd_code(SIE_CMDPHASE_COMMAND, cmd_code);
|
|
|
|
|
|
|
|
if (data_len)
|
|
|
|
{
|
|
|
|
sie_commamd_code(SIE_CMDPHASE_WRITE, data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint32_t sie_command_read (uint8_t cmd_code, uint8_t data_len) ATTR_ALWAYS_INLINE;
|
|
|
|
static inline uint32_t sie_command_read (uint8_t cmd_code, uint8_t data_len)
|
|
|
|
{
|
|
|
|
// TODO multiple read
|
2013-06-11 16:01:30 +07:00
|
|
|
sie_commamd_code(SIE_CMDPHASE_COMMAND, cmd_code);
|
2013-06-08 02:50:10 +07:00
|
|
|
sie_commamd_code(SIE_CMDPHASE_READ, cmd_code);
|
|
|
|
return LPC_USB->USBCmdData;
|
|
|
|
}
|
2013-05-25 16:03:40 +07:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// IMPLEMENTATION
|
|
|
|
//--------------------------------------------------------------------+
|
2013-06-08 02:50:10 +07:00
|
|
|
void endpoint_control_isr(uint8_t coreid)
|
|
|
|
{
|
|
|
|
(void) coreid; // suppress compiler warning
|
|
|
|
uint32_t const endpoint_int_status = LPC_USB->USBEpIntSt & LPC_USB->USBEpIntEn;
|
|
|
|
|
2013-06-16 14:41:48 +07:00
|
|
|
//------------- control OUT -------------//
|
2013-06-08 02:50:10 +07:00
|
|
|
if (endpoint_int_status & BIT_(0))
|
|
|
|
{
|
|
|
|
uint32_t const endpoint_status = sie_command_read(SIE_CMDCODE_ENDPOINT_SELECT+0, 1);
|
|
|
|
if (endpoint_status & SIE_ENDPOINT_STATUS_SETUP_RECEIVED_MASK)
|
|
|
|
{
|
|
|
|
(void) sie_command_read(SIE_CMDCODE_ENDPOINT_SELECT_CLEAR_INTERRUPT+0, 1); // clear setup bit
|
|
|
|
|
2013-06-14 19:10:11 +07:00
|
|
|
dcd_pipe_control_read(0, &usbd_devices[0].setup_packet, 8);
|
2013-06-08 02:50:10 +07:00
|
|
|
usbd_isr(0, TUSB_EVENT_SETUP_RECEIVED);
|
|
|
|
}else
|
|
|
|
{
|
2013-06-16 14:41:48 +07:00
|
|
|
// Current not support any out control with data yet
|
|
|
|
// dcd_pipe_control_read(0,..
|
2013-06-08 02:50:10 +07:00
|
|
|
}
|
2013-06-11 16:01:30 +07:00
|
|
|
sie_command_write(SIE_CMDCODE_ENDPOINT_SELECT+0, 0, 0);
|
2013-06-08 02:50:10 +07:00
|
|
|
sie_command_write(SIE_CMDCODE_BUFFER_CLEAR, 0, 0);
|
|
|
|
}
|
|
|
|
|
2013-06-16 14:41:48 +07:00
|
|
|
//------------- control IN -------------//
|
2013-06-08 02:50:10 +07:00
|
|
|
if (endpoint_int_status & BIT_(1))
|
|
|
|
{
|
2013-06-11 16:01:30 +07:00
|
|
|
(void) endpoint_int_status;
|
2013-06-08 02:50:10 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
LPC_USB->USBEpIntClr = endpoint_int_status; // acknowledge interrupt
|
|
|
|
}
|
|
|
|
|
2013-06-04 13:36:18 +07:00
|
|
|
void dcd_isr(uint8_t coreid)
|
|
|
|
{
|
2013-06-11 16:01:30 +07:00
|
|
|
uint32_t const device_int_status = LPC_USB->USBDevIntSt & LPC_USB->USBDevIntEn & DEV_INT_ALL_MASK;
|
2013-06-08 02:50:10 +07:00
|
|
|
LPC_USB->USBDevIntClr = device_int_status;// Acknowledge handled interrupt
|
|
|
|
|
|
|
|
//------------- usb bus event -------------//
|
|
|
|
if (device_int_status & DEV_INT_DEVICE_STATUS_MASK)
|
|
|
|
{
|
2013-06-11 16:01:30 +07:00
|
|
|
uint32_t const dev_status_reg = sie_command_read(SIE_CMDCODE_DEVICE_STATUS, 1);
|
2013-06-08 02:50:10 +07:00
|
|
|
if (dev_status_reg & SIE_DEV_STATUS_RESET_MASK)
|
|
|
|
{
|
|
|
|
usbd_isr(coreid, TUSB_EVENT_BUS_RESET);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO invoke some callbacks
|
2013-06-16 14:41:48 +07:00
|
|
|
if (dev_status_reg & SIE_DEV_STATUS_CONNECT_CHANGE_MASK) { }
|
|
|
|
if (dev_status_reg & SIE_DEV_STATUS_SUSPEND_CHANGE_MASK) {
|
2013-06-08 02:50:10 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//------------- slave mode, control endpoint -------------//
|
|
|
|
if (device_int_status & DEV_INT_ENDPOINT_SLOW_MASK)
|
|
|
|
{
|
|
|
|
// only occur on control endpoint, all other use DMA
|
|
|
|
endpoint_control_isr(coreid);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (device_int_status & DEV_INT_ERROR_MASK)
|
|
|
|
{
|
2013-06-11 16:01:30 +07:00
|
|
|
uint32_t error_status = sie_command_read(SIE_CMDCODE_READ_ERROR_STATUS, 1);
|
2013-06-11 16:30:06 +07:00
|
|
|
(void) error_status;
|
2013-06-11 16:01:30 +07:00
|
|
|
// ASSERT(false, (void) 0);
|
2013-06-08 02:50:10 +07:00
|
|
|
}
|
2013-05-25 16:03:40 +07:00
|
|
|
|
2013-06-04 13:36:18 +07:00
|
|
|
}
|
|
|
|
|
2013-06-08 02:50:10 +07:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// USBD-DCD API
|
|
|
|
//--------------------------------------------------------------------+
|
2013-06-04 13:36:18 +07:00
|
|
|
tusb_error_t dcd_init(void)
|
2013-06-08 02:50:10 +07:00
|
|
|
{
|
|
|
|
//------------- user manual 11.13 usb device controller initialization -------------// LPC_USB->USBEpInd = 0;
|
|
|
|
// step 6 : set up control endpoint
|
|
|
|
endpoint_set_max_packet_size(0, TUSB_CFG_DEVICE_CONTROL_PACKET_SIZE);
|
|
|
|
endpoint_set_max_packet_size(1, TUSB_CFG_DEVICE_CONTROL_PACKET_SIZE);
|
2013-06-12 21:06:43 +07:00
|
|
|
|
2013-06-08 02:50:10 +07:00
|
|
|
// step 7 : slave mode set up
|
|
|
|
LPC_USB->USBEpIntEn = (uint32_t) BIN8(11); // control endpoint cannot use DMA, non-control all use DMA
|
|
|
|
|
|
|
|
LPC_USB->USBDevIntEn = (DEV_INT_DEVICE_STATUS_MASK | DEV_INT_ENDPOINT_SLOW_MASK | DEV_INT_ERROR_MASK);
|
|
|
|
LPC_USB->USBDevIntClr = 0xFFFFFFFF; // clear all pending interrupt
|
|
|
|
|
|
|
|
LPC_USB->USBEpIntClr = 0xFFFFFFFF; // clear all pending interrupt
|
|
|
|
LPC_USB->USBEpIntPri = 0; // same priority for all endpoint
|
|
|
|
|
|
|
|
// step 8 : DMA set up
|
|
|
|
LPC_USB->USBEpDMADis = 0xFFFFFFFF; // firstly disable all dma
|
|
|
|
LPC_USB->USBDMARClr = 0xFFFFFFFF; // clear all pending interrupt
|
|
|
|
LPC_USB->USBEoTIntClr = 0xFFFFFFFF;
|
|
|
|
LPC_USB->USBNDDRIntClr = 0xFFFFFFFF;
|
|
|
|
LPC_USB->USBSysErrIntClr = 0xFFFFFFFF;
|
|
|
|
|
|
|
|
for (uint8_t index = 0; index < DCD_MAX_DD; index++)
|
|
|
|
{
|
2013-06-12 21:06:43 +07:00
|
|
|
dcd_udca[index] = dcd_dd + index;
|
2013-06-08 02:50:10 +07:00
|
|
|
}
|
|
|
|
LPC_USB->USBUDCAH = (uint32_t) dcd_udca;
|
|
|
|
LPC_USB->USBDMAIntEn = (DMA_INT_END_OF_XFER_MASK | DMA_INT_NEW_DD_REQUEST_MASK | DMA_INT_ERROR_MASK );
|
|
|
|
|
|
|
|
// clear all stall on control endpoint IN & OUT if any
|
2013-06-11 16:01:30 +07:00
|
|
|
sie_command_write(SIE_CMDCODE_ENDPOINT_SET_STATUS , 1, 0);
|
|
|
|
sie_command_write(SIE_CMDCODE_ENDPOINT_SET_STATUS + 1, 1, 0);
|
2013-06-08 02:50:10 +07:00
|
|
|
|
|
|
|
return TUSB_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
2013-06-04 13:36:18 +07:00
|
|
|
void dcd_controller_connect(uint8_t coreid)
|
|
|
|
{
|
2013-06-08 02:50:10 +07:00
|
|
|
sie_command_write(SIE_CMDCODE_DEVICE_STATUS, 1, 1);
|
|
|
|
}
|
2013-06-04 13:36:18 +07:00
|
|
|
|
2013-06-08 02:50:10 +07:00
|
|
|
void dcd_device_set_address(uint8_t coreid, uint8_t dev_addr)
|
|
|
|
{
|
|
|
|
sie_command_write(SIE_CMDCODE_SET_ADDRESS, 1, 0x80 | dev_addr); // 7th bit is : device_enable
|
2013-06-04 13:36:18 +07:00
|
|
|
}
|
2013-06-08 02:50:10 +07:00
|
|
|
|
2013-06-11 17:53:33 +07:00
|
|
|
void dcd_device_set_configuration(uint8_t coreid, uint8_t config_num)
|
|
|
|
{
|
|
|
|
(void) config_num; // supress compiler's warnings
|
|
|
|
sie_command_write(SIE_CMDCODE_CONFIGURE_DEVICE, 1, 1);
|
|
|
|
}
|
|
|
|
|
2013-06-16 14:41:48 +07:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// PIPE API
|
|
|
|
//--------------------------------------------------------------------+
|
2013-06-14 19:06:33 +07:00
|
|
|
static inline uint16_t length_unit_byte2dword(uint16_t length_in_bytes) ATTR_ALWAYS_INLINE ATTR_CONST;
|
|
|
|
static inline uint16_t length_unit_byte2dword(uint16_t length_in_bytes)
|
|
|
|
{
|
|
|
|
return (length_in_bytes + 3) / 4; // length_in_dword
|
|
|
|
}
|
|
|
|
|
2013-06-11 16:01:30 +07:00
|
|
|
tusb_error_t dcd_pipe_control_write(uint8_t coreid, void const * buffer, uint16_t length)
|
|
|
|
{
|
2013-06-11 16:30:06 +07:00
|
|
|
(void) coreid; // suppress compiler warning
|
|
|
|
|
|
|
|
ASSERT( length !=0 || buffer == NULL, TUSB_ERROR_INVALID_PARA);
|
|
|
|
|
2013-06-11 16:01:30 +07:00
|
|
|
LPC_USB->USBCtrl = SLAVE_CONTROL_WRITE_ENABLE_MASK; // logical endpoint = 0
|
|
|
|
LPC_USB->USBTxPLen = length;
|
|
|
|
|
2013-06-14 19:06:33 +07:00
|
|
|
for (uint16_t count = 0; count < length_unit_byte2dword(length); count++)
|
2013-06-11 16:01:30 +07:00
|
|
|
{
|
|
|
|
LPC_USB->USBTxData = *((uint32_t *)buffer); // NOTE: cortex M3 have no problem with alignment
|
|
|
|
buffer += 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
LPC_USB->USBCtrl = 0;
|
|
|
|
|
|
|
|
sie_command_write(SIE_CMDCODE_ENDPOINT_SELECT+1, 0, 0); // select control IN endpoint
|
|
|
|
sie_command_write(SIE_CMDCODE_BUFFER_VALIDATE, 0, 0);
|
|
|
|
|
|
|
|
return TUSB_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
2013-06-14 19:06:33 +07:00
|
|
|
tusb_error_t dcd_pipe_control_read(uint8_t coreid, void * buffer, uint16_t length)
|
2013-06-11 16:01:30 +07:00
|
|
|
{
|
2013-06-14 19:06:33 +07:00
|
|
|
LPC_USB->USBCtrl = SLAVE_CONTROL_READ_ENABLE_MASK; // logical endpoint = 0
|
|
|
|
while ((LPC_USB->USBRxPLen & SLAVE_RXPLEN_PACKET_READY_MASK) == 0) {}
|
|
|
|
|
|
|
|
uint16_t actual_length = min16_of(length, (uint16_t) (LPC_USB->USBRxPLen & SLAVE_RXPLEN_PACKET_LENGTH_MASK) );
|
|
|
|
uint32_t *p_read_data = (uint32_t*) buffer;
|
|
|
|
for( uint16_t count=0; count < length_unit_byte2dword(actual_length); count++)
|
|
|
|
{
|
|
|
|
*p_read_data = LPC_USB->USBRxData;
|
|
|
|
p_read_data++; // increase by 4 ( sizeof(uint32_t) )
|
|
|
|
}
|
|
|
|
LPC_USB->USBCtrl = 0;
|
|
|
|
|
2013-06-11 16:01:30 +07:00
|
|
|
return TUSB_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
2013-06-12 21:06:43 +07:00
|
|
|
// TODO inline function
|
2013-06-08 02:50:10 +07:00
|
|
|
void dcd_pipe_control_write_zero_length(uint8_t coreid)
|
|
|
|
{
|
2013-06-11 16:30:06 +07:00
|
|
|
dcd_pipe_control_write(coreid, NULL, 0);
|
2013-06-08 02:50:10 +07:00
|
|
|
}
|
|
|
|
|
2013-06-16 14:41:48 +07:00
|
|
|
static inline uint8_t endpoint_address_to_physical_index(uint8_t ep_address) ATTR_ALWAYS_INLINE ATTR_CONST;
|
|
|
|
static inline uint8_t endpoint_address_to_physical_index(uint8_t ep_address)
|
|
|
|
{
|
|
|
|
return (ep_address << 1) + (ep_address & 0x80 ? 1 : 0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
tusb_error_t dcd_endpoint_configure(uint8_t coreid, tusb_descriptor_endpoint_t const * p_endpoint_desc)
|
|
|
|
{
|
|
|
|
uint8_t phy_ep = endpoint_address_to_physical_index( p_endpoint_desc->bEndpointAddress );
|
|
|
|
|
|
|
|
//------------- Realize Endpoint with Max Packet Size -------------//
|
|
|
|
endpoint_set_max_packet_size(phy_ep, p_endpoint_desc->wMaxPacketSize.size);
|
|
|
|
|
|
|
|
//------------- DMA set up -------------//
|
|
|
|
memclr_(dcd_dd + phy_ep, sizeof(dcd_dma_descriptor_t));
|
|
|
|
dcd_dd[phy_ep].is_isochronous = (p_endpoint_desc->bmAttributes.xfer == TUSB_XFER_ISOCHRONOUS) ? 1 : 0;
|
|
|
|
dcd_dd[phy_ep].max_packet_size = p_endpoint_desc->wMaxPacketSize.size;
|
|
|
|
dcd_dd[phy_ep].is_retired = 1; // dd is not active at first
|
|
|
|
|
|
|
|
LPC_USB->USBEpDMAEn = BIT_(phy_ep);
|
|
|
|
|
|
|
|
sie_command_write(SIE_CMDCODE_ENDPOINT_SET_STATUS+phy_ep, 1, 0); // clear all endpoint status
|
|
|
|
|
|
|
|
return TUSB_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
//tusb_error_t dcd_pipe_xfer()
|
|
|
|
|
2013-05-25 16:03:40 +07:00
|
|
|
#endif
|