| 
									
										
										
										
											2019-09-14 18:01:52 -04:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * The MIT License (MIT) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2019 William D. Jones | 
					
						
							|  |  |  |  * Copyright (c) 2019 Ha Thach (tinyusb.org) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Permission is hereby granted, free of charge, to any person obtaining a copy | 
					
						
							|  |  |  |  * of this software and associated documentation files (the "Software"), to deal | 
					
						
							|  |  |  |  * in the Software without restriction, including without limitation the rights | 
					
						
							|  |  |  |  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | 
					
						
							|  |  |  |  * copies of the Software, and to permit persons to whom the Software is | 
					
						
							|  |  |  |  * furnished to do so, subject to the following conditions: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The above copyright notice and this permission notice shall be included in | 
					
						
							|  |  |  |  * all copies or substantial portions of the Software. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
					
						
							|  |  |  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
					
						
							|  |  |  |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | 
					
						
							|  |  |  |  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
					
						
							|  |  |  |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | 
					
						
							|  |  |  |  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | 
					
						
							|  |  |  |  * THE SOFTWARE. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This file is part of the TinyUSB stack. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "tusb_option.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if TUSB_OPT_DEVICE_ENABLED && ( CFG_TUSB_MCU == OPT_MCU_MSP430x5xx )
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 17:48:18 -04:00
										 |  |  | #include "msp430.h"
 | 
					
						
							| 
									
										
										
										
											2019-09-14 18:01:52 -04:00
										 |  |  | #include "device/dcd.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*------------------------------------------------------------------*/ | 
					
						
							|  |  |  | /* MACRO TYPEDEF CONSTANT ENUM
 | 
					
						
							|  |  |  |  *------------------------------------------------------------------*/ | 
					
						
							| 
									
										
										
										
											2019-09-24 03:43:51 -04:00
										 |  |  | #define USB_BUF_PTR(_x) (uint8_t *) ((uint16_t) _x)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 22:58:49 -04:00
										 |  |  | // usbpllir_mirror and usbmaintl_mirror can be added later if needed.
 | 
					
						
							|  |  |  | static volatile uint16_t usbiepie_mirror = 0; | 
					
						
							|  |  |  | static volatile uint16_t usboepie_mirror = 0; | 
					
						
							| 
									
										
										
										
											2019-09-24 02:41:11 -04:00
										 |  |  | static volatile uint8_t usbie_mirror = 0; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:58:49 -04:00
										 |  |  | static volatile uint16_t usbpwrctl_mirror = 0; | 
					
						
							| 
									
										
										
										
											2019-09-24 02:41:11 -04:00
										 |  |  | static bool in_isr = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | uint8_t _setup_packet[8]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void bus_reset(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-09-24 03:43:51 -04:00
										 |  |  |   // Enable the control EP 0. Also enable Indication Enable- a guard flag
 | 
					
						
							|  |  |  |   // separate from the Interrupt Enable mask.
 | 
					
						
							|  |  |  |   USBOEPCNF_0 |= (UBME | USBIIE); | 
					
						
							|  |  |  |   USBIEPCNF_0 |= (UBME | USBIIE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Enable interrupts for this endpoint.
 | 
					
						
							|  |  |  |   USBOEPIE |= BIT0; | 
					
						
							|  |  |  |   USBIEPIE |= BIT0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Clear NAK so packets can be received.
 | 
					
						
							|  |  |  |   // Dedicated buffers in hardware for SETUP and EP0, no setup needed.
 | 
					
						
							|  |  |  |   USBOEPCNT_0 &= ~NAK; | 
					
						
							|  |  |  |   USBIEPCNT_0 &= ~NAK; | 
					
						
							| 
									
										
										
										
											2019-09-24 04:06:34 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Now safe to respond to SETUP packets.
 | 
					
						
							|  |  |  |   USBIE |= SETUPIE; | 
					
						
							| 
									
										
										
										
											2019-09-24 02:41:11 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-09-14 18:01:52 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*------------------------------------------------------------------*/ | 
					
						
							|  |  |  | /* Controller API
 | 
					
						
							|  |  |  |  *------------------------------------------------------------------*/ | 
					
						
							|  |  |  | void dcd_init (uint8_t rhport) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							| 
									
										
										
										
											2019-09-23 18:53:31 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   USBKEYPID = USBKEY; | 
					
						
							| 
									
										
										
										
											2019-09-24 02:41:11 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Enable the module (required to write config regs)!
 | 
					
						
							|  |  |  |   USBCNF |= USB_EN; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Reset used interrupts
 | 
					
						
							|  |  |  |   USBOEPIE = 0; | 
					
						
							|  |  |  |   USBIEPIE = 0; | 
					
						
							|  |  |  |   USBIE = 0; | 
					
						
							|  |  |  |   USBOEPIFG = 0; | 
					
						
							|  |  |  |   USBIEPIFG = 0; | 
					
						
							|  |  |  |   USBIFG = 0; | 
					
						
							|  |  |  |   USBPWRCTL &= ~(VUOVLIE | VBONIE | VBOFFIE | VUOVLIFG | VBONIFG | VBOFFIFG); | 
					
						
							|  |  |  |   USBVECINT = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Enable reset and wait for it before continuing.
 | 
					
						
							|  |  |  |   USBIE |= RSTRIE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Enable pullup.
 | 
					
						
							|  |  |  |   USBCNF |= PUR_EN; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 18:53:31 -04:00
										 |  |  |   USBKEYPID = 0; | 
					
						
							| 
									
										
										
										
											2019-09-14 18:01:52 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 22:58:49 -04:00
										 |  |  | // There is no "USB peripheral interrupt disable" bit on MSP430, so we have
 | 
					
						
							|  |  |  | // to save the relevant registers individually.
 | 
					
						
							|  |  |  | // WARNING: Unlike the ARM/NVIC routines, these functions are _not_ idempotent
 | 
					
						
							|  |  |  | // if you modified the registers saved in between calls so they don't match
 | 
					
						
							|  |  |  | // the mirrors; mirrors will be updated to reflect most recent register
 | 
					
						
							|  |  |  | // contents.
 | 
					
						
							| 
									
										
										
										
											2019-09-14 18:01:52 -04:00
										 |  |  | void dcd_int_enable (uint8_t rhport) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:58:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   __bic_SR_register(GIE); // Unlikely to be called in ISR, but let's be safe.
 | 
					
						
							|  |  |  |                           // Also, this cleanly disables all USB interrupts
 | 
					
						
							|  |  |  |                           // atomically from application's POV.
 | 
					
						
							| 
									
										
										
										
											2019-09-24 02:41:11 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // This guard is required because tinyusb can enable interrupts without
 | 
					
						
							|  |  |  |   // having disabled them first.
 | 
					
						
							|  |  |  |   if(in_isr) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     USBOEPIE = usboepie_mirror; | 
					
						
							|  |  |  |     USBIEPIE = usbiepie_mirror; | 
					
						
							|  |  |  |     USBIE = usbie_mirror; | 
					
						
							|  |  |  |     USBPWRCTL |= usbpwrctl_mirror; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   in_isr = false; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:58:49 -04:00
										 |  |  |   __bis_SR_register(GIE); | 
					
						
							| 
									
										
										
										
											2019-09-14 18:01:52 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dcd_int_disable (uint8_t rhport) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:58:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   __bic_SR_register(GIE); | 
					
						
							|  |  |  |   usboepie_mirror = USBOEPIE; | 
					
						
							|  |  |  |   usbiepie_mirror = USBIEPIE; | 
					
						
							|  |  |  |   usbie_mirror = USBIE; | 
					
						
							|  |  |  |   usbpwrctl_mirror = (USBPWRCTL & (VUOVLIE | VBONIE | VBOFFIE)); | 
					
						
							|  |  |  |   USBOEPIE = 0; | 
					
						
							|  |  |  |   USBIEPIE = 0; | 
					
						
							|  |  |  |   USBIE = 0; | 
					
						
							|  |  |  |   USBPWRCTL &= ~(VUOVLIE | VBONIE | VBOFFIE); | 
					
						
							| 
									
										
										
										
											2019-09-24 02:41:11 -04:00
										 |  |  |   in_isr = true; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:58:49 -04:00
										 |  |  |   __bis_SR_register(GIE); | 
					
						
							| 
									
										
										
										
											2019-09-14 18:01:52 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dcd_set_address (uint8_t rhport, uint8_t dev_addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							|  |  |  |   (void) dev_addr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dcd_set_config (uint8_t rhport, uint8_t config_num) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							|  |  |  |   (void) config_num; | 
					
						
							|  |  |  |   // Nothing to do
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dcd_remote_wakeup(uint8_t rhport) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*------------------------------------------------------------------*/ | 
					
						
							|  |  |  | /* DCD Endpoint port
 | 
					
						
							|  |  |  |  *------------------------------------------------------------------*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool dcd_edpt_open (uint8_t rhport, tusb_desc_endpoint_t const * desc_edpt) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							|  |  |  |   (void) desc_edpt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool dcd_edpt_xfer (uint8_t rhport, uint8_t ep_addr, uint8_t * buffer, uint16_t total_bytes) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							|  |  |  |   (void) ep_addr; | 
					
						
							|  |  |  |   (void) buffer; | 
					
						
							|  |  |  |   (void) total_bytes; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dcd_edpt_stall (uint8_t rhport, uint8_t ep_addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							|  |  |  |   (void) ep_addr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dcd_edpt_clear_stall (uint8_t rhport, uint8_t ep_addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							|  |  |  |   (void) ep_addr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*------------------------------------------------------------------*/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 03:43:51 -04:00
										 |  |  | static void receive_packet(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void transmit_packet(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 02:41:11 -04:00
										 |  |  | static void handle_setup_packet(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-09-24 03:43:51 -04:00
										 |  |  |   volatile uint8_t * setup_buf = &USBSUBLK; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for(int i = 0; i < 8; i++) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     _setup_packet[i] = setup_buf[i]; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-09-24 02:41:11 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 03:43:51 -04:00
										 |  |  |   dcd_event_setup_received(0, (uint8_t*) &_setup_packet[0], true); | 
					
						
							| 
									
										
										
										
											2019-09-24 02:41:11 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-14 18:01:52 -04:00
										 |  |  | void __attribute__ ((interrupt(USB_UBM_VECTOR))) USB_UBM_ISR(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-09-24 02:41:11 -04:00
										 |  |  |   // Setup is special- reading USBVECINT to handle setup packets is done to
 | 
					
						
							|  |  |  |   // stop NAKs on EP0.
 | 
					
						
							|  |  |  |   uint8_t setup_status = USBIFG & SETUPIFG; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if(setup_status) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     handle_setup_packet(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   uint16_t curr_vector = USBVECINT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch(curr_vector) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     case USBVECINT_RSTR: | 
					
						
							|  |  |  |       bus_reset(); | 
					
						
							|  |  |  |       dcd_event_bus_signal(0, DCD_EVENT_BUS_RESET, true); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Clear the NAK on EP 0 after a SETUP packet is received.
 | 
					
						
							|  |  |  |     case USBVECINT_SETUP_PACKET_RECEIVED: | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 03:43:51 -04:00
										 |  |  |     case USBVECINT_INPUT_ENDPOINT0: | 
					
						
							|  |  |  |       transmit_packet(); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case USBVECINT_OUTPUT_ENDPOINT0: | 
					
						
							|  |  |  |       receive_packet(); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 02:41:11 -04:00
										 |  |  |     default: | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-09-14 18:01:52 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 |