| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2013-06-16 15:39:09 +07:00
										 |  |  |   LPC_USB->USBCmdCode   = (phase << 8) | (code_data << 16); | 
					
						
							| 
									
										
										
										
											2013-06-08 02:50:10 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   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-16 15:39:09 +07:00
										 |  |  |   sie_commamd_code(SIE_CMDPHASE_COMMAND , cmd_code); | 
					
						
							|  |  |  |   sie_commamd_code(SIE_CMDPHASE_READ    , cmd_code); | 
					
						
							| 
									
										
										
										
											2013-06-08 02:50:10 +07:00
										 |  |  |   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-16 15:39:09 +07:00
										 |  |  |     sie_command_write(SIE_CMDCODE_BUFFER_CLEAR     , 0, 0); | 
					
						
							| 
									
										
										
										
											2013-06-08 02:50:10 +07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							| 
									
										
										
										
											2013-06-16 15:39:09 +07:00
										 |  |  | 	LPC_USB->USBEpIntEn      = (uint32_t) BIN8(11); // control endpoint cannot use DMA, non-control all use DMA
 | 
					
						
							| 
									
										
										
										
											2013-06-08 02:50:10 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-16 15:39:09 +07:00
										 |  |  | 	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
 | 
					
						
							| 
									
										
										
										
											2013-06-08 02:50:10 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-16 15:39:09 +07:00
										 |  |  | 	LPC_USB->USBEpIntClr     = 0xFFFFFFFF; // clear all pending interrupt
 | 
					
						
							|  |  |  | 	LPC_USB->USBEpIntPri     = 0;          // same priority for all endpoint
 | 
					
						
							| 
									
										
										
										
											2013-06-08 02:50:10 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// 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
 | 
					
						
							| 
									
										
										
										
											2013-06-16 15:39:09 +07:00
										 |  |  | 	sie_command_write(SIE_CMDCODE_BUFFER_VALIDATE  , 0, 0); | 
					
						
							| 
									
										
										
										
											2013-06-11 16:01:30 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   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 ); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-21 13:11:16 +07:00
										 |  |  | tusb_error_t dcd_pipe_open(uint8_t coreid, tusb_descriptor_endpoint_t const * p_endpoint_desc) | 
					
						
							| 
									
										
										
										
											2013-06-16 14:41:48 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   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
 |