2016-08-22 23:48:09 -07:00
|
|
|
#include <string.h>
|
|
|
|
|
2017-05-12 16:45:38 -07:00
|
|
|
#include "autoreload.h"
|
2016-10-05 11:07:29 -07:00
|
|
|
#include "compiler.h"
|
|
|
|
#include "asf/common/services/sleepmgr/sleepmgr.h"
|
2016-08-22 23:48:09 -07:00
|
|
|
#include "asf/common/services/usb/class/cdc/device/udi_cdc.h"
|
2016-08-24 13:17:55 -07:00
|
|
|
#include "asf/common2/services/delay/delay.h"
|
2016-09-15 17:01:19 -07:00
|
|
|
#include "asf/sam0/drivers/port/port.h"
|
2016-08-23 16:47:53 -07:00
|
|
|
#include "asf/sam0/drivers/sercom/usart/usart.h"
|
2016-10-28 20:16:39 -07:00
|
|
|
#include "lib/mp-readline/readline.h"
|
2017-01-10 15:57:04 -08:00
|
|
|
#include "lib/utils/interrupt_char.h"
|
2016-08-22 23:48:09 -07:00
|
|
|
#include "py/mphal.h"
|
2016-08-23 16:47:53 -07:00
|
|
|
#include "py/mpstate.h"
|
2017-09-06 17:15:47 -04:00
|
|
|
#include "py/runtime.h"
|
2016-10-26 13:32:41 -07:00
|
|
|
#include "py/smallint.h"
|
2016-11-03 15:50:59 -07:00
|
|
|
#include "shared-bindings/time/__init__.h"
|
2016-08-22 23:48:09 -07:00
|
|
|
|
2016-08-23 16:47:53 -07:00
|
|
|
#include "mpconfigboard.h"
|
2016-08-31 00:11:56 -07:00
|
|
|
#include "mphalport.h"
|
|
|
|
|
|
|
|
// Store received characters on our own so that we can filter control characters
|
|
|
|
// and act immediately on CTRL-C for example.
|
|
|
|
// This is adapted from asf/thirdparty/wireless/addons/sio2host
|
|
|
|
|
|
|
|
// Receive buffer
|
|
|
|
static uint8_t usb_rx_buf[USB_RX_BUF_SIZE];
|
|
|
|
|
|
|
|
// Receive buffer head
|
2016-09-21 15:13:37 -07:00
|
|
|
static volatile uint8_t usb_rx_buf_head;
|
2016-08-31 00:11:56 -07:00
|
|
|
|
|
|
|
// Receive buffer tail
|
2016-09-21 15:13:37 -07:00
|
|
|
static volatile uint8_t usb_rx_buf_tail;
|
2016-08-31 00:11:56 -07:00
|
|
|
|
|
|
|
// Number of bytes in receive buffer
|
2016-12-09 19:27:41 -08:00
|
|
|
volatile uint8_t usb_rx_count;
|
2016-08-23 16:47:53 -07:00
|
|
|
|
2016-12-09 19:27:41 -08:00
|
|
|
volatile bool mp_cdc_enabled = false;
|
2016-08-31 00:11:56 -07:00
|
|
|
|
2016-08-23 16:47:53 -07:00
|
|
|
extern struct usart_module usart_instance;
|
|
|
|
|
2017-01-26 18:05:46 -08:00
|
|
|
// Read by main to know when USB is connected.
|
|
|
|
volatile bool mp_msc_enabled = false;
|
2017-07-06 13:41:13 -07:00
|
|
|
bool mp_msc_enable() {
|
|
|
|
mp_msc_enabled = true;
|
|
|
|
return true;
|
2016-10-05 11:07:29 -07:00
|
|
|
}
|
|
|
|
|
2017-07-06 13:41:13 -07:00
|
|
|
void mp_msc_disable() {
|
|
|
|
mp_msc_enabled = false;
|
2016-10-05 11:07:29 -07:00
|
|
|
}
|
|
|
|
|
2017-07-06 13:41:13 -07:00
|
|
|
bool mp_cdc_enable(uint8_t port) {
|
|
|
|
mp_cdc_enabled = false;
|
|
|
|
return true;
|
2016-08-22 23:48:09 -07:00
|
|
|
}
|
|
|
|
|
2017-07-06 13:41:13 -07:00
|
|
|
void mp_cdc_disable(uint8_t port) {
|
|
|
|
mp_cdc_enabled = false;
|
2016-08-22 23:48:09 -07:00
|
|
|
}
|
|
|
|
|
2017-07-06 13:41:13 -07:00
|
|
|
volatile bool reset_on_disconnect = false;
|
|
|
|
|
2016-10-25 18:23:04 -07:00
|
|
|
void usb_dtr_notify(uint8_t port, bool set) {
|
2017-07-06 13:41:13 -07:00
|
|
|
mp_cdc_enabled = set;
|
|
|
|
if (!set && reset_on_disconnect) {
|
|
|
|
reset_to_bootloader();
|
|
|
|
}
|
2016-10-25 18:23:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void usb_rts_notify(uint8_t port, bool set) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-07-06 13:41:13 -07:00
|
|
|
void usb_coding_notify(uint8_t port, usb_cdc_line_coding_t* coding) {
|
|
|
|
reset_on_disconnect = coding->dwDTERate == 1200;
|
|
|
|
}
|
|
|
|
|
|
|
|
void usb_rx_notify(void) {
|
2016-09-21 15:13:37 -07:00
|
|
|
irqflags_t flags;
|
2016-08-31 00:11:56 -07:00
|
|
|
if (mp_cdc_enabled) {
|
|
|
|
while (udi_cdc_is_rx_ready()) {
|
|
|
|
uint8_t c;
|
|
|
|
|
2016-09-21 15:13:37 -07:00
|
|
|
// Introduce a critical section to avoid buffer corruption. We use
|
|
|
|
// cpu_irq_save instead of cpu_irq_disable because we don't know the
|
|
|
|
// current state of IRQs. They may have been turned off already and
|
|
|
|
// we don't want to accidentally turn them back on.
|
|
|
|
flags = cpu_irq_save();
|
|
|
|
// If our buffer is full, then don't get another character otherwise
|
|
|
|
// we'll lose a previous character.
|
|
|
|
if (usb_rx_count >= USB_RX_BUF_SIZE) {
|
|
|
|
cpu_irq_restore(flags);
|
|
|
|
break;
|
2016-08-31 00:11:56 -07:00
|
|
|
}
|
|
|
|
|
2016-09-21 15:13:37 -07:00
|
|
|
uint8_t current_tail = usb_rx_buf_tail;
|
|
|
|
// Pretend we've received a character so that any nested calls to
|
|
|
|
// this function have to consider the spot we've reserved.
|
2016-08-31 00:11:56 -07:00
|
|
|
if ((USB_RX_BUF_SIZE - 1) == usb_rx_buf_tail) {
|
|
|
|
// Reached the end of buffer, revert back to beginning of
|
|
|
|
// buffer.
|
|
|
|
usb_rx_buf_tail = 0x00;
|
|
|
|
} else {
|
|
|
|
usb_rx_buf_tail++;
|
|
|
|
}
|
2016-09-21 15:13:37 -07:00
|
|
|
// The count of characters present in receive buffer is
|
|
|
|
// incremented.
|
|
|
|
usb_rx_count++;
|
|
|
|
// WARNING(tannewt): This call can call us back with the next
|
|
|
|
// character!
|
|
|
|
c = udi_cdc_getc();
|
|
|
|
|
2017-01-10 15:57:04 -08:00
|
|
|
if (c == mp_interrupt_char) {
|
2016-09-21 15:13:37 -07:00
|
|
|
// We consumed a character rather than adding it to the rx
|
|
|
|
// buffer so undo the modifications we made to count and the
|
|
|
|
// tail.
|
|
|
|
usb_rx_count--;
|
|
|
|
usb_rx_buf_tail = current_tail;
|
|
|
|
cpu_irq_restore(flags);
|
|
|
|
mp_keyboard_interrupt();
|
|
|
|
// Don't put the interrupt into the buffer, just continue.
|
|
|
|
continue;
|
|
|
|
}
|
2016-08-31 00:11:56 -07:00
|
|
|
|
2016-09-21 15:13:37 -07:00
|
|
|
// We put the next character where we expected regardless of whether
|
|
|
|
// the next character was already loaded in the buffer.
|
|
|
|
usb_rx_buf[current_tail] = c;
|
|
|
|
|
|
|
|
cpu_irq_restore(flags);
|
2016-08-31 00:11:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-03 15:50:59 -07:00
|
|
|
int receive_usb(void) {
|
2016-09-21 15:13:37 -07:00
|
|
|
if (usb_rx_count == 0) {
|
2016-08-31 00:11:56 -07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy from head.
|
2016-09-21 15:13:37 -07:00
|
|
|
cpu_irq_disable();
|
2016-08-31 00:11:56 -07:00
|
|
|
int data = usb_rx_buf[usb_rx_buf_head];
|
|
|
|
usb_rx_buf_head++;
|
|
|
|
usb_rx_count--;
|
|
|
|
if ((USB_RX_BUF_SIZE) == usb_rx_buf_head) {
|
|
|
|
usb_rx_buf_head = 0;
|
|
|
|
}
|
2016-09-21 15:13:37 -07:00
|
|
|
cpu_irq_enable();
|
|
|
|
|
|
|
|
// Call usb_rx_notify if we just emptied a spot in the buffer.
|
|
|
|
if (usb_rx_count == USB_RX_BUF_SIZE - 1) {
|
|
|
|
usb_rx_notify();
|
|
|
|
}
|
2016-08-31 00:11:56 -07:00
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2016-08-22 23:48:09 -07:00
|
|
|
int mp_hal_stdin_rx_chr(void) {
|
2016-09-15 17:01:19 -07:00
|
|
|
for (;;) {
|
2016-11-29 14:58:37 -08:00
|
|
|
#ifdef MICROPY_VM_HOOK_LOOP
|
|
|
|
MICROPY_VM_HOOK_LOOP
|
|
|
|
#endif
|
2017-09-06 17:15:47 -04:00
|
|
|
// Check to see if we've been CTRL-Ced by autoreload or the user. Raise the exception if so.
|
|
|
|
if(MP_STATE_VM(mp_pending_exception) == MP_OBJ_FROM_PTR(&MP_STATE_VM(mp_kbd_exception))) {
|
|
|
|
mp_handle_pending();
|
|
|
|
}
|
|
|
|
|
2016-09-15 17:01:19 -07:00
|
|
|
#ifdef USB_REPL
|
2017-05-12 16:45:38 -07:00
|
|
|
if (reload_next_character) {
|
2016-10-28 20:16:39 -07:00
|
|
|
return CHAR_CTRL_D;
|
|
|
|
}
|
2016-10-27 17:48:42 -07:00
|
|
|
if (usb_rx_count > 0) {
|
2016-09-15 17:01:19 -07:00
|
|
|
#ifdef MICROPY_HW_LED_RX
|
|
|
|
port_pin_toggle_output_level(MICROPY_HW_LED_RX);
|
|
|
|
#endif
|
|
|
|
return receive_usb();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef UART_REPL
|
|
|
|
uint16_t temp;
|
|
|
|
if (usart_read_wait(&usart_instance, &temp) == STATUS_OK) {
|
|
|
|
#ifdef MICROPY_HW_LED_RX
|
|
|
|
port_pin_toggle_output_level(MICROPY_HW_LED_RX);
|
|
|
|
#endif
|
|
|
|
return temp;
|
|
|
|
}
|
|
|
|
#endif
|
2016-10-05 11:07:29 -07:00
|
|
|
// TODO(tannewt): Switch to callback/interrupt based UART so it can work
|
|
|
|
// with the sleepmgr.
|
|
|
|
sleepmgr_enter_sleep();
|
2016-08-22 23:48:09 -07:00
|
|
|
}
|
2016-09-15 17:01:19 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void mp_hal_stdout_tx_strn(const char *str, size_t len) {
|
|
|
|
#ifdef MICROPY_HW_LED_TX
|
|
|
|
port_pin_toggle_output_level(MICROPY_HW_LED_TX);
|
2016-08-23 16:47:53 -07:00
|
|
|
#endif
|
2016-09-15 17:01:19 -07:00
|
|
|
|
2016-08-23 16:47:53 -07:00
|
|
|
#ifdef UART_REPL
|
2016-09-15 17:01:19 -07:00
|
|
|
usart_write_buffer_wait(&usart_instance, (uint8_t*) str, len);
|
2016-08-23 16:47:53 -07:00
|
|
|
#endif
|
2016-08-22 23:48:09 -07:00
|
|
|
|
2017-05-12 18:26:14 -07:00
|
|
|
#ifdef CIRCUITPY_BOOT_OUTPUT_FILE
|
|
|
|
if (boot_output_file != NULL) {
|
|
|
|
UINT bytes_written = 0;
|
|
|
|
f_write(boot_output_file, str, len, &bytes_written);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2016-09-15 17:01:19 -07:00
|
|
|
#ifdef USB_REPL
|
2016-10-25 18:23:04 -07:00
|
|
|
// Always make sure there is enough room in the usb buffer for the outgoing
|
|
|
|
// string. If there isn't we risk getting caught in a loop within the usb
|
|
|
|
// code as it tries to send all the characters it can't buffer.
|
2016-10-26 13:32:41 -07:00
|
|
|
uint32_t start = 0;
|
2016-11-03 15:50:59 -07:00
|
|
|
uint64_t start_tick = common_hal_time_monotonic();
|
|
|
|
uint64_t duration = 0;
|
2016-10-26 13:32:41 -07:00
|
|
|
if (mp_cdc_enabled) {
|
|
|
|
while (start < len && duration < 10) {
|
|
|
|
uint8_t buffer_space = udi_cdc_get_free_tx_buffer();
|
|
|
|
uint8_t transmit = min(len - start, buffer_space);
|
|
|
|
if (transmit > 0) {
|
|
|
|
if (udi_cdc_write_buf(str + start, transmit) > 0) {
|
|
|
|
// It didn't transmit successfully so give up.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
start += transmit;
|
2016-11-29 14:58:37 -08:00
|
|
|
#ifdef MICROPY_VM_HOOK_LOOP
|
|
|
|
MICROPY_VM_HOOK_LOOP
|
|
|
|
#endif
|
2016-11-03 15:50:59 -07:00
|
|
|
duration = (common_hal_time_monotonic() - start_tick);
|
2016-10-26 13:32:41 -07:00
|
|
|
}
|
2016-09-15 17:01:19 -07:00
|
|
|
}
|
|
|
|
#endif
|
2016-08-22 23:48:09 -07:00
|
|
|
}
|
|
|
|
|
2016-08-24 13:17:55 -07:00
|
|
|
void mp_hal_delay_ms(mp_uint_t delay) {
|
2017-05-10 18:21:02 -07:00
|
|
|
uint64_t start_tick = ticks_ms;
|
|
|
|
uint64_t duration = 0;
|
|
|
|
while (duration < delay) {
|
|
|
|
#ifdef MICROPY_VM_HOOK_LOOP
|
|
|
|
MICROPY_VM_HOOK_LOOP
|
|
|
|
#endif
|
|
|
|
// Check to see if we've been CTRL-Ced by autoreload or the user.
|
|
|
|
if(MP_STATE_VM(mp_pending_exception) == MP_OBJ_FROM_PTR(&MP_STATE_VM(mp_kbd_exception))) {
|
|
|
|
break;
|
2016-10-27 17:48:42 -07:00
|
|
|
}
|
2017-05-10 18:21:02 -07:00
|
|
|
duration = (ticks_ms - start_tick);
|
|
|
|
// TODO(tannewt): Go to sleep for a little while while we wait.
|
2016-10-05 11:07:29 -07:00
|
|
|
}
|
2016-08-24 13:17:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void mp_hal_delay_us(mp_uint_t delay) {
|
2016-10-05 11:07:29 -07:00
|
|
|
delay_us(delay);
|
2016-08-24 13:17:55 -07:00
|
|
|
}
|
2016-10-13 04:59:43 +00:00
|
|
|
|
|
|
|
// Interrupt flags that will be saved and restored during disable/Enable
|
|
|
|
// interrupt functions below.
|
|
|
|
static irqflags_t irq_flags;
|
|
|
|
|
|
|
|
void mp_hal_disable_all_interrupts(void) {
|
2017-01-17 19:26:28 -08:00
|
|
|
// Disable all interrupt sources for timing critical sections.
|
|
|
|
// Disable ASF-based interrupts.
|
|
|
|
irq_flags = cpu_irq_save();
|
2016-10-13 04:59:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void mp_hal_enable_all_interrupts(void) {
|
2017-01-17 19:26:28 -08:00
|
|
|
// Enable all interrupt sources after timing critical sections.
|
|
|
|
// Restore ASF-based interrupts.
|
|
|
|
cpu_irq_restore(irq_flags);
|
2016-10-13 04:59:43 +00:00
|
|
|
}
|