| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /*!
 | 
					
						
							|  |  |  |     @file     dcd_lpc175x_6x.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 "tusb_option.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if MODE_DEVICE_SUPPORTED && (TUSB_CFG_MCU == MCU_LPC175X_6X)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define _TINY_USB_SOURCE_FILE_
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // INCLUDE
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | #include "dcd.h"
 | 
					
						
							|  |  |  | #include "dcd_lpc175x_6x.h"
 | 
					
						
							|  |  |  | #include "usbd_dcd.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // MACRO CONSTANT TYPEDEF
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | #define DCD_QHD_MAX 32
 | 
					
						
							|  |  |  | #define DCD_QTD_MAX  32 // TODO scale with configure
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							| 
									
										
										
										
											2014-03-12 15:11:48 +07:00
										 |  |  |   volatile dcd_dma_descriptor_t* udca[DCD_QHD_MAX]; // must be 128 byte aligned
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |   dcd_dma_descriptor_t dd[DCD_QTD_MAX][2]; // each endpoints can have up to 2 DD queued at a time TODO 0-1 are not used, offset to reduce memory
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   uint8_t class_code[DCD_QHD_MAX]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   struct { | 
					
						
							|  |  |  |     uint8_t* p_data; | 
					
						
							|  |  |  |     uint16_t remaining_bytes; | 
					
						
							|  |  |  |     uint8_t  int_on_complete; | 
					
						
							|  |  |  |   }control_dma; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }dcd_data_t; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-17 18:02:51 +07:00
										 |  |  | TUSB_CFG_ATTR_USBRAM ATTR_ALIGNED(128) STATIC_VAR dcd_data_t dcd_data; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // INTERNAL OBJECT & FUNCTION DECLARATION
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | static void bus_reset(void); | 
					
						
							|  |  |  | static tusb_error_t pipe_control_read(void * buffer, uint16_t length); | 
					
						
							|  |  |  | static tusb_error_t pipe_control_write(void const * buffer, uint16_t length); | 
					
						
							|  |  |  | static tusb_error_t pipe_control_xfer(uint8_t ep_id, uint8_t* p_buffer, uint16_t length); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // PIPE HELPER
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | static inline uint8_t edpt_addr2phy(uint8_t endpoint_addr) ATTR_CONST ATTR_ALWAYS_INLINE; | 
					
						
							|  |  |  | static inline uint8_t edpt_addr2phy(uint8_t endpoint_addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return 2*(endpoint_addr & 0x0F) + ((endpoint_addr & TUSB_DIR_DEV_TO_HOST_MASK) ? 1 : 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void edpt_set_max_packet_size(uint8_t ep_id, uint16_t max_packet_size) ATTR_ALWAYS_INLINE; | 
					
						
							|  |  |  | static inline void edpt_set_max_packet_size(uint8_t ep_id, uint16_t max_packet_size) | 
					
						
							|  |  |  | { // follows example in 11.10.4.2
 | 
					
						
							|  |  |  |   LPC_USB->USBReEp    |= BIT_(ep_id); | 
					
						
							|  |  |  |   LPC_USB->USBEpInd    = ep_id; // select index before setting packet size
 | 
					
						
							|  |  |  |   LPC_USB->USBMaxPSize = max_packet_size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef _TEST_
 | 
					
						
							|  |  |  |   while ((LPC_USB->USBDevIntSt & DEV_INT_ENDPOINT_REALIZED_MASK) == 0) {} // TODO can be omitted
 | 
					
						
							|  |  |  |   LPC_USB->USBDevIntClr = DEV_INT_ENDPOINT_REALIZED_MASK; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // USBD-DCD API
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | static void bus_reset(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   // step 7 : slave mode set up
 | 
					
						
							|  |  |  |   LPC_USB->USBEpIntClr     = 0xFFFFFFFF;          // clear all pending interrupt
 | 
					
						
							|  |  |  | 	LPC_USB->USBDevIntClr    = 0xFFFFFFFF;          // clear all pending interrupt
 | 
					
						
							|  |  |  | 	LPC_USB->USBEpIntEn      = (uint32_t) BIN8(11); // control endpoint cannot use DMA, non-control all use DMA
 | 
					
						
							|  |  |  | 	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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	memclr_(&dcd_data, sizeof(dcd_data_t)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-11 13:13:04 +07:00
										 |  |  | bool tusb_dcd_init(uint8_t port) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-09 14:29:23 +07:00
										 |  |  |   (void) port; | 
					
						
							| 
									
										
										
										
											2018-03-02 22:46:36 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |   //------------- user manual 11.13 usb device controller initialization -------------//  LPC_USB->USBEpInd = 0;
 | 
					
						
							|  |  |  |   // step 6 : set up control endpoint
 | 
					
						
							|  |  |  |   edpt_set_max_packet_size(0, TUSB_CFG_DEVICE_CONTROL_ENDOINT_SIZE); | 
					
						
							|  |  |  |   edpt_set_max_packet_size(1, TUSB_CFG_DEVICE_CONTROL_ENDOINT_SIZE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bus_reset(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   LPC_USB->USBDevIntEn = (DEV_INT_DEVICE_STATUS_MASK | DEV_INT_ENDPOINT_SLOW_MASK | DEV_INT_ERROR_MASK); | 
					
						
							|  |  |  | 	LPC_USB->USBUDCAH    = (uint32_t) dcd_data.udca; | 
					
						
							|  |  |  | 	LPC_USB->USBDMAIntEn = (DMA_INT_END_OF_XFER_MASK | DMA_INT_ERROR_MASK ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sie_write(SIE_CMDCODE_DEVICE_STATUS, 1, 1); // connect
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return TUSB_ERROR_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void endpoint_non_control_isr(uint32_t eot_int) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   for(uint8_t ep_id = 2; ep_id < DCD_QHD_MAX; ep_id++ ) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if ( BIT_TEST_(eot_int, ep_id) ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       dcd_dma_descriptor_t* const p_first_dd = &dcd_data.dd[ep_id][0]; | 
					
						
							|  |  |  |       dcd_dma_descriptor_t* const p_last_dd  = dcd_data.dd[ep_id] + (p_first_dd->is_next_valid ? 1 : 0); // Maximum is 2 QTD are queued in an endpoint
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // only handle when Controller already finished the last DD
 | 
					
						
							|  |  |  |       if ( dcd_data.udca[ep_id] == p_last_dd ) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         dcd_data.udca[ep_id] = p_first_dd; // UDCA currently points to the last DD, change to the fixed DD
 | 
					
						
							|  |  |  |         p_first_dd->buffer_length = 0; // buffer length is used to determined if first dd is queued in pipe xfer function
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ( p_last_dd->int_on_complete ) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2018-03-11 12:37:07 +07:00
										 |  |  |           edpt_hdl_t edpt_hdl = | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |           { | 
					
						
							| 
									
										
										
										
											2018-03-09 14:29:23 +07:00
										 |  |  |               .port     = 0, | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |               .index      = ep_id, | 
					
						
							|  |  |  |               .class_code = dcd_data.class_code[ep_id] | 
					
						
							|  |  |  |           }; | 
					
						
							| 
									
										
										
										
											2018-03-11 13:59:37 +07:00
										 |  |  |           bool succeeded = (p_last_dd->status == DD_STATUS_NORMAL || p_last_dd->status == DD_STATUS_DATA_UNDERUN) ? true : false; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-11 13:59:37 +07:00
										 |  |  |           tusb_dcd_xfer_complete(edpt_hdl, p_last_dd->present_count, succeeded); // report only xferred bytes in the IOC qtd
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void endpoint_control_isr(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-03-23 17:17:26 +07:00
										 |  |  |   uint32_t const interrupt_enable = LPC_USB->USBEpIntEn; | 
					
						
							|  |  |  |   uint32_t const endpoint_int_status = LPC_USB->USBEpIntSt & interrupt_enable; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | //  LPC_USB->USBEpIntClr = endpoint_int_status; // acknowledge interrupt TODO cannot immediately acknowledge setup packet
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //------------- Setup Recieved-------------//
 | 
					
						
							|  |  |  |   if ( (endpoint_int_status & BIT_(0)) && | 
					
						
							|  |  |  |        (sie_read(SIE_CMDCODE_ENDPOINT_SELECT+0, 1) & SIE_SELECT_ENDPOINT_SETUP_RECEIVED_MASK) ) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     (void) sie_read(SIE_CMDCODE_ENDPOINT_SELECT_CLEAR_INTERRUPT+0, 1); // clear setup bit
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     tusb_control_request_t control_request; | 
					
						
							|  |  |  |     pipe_control_read(&control_request, 8); // TODO read before clear setup above
 | 
					
						
							| 
									
										
										
										
											2018-03-11 13:13:04 +07:00
										 |  |  |     tusb_dcd_setup_received(0, (uint8_t*) &control_request); | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |   } | 
					
						
							|  |  |  |   else if (endpoint_int_status & 0x03) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     uint8_t const ep_id = ( endpoint_int_status & BIT_(0) ) ? 0 : 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ( dcd_data.control_dma.remaining_bytes > 0 ) | 
					
						
							|  |  |  |     { // there are still data to transfer
 | 
					
						
							|  |  |  |       pipe_control_xfer(ep_id, dcd_data.control_dma.p_data, dcd_data.control_dma.remaining_bytes); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       dcd_data.control_dma.remaining_bytes = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if ( BIT_TEST_(dcd_data.control_dma.int_on_complete, ep_id) ) | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2018-03-11 12:37:07 +07:00
										 |  |  |         edpt_hdl_t edpt_hdl = { .port = 0, .class_code = 0 }; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |         dcd_data.control_dma.int_on_complete = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // FIXME xferred_byte for control xfer is not needed now !!!
 | 
					
						
							| 
									
										
										
										
											2018-03-11 13:59:37 +07:00
										 |  |  |         tusb_dcd_xfer_complete(edpt_hdl, 0, true); | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   LPC_USB->USBEpIntClr = endpoint_int_status; // acknowledge interrupt TODO cannot immediately acknowledge setup packet
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-09 14:29:23 +07:00
										 |  |  | void hal_dcd_isr(uint8_t port) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-09 14:29:23 +07:00
										 |  |  |   (void) port; | 
					
						
							| 
									
										
										
										
											2014-03-23 17:17:26 +07:00
										 |  |  |   uint32_t const device_int_enable = LPC_USB->USBDevIntEn; | 
					
						
							|  |  |  |   uint32_t const device_int_status = LPC_USB->USBDevIntSt & device_int_enable; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |   LPC_USB->USBDevIntClr = device_int_status;// Acknowledge handled interrupt
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //------------- usb bus event -------------//
 | 
					
						
							|  |  |  |   if (device_int_status & DEV_INT_DEVICE_STATUS_MASK) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     uint8_t const dev_status_reg = sie_read(SIE_CMDCODE_DEVICE_STATUS, 1); | 
					
						
							|  |  |  |     if (dev_status_reg & SIE_DEV_STATUS_RESET_MASK) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       bus_reset(); | 
					
						
							| 
									
										
										
										
											2018-03-11 13:13:04 +07:00
										 |  |  |       tusb_dcd_bus_event(0, USBD_BUS_EVENT_RESET); | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (dev_status_reg & SIE_DEV_STATUS_CONNECT_CHANGE_MASK) | 
					
						
							|  |  |  |     { // device is disconnected, require using VBUS (P1_30)
 | 
					
						
							| 
									
										
										
										
											2018-03-11 13:13:04 +07:00
										 |  |  |       tusb_dcd_bus_event(0, USBD_BUS_EVENT_UNPLUGGED); | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (dev_status_reg & SIE_DEV_STATUS_SUSPEND_CHANGE_MASK) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (dev_status_reg & SIE_DEV_STATUS_SUSPEND_MASK) | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2018-03-11 13:13:04 +07:00
										 |  |  |         tusb_dcd_bus_event(0, USBD_BUS_EVENT_SUSPENDED); | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |       } | 
					
						
							|  |  |  | //      else
 | 
					
						
							|  |  |  | //      {
 | 
					
						
							| 
									
										
										
										
											2018-03-11 13:13:04 +07:00
										 |  |  | //        tusb_dcd_bus_event(0, USBD_BUS_EVENT_RESUME);
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | //      }
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //------------- Control Endpoint (Slave Mode) -------------//
 | 
					
						
							|  |  |  |   if (device_int_status & DEV_INT_ENDPOINT_SLOW_MASK) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     endpoint_control_isr(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //------------- Non-Control Endpoint (DMA Mode) -------------//
 | 
					
						
							| 
									
										
										
										
											2014-03-23 17:17:26 +07:00
										 |  |  |   uint32_t const dma_int_enable = LPC_USB->USBDMAIntEn; | 
					
						
							|  |  |  |   uint32_t const dma_int_status = LPC_USB->USBDMAIntSt & dma_int_enable; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (dma_int_status & DMA_INT_END_OF_XFER_MASK) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     uint32_t eot_int = LPC_USB->USBEoTIntSt; | 
					
						
							|  |  |  |     LPC_USB->USBEoTIntClr = eot_int; // acknowledge interrupt source
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     endpoint_non_control_isr(eot_int); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (device_int_status & DEV_INT_ERROR_MASK || dma_int_status & DMA_INT_ERROR_MASK) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     uint32_t error_status = sie_read(SIE_CMDCODE_READ_ERROR_STATUS, 1); | 
					
						
							|  |  |  |     (void) error_status; | 
					
						
							|  |  |  | //    ASSERT(false, (void) 0);
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // USBD API - CONTROLLER
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-03-11 13:13:04 +07:00
										 |  |  | void tusb_dcd_connect(uint8_t port) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-09 14:29:23 +07:00
										 |  |  |   (void) port; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |   sie_write(SIE_CMDCODE_DEVICE_STATUS, 1, 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-11 13:13:04 +07:00
										 |  |  | void tusb_dcd_set_address(uint8_t port, uint8_t dev_addr) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-09 14:29:23 +07:00
										 |  |  |   (void) port; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |   sie_write(SIE_CMDCODE_SET_ADDRESS, 1, 0x80 | dev_addr); // 7th bit is : device_enable
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-11 13:13:04 +07:00
										 |  |  | void tusb_dcd_set_config(uint8_t port, uint8_t config_num) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-09 14:29:23 +07:00
										 |  |  |   (void) port; | 
					
						
							| 
									
										
										
										
											2018-03-02 22:53:26 +07:00
										 |  |  |   (void) config_num; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |   sie_write(SIE_CMDCODE_CONFIGURE_DEVICE, 1, 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // PIPE CONTROL HELPER
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | static inline uint16_t length_byte2dword(uint16_t length_in_bytes) ATTR_ALWAYS_INLINE ATTR_CONST; | 
					
						
							|  |  |  | static inline uint16_t length_byte2dword(uint16_t length_in_bytes) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (length_in_bytes + 3) / 4; // length_in_dword
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static tusb_error_t pipe_control_xfer(uint8_t ep_id, uint8_t* p_buffer, uint16_t length) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   uint16_t const packet_len = min16_of(length, TUSB_CFG_DEVICE_CONTROL_ENDOINT_SIZE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (ep_id) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     ASSERT_STATUS ( pipe_control_write(p_buffer, packet_len) ); | 
					
						
							|  |  |  |   }else | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     ASSERT_STATUS ( pipe_control_read(p_buffer, packet_len) ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dcd_data.control_dma.remaining_bytes -= packet_len; | 
					
						
							|  |  |  |   dcd_data.control_dma.p_data          += packet_len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return TUSB_ERROR_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static tusb_error_t pipe_control_write(void const * buffer, uint16_t length) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   uint32_t const * p_write_data = (uint32_t const *) buffer; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   LPC_USB->USBCtrl   = USBCTRL_WRITE_ENABLE_MASK; // logical endpoint = 0
 | 
					
						
							|  |  |  | 	LPC_USB->USBTxPLen = length; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (uint16_t count = 0; count < length_byte2dword(length); count++) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		LPC_USB->USBTxData = *p_write_data; // NOTE: cortex M3 have no problem with alignment
 | 
					
						
							|  |  |  | 		p_write_data++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	LPC_USB->USBCtrl   = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// select control IN & validate the endpoint
 | 
					
						
							|  |  |  | 	sie_write(SIE_CMDCODE_ENDPOINT_SELECT+1, 0, 0); | 
					
						
							|  |  |  | 	sie_write(SIE_CMDCODE_BUFFER_VALIDATE  , 0, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return TUSB_ERROR_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static tusb_error_t pipe_control_read(void * buffer, uint16_t length) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   LPC_USB->USBCtrl = USBCTRL_READ_ENABLE_MASK; // logical endpoint = 0
 | 
					
						
							|  |  |  |   while ((LPC_USB->USBRxPLen & USBRXPLEN_PACKET_READY_MASK) == 0) {} // TODO blocking, should have timeout
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   uint16_t actual_length = min16_of(length, (uint16_t) (LPC_USB->USBRxPLen & USBRXPLEN_PACKET_LENGTH_MASK) ); | 
					
						
							|  |  |  |   uint32_t *p_read_data = (uint32_t*) buffer; | 
					
						
							|  |  |  |   for( uint16_t count=0; count < length_byte2dword(actual_length); count++) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     *p_read_data = LPC_USB->USBRxData; | 
					
						
							|  |  |  |     p_read_data++; // increase by 4 ( sizeof(uint32_t) )
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   LPC_USB->USBCtrl = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // select control OUT & clear the endpoint
 | 
					
						
							|  |  |  |   sie_write(SIE_CMDCODE_ENDPOINT_SELECT+0, 0, 0); | 
					
						
							|  |  |  |   sie_write(SIE_CMDCODE_BUFFER_CLEAR     , 0, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return TUSB_ERROR_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // CONTROL PIPE API
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-03-11 13:13:04 +07:00
										 |  |  | void tusb_dcd_control_stall(uint8_t port) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   sie_write(SIE_CMDCODE_ENDPOINT_SET_STATUS+0, 1, SIE_SET_ENDPOINT_STALLED_MASK | SIE_SET_ENDPOINT_CONDITION_STALLED_MASK); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-11 13:13:04 +07:00
										 |  |  | bool tusb_dcd_control_xfer(uint8_t port, tusb_direction_t dir, uint8_t * p_buffer, uint16_t length, bool int_on_complete) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-09 14:29:23 +07:00
										 |  |  |   (void) port; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-03 12:30:38 +07:00
										 |  |  |   VERIFY( !(length != 0 && p_buffer == NULL) ); | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // determine Endpoint where Data & Status phase occurred (IN or OUT)
 | 
					
						
							|  |  |  |   uint8_t const ep_data   = (dir == TUSB_DIR_DEV_TO_HOST) ? 1 : 0; | 
					
						
							|  |  |  |   uint8_t const ep_status = 1 - ep_data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dcd_data.control_dma.int_on_complete = int_on_complete ? BIT_(ep_status) : 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //------------- Data Phase -------------//
 | 
					
						
							|  |  |  |   if ( length ) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     dcd_data.control_dma.p_data          = (uint8_t*) p_buffer; | 
					
						
							|  |  |  |     dcd_data.control_dma.remaining_bytes = length; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // lpc17xx already received the first DATA OUT packet by now
 | 
					
						
							| 
									
										
										
										
											2018-03-03 12:30:38 +07:00
										 |  |  |     VERIFY_STATUS ( pipe_control_xfer(ep_data, p_buffer, length), false ); | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //------------- Status Phase (opposite direct to Data) -------------//
 | 
					
						
							|  |  |  |   if (dir == TUSB_DIR_HOST_TO_DEV) | 
					
						
							| 
									
										
										
										
											2018-03-06 17:38:35 +07:00
										 |  |  |   { // only write for CONTROL OUT, CONTROL IN data will be retrieved in hal_dcd_isr // TODO ????
 | 
					
						
							| 
									
										
										
										
											2018-03-03 12:30:38 +07:00
										 |  |  |     VERIFY_STATUS ( pipe_control_write(NULL, 0), false ); | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-03 12:30:38 +07:00
										 |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // BULK/INTERRUPT/ISO PIPE API
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-03-11 14:22:04 +07:00
										 |  |  | edpt_hdl_t tusb_dcd_edpt_open(uint8_t port, tusb_descriptor_endpoint_t const * p_endpoint_desc, uint8_t class_code) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-09 14:29:23 +07:00
										 |  |  |   (void) port; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-11 12:37:07 +07:00
										 |  |  |   edpt_hdl_t const null_handle = { 0 }; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // TODO refractor to universal pipe open validation function
 | 
					
						
							|  |  |  |   if (p_endpoint_desc->bmAttributes.xfer == TUSB_XFER_ISOCHRONOUS) return null_handle; // TODO not support ISO yet
 | 
					
						
							|  |  |  |   ASSERT (p_endpoint_desc->wMaxPacketSize.size <= 64, null_handle); // TODO ISO can be 1023, but ISO not supported now
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   uint8_t ep_id = edpt_addr2phy( p_endpoint_desc->bEndpointAddress ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //------------- Realize Endpoint with Max Packet Size -------------//
 | 
					
						
							|  |  |  |   edpt_set_max_packet_size(ep_id, p_endpoint_desc->wMaxPacketSize.size); | 
					
						
							|  |  |  | 	dcd_data.class_code[ep_id] = class_code; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	//------------- first DD prepare -------------//
 | 
					
						
							|  |  |  | 	dcd_dma_descriptor_t* const p_dd = &dcd_data.dd[ep_id][0]; | 
					
						
							|  |  |  | 	memclr_(p_dd, sizeof(dcd_dma_descriptor_t)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	p_dd->is_isochronous  = (p_endpoint_desc->bmAttributes.xfer == TUSB_XFER_ISOCHRONOUS) ? 1 : 0; | 
					
						
							|  |  |  | 	p_dd->max_packet_size = p_endpoint_desc->wMaxPacketSize.size; | 
					
						
							|  |  |  | 	p_dd->is_retired      = 1; // inactive at first
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dcd_data.udca[ ep_id ] = p_dd; // hook to UDCA
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sie_write(SIE_CMDCODE_ENDPOINT_SET_STATUS+ep_id, 1, 0); // clear all endpoint status
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-11 12:37:07 +07:00
										 |  |  |   return (edpt_hdl_t) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |       { | 
					
						
							| 
									
										
										
										
											2018-03-09 14:29:23 +07:00
										 |  |  |           .port     = 0, | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  |           .index      = ep_id, | 
					
						
							|  |  |  |           .class_code = class_code | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-11 14:22:04 +07:00
										 |  |  | bool tusb_dcd_edpt_busy(edpt_hdl_t edpt_hdl) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   return (dcd_data.udca[edpt_hdl.index] != NULL && !dcd_data.udca[edpt_hdl.index]->is_retired); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-11 14:22:04 +07:00
										 |  |  | void tusb_dcd_edpt_stall(edpt_hdl_t edpt_hdl) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   sie_write(SIE_CMDCODE_ENDPOINT_SET_STATUS+edpt_hdl.index, 1, SIE_SET_ENDPOINT_STALLED_MASK); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-11 14:22:04 +07:00
										 |  |  | void tusb_dcd_edpt_clear_stall(uint8_t port, uint8_t edpt_addr) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   uint8_t ep_id = edpt_addr2phy(edpt_addr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sie_write(SIE_CMDCODE_ENDPOINT_SET_STATUS+ep_id, 1, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dd_xfer_init(dcd_dma_descriptor_t* p_dd, void* buffer, uint16_t total_bytes) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   p_dd->next                  = 0; | 
					
						
							|  |  |  |   p_dd->is_next_valid         = 0; | 
					
						
							|  |  |  |   p_dd->buffer_addr           = (uint32_t) buffer; | 
					
						
							|  |  |  |   p_dd->buffer_length         = total_bytes; | 
					
						
							|  |  |  |   p_dd->status                = DD_STATUS_NOT_SERVICED; | 
					
						
							|  |  |  |   p_dd->iso_last_packet_valid = 0; | 
					
						
							|  |  |  |   p_dd->present_count         = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-11 14:22:04 +07:00
										 |  |  | tusb_error_t tusb_dcd_edpt_queue_xfer(edpt_hdl_t edpt_hdl, uint8_t * buffer, uint16_t total_bytes) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | { // NOTE for sure the qhd has no dds
 | 
					
						
							|  |  |  |   dcd_dma_descriptor_t* const p_fixed_dd = &dcd_data.dd[edpt_hdl.index][0]; // always queue with the fixed DD
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dd_xfer_init(p_fixed_dd, buffer, total_bytes); | 
					
						
							|  |  |  |   p_fixed_dd->is_retired      = 1; | 
					
						
							|  |  |  |   p_fixed_dd->int_on_complete = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return TUSB_ERROR_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-11 14:22:04 +07:00
										 |  |  | tusb_error_t tusb_dcd_edpt_xfer(edpt_hdl_t edpt_hdl, uint8_t* buffer, uint16_t total_bytes, bool int_on_complete) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:08:52 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   dcd_dma_descriptor_t* const p_first_dd = &dcd_data.dd[edpt_hdl.index][0]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //------------- fixed DD is already queued a xfer -------------//
 | 
					
						
							|  |  |  |   if ( p_first_dd->buffer_length ) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     // setup new dd
 | 
					
						
							|  |  |  |     dcd_dma_descriptor_t* const p_dd = &dcd_data.dd[ edpt_hdl.index ][1]; | 
					
						
							|  |  |  |     memclr_(p_dd, sizeof(dcd_dma_descriptor_t)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     dd_xfer_init(p_dd, buffer, total_bytes); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     p_dd->max_packet_size     = p_first_dd->max_packet_size; | 
					
						
							|  |  |  |     p_dd->is_isochronous      = p_first_dd->is_isochronous; | 
					
						
							|  |  |  |     p_dd->int_on_complete     = int_on_complete; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // hook to fixed dd
 | 
					
						
							|  |  |  |     p_first_dd->next          = (uint32_t) p_dd; | 
					
						
							|  |  |  |     p_first_dd->is_next_valid = 1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   //------------- fixed DD is free -------------//
 | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     dd_xfer_init(p_first_dd, buffer, total_bytes); | 
					
						
							|  |  |  |     p_first_dd->int_on_complete = int_on_complete; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   p_first_dd->is_retired = 0; // activate xfer
 | 
					
						
							|  |  |  |   dcd_data.udca[edpt_hdl.index] = p_first_dd; | 
					
						
							|  |  |  |   LPC_USB->USBEpDMAEn = BIT_(edpt_hdl.index); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if ( edpt_hdl.index % 2 ) | 
					
						
							|  |  |  |   { // endpoint IN need to actively raise DMA request
 | 
					
						
							|  |  |  |     LPC_USB->USBDMARSet = BIT_(edpt_hdl.index); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return TUSB_ERROR_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 |