Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 1 | /* |
| 2 | * This file is part of the Micro Python project, http://micropython.org/ |
| 3 | * |
| 4 | * The MIT License (MIT) |
| 5 | * |
| 6 | * Copyright (c) 2014 Damien P. George |
| 7 | * |
| 8 | * Permission is hereby granted, free of charge, to any person obtaining a copy |
| 9 | * of this software and associated documentation files (the "Software"), to deal |
| 10 | * in the Software without restriction, including without limitation the rights |
| 11 | * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
| 12 | * copies of the Software, and to permit persons to whom the Software is |
| 13 | * furnished to do so, subject to the following conditions: |
| 14 | * |
| 15 | * The above copyright notice and this permission notice shall be included in |
| 16 | * all copies or substantial portions of the Software. |
| 17 | * |
| 18 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
| 19 | * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
| 20 | * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
| 21 | * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
| 22 | * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
| 23 | * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN |
| 24 | * THE SOFTWARE. |
| 25 | */ |
| 26 | |
| 27 | #include <stdio.h> |
| 28 | #include "ets_sys.h" |
| 29 | #include "etshal.h" |
| 30 | #include "uart.h" |
| 31 | #include "esp_mphal.h" |
Josef Gajdusek | 800d5cd | 2015-05-10 17:54:09 +0200 | [diff] [blame] | 32 | #include "user_interface.h" |
Paul Sokolovsky | a099bfe | 2016-03-11 09:38:20 +0700 | [diff] [blame] | 33 | #include "ets_alt_task.h" |
Paul Sokolovsky | 8ab16b6 | 2015-12-29 20:44:55 +0200 | [diff] [blame] | 34 | #include "py/obj.h" |
| 35 | #include "py/mpstate.h" |
Paul Sokolovsky | c961889 | 2016-03-29 11:13:32 +0300 | [diff] [blame] | 36 | #include "extmod/misc.h" |
Paul Sokolovsky | 98af891 | 2016-03-31 19:49:55 +0300 | [diff] [blame^] | 37 | #include "lib/utils/pyexec.h" |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 38 | |
| 39 | extern void ets_wdt_disable(void); |
| 40 | extern void wdt_feed(void); |
| 41 | extern void ets_delay_us(); |
| 42 | |
Paul Sokolovsky | 61fa7c8 | 2016-03-30 18:50:38 +0300 | [diff] [blame] | 43 | STATIC byte input_buf_array[256]; |
| 44 | ringbuf_t input_buf = {input_buf_array, sizeof(input_buf_array)}; |
Paul Sokolovsky | 402a743 | 2016-03-29 11:48:43 +0300 | [diff] [blame] | 45 | void mp_hal_debug_tx_strn_cooked(void *env, const char *str, uint32_t len); |
| 46 | const mp_print_t mp_debug_print = {NULL, mp_hal_debug_tx_strn_cooked}; |
| 47 | |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 48 | void mp_hal_init(void) { |
| 49 | ets_wdt_disable(); // it's a pain while developing |
Paul Sokolovsky | a4c8ef9 | 2016-02-08 21:43:37 +0200 | [diff] [blame] | 50 | mp_hal_rtc_init(); |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 51 | uart_init(UART_BIT_RATE_115200, UART_BIT_RATE_115200); |
| 52 | } |
| 53 | |
| 54 | void mp_hal_feed_watchdog(void) { |
| 55 | //ets_wdt_disable(); // it's a pain while developing |
| 56 | //WRITE_PERI_REG(0x60000914, 0x73); |
| 57 | //wdt_feed(); // might also work |
| 58 | } |
| 59 | |
Paul Sokolovsky | 5699fc9 | 2015-10-29 02:06:13 +0300 | [diff] [blame] | 60 | void mp_hal_delay_us(uint32_t us) { |
Damien George | e673714 | 2016-03-23 13:01:21 +0200 | [diff] [blame] | 61 | uint32_t start = system_get_time(); |
| 62 | while (system_get_time() - start < us) { |
| 63 | ets_event_poll(); |
| 64 | } |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 65 | } |
| 66 | |
Damien George | 0b32e50 | 2015-02-13 15:04:53 +0000 | [diff] [blame] | 67 | int mp_hal_stdin_rx_chr(void) { |
| 68 | for (;;) { |
| 69 | int c = uart0_rx(); |
| 70 | if (c != -1) { |
| 71 | return c; |
| 72 | } |
Paul Sokolovsky | 5699fc9 | 2015-10-29 02:06:13 +0300 | [diff] [blame] | 73 | mp_hal_delay_us(1); |
Damien George | 0b32e50 | 2015-02-13 15:04:53 +0000 | [diff] [blame] | 74 | mp_hal_feed_watchdog(); |
| 75 | } |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 76 | } |
| 77 | |
Paul Sokolovsky | c961889 | 2016-03-29 11:13:32 +0300 | [diff] [blame] | 78 | void mp_hal_stdout_tx_char(char c) { |
| 79 | uart_tx_one_char(UART0, c); |
| 80 | mp_uos_dupterm_tx_strn(&c, 1); |
| 81 | } |
| 82 | |
Paul Sokolovsky | 402a743 | 2016-03-29 11:48:43 +0300 | [diff] [blame] | 83 | #if 0 |
| 84 | void mp_hal_debug_str(const char *str) { |
| 85 | while (*str) { |
| 86 | uart_tx_one_char(UART0, *str++); |
| 87 | } |
| 88 | uart_flush(UART0); |
| 89 | } |
| 90 | #endif |
| 91 | |
Damien George | 0b32e50 | 2015-02-13 15:04:53 +0000 | [diff] [blame] | 92 | void mp_hal_stdout_tx_str(const char *str) { |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 93 | while (*str) { |
Paul Sokolovsky | c961889 | 2016-03-29 11:13:32 +0300 | [diff] [blame] | 94 | mp_hal_stdout_tx_char(*str++); |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 95 | } |
| 96 | } |
| 97 | |
Damien George | 0b32e50 | 2015-02-13 15:04:53 +0000 | [diff] [blame] | 98 | void mp_hal_stdout_tx_strn(const char *str, uint32_t len) { |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 99 | while (len--) { |
Paul Sokolovsky | c961889 | 2016-03-29 11:13:32 +0300 | [diff] [blame] | 100 | mp_hal_stdout_tx_char(*str++); |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 101 | } |
| 102 | } |
| 103 | |
Damien George | 0b32e50 | 2015-02-13 15:04:53 +0000 | [diff] [blame] | 104 | void mp_hal_stdout_tx_strn_cooked(const char *str, uint32_t len) { |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 105 | while (len--) { |
| 106 | if (*str == '\n') { |
Paul Sokolovsky | c961889 | 2016-03-29 11:13:32 +0300 | [diff] [blame] | 107 | mp_hal_stdout_tx_char('\r'); |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 108 | } |
Paul Sokolovsky | c961889 | 2016-03-29 11:13:32 +0300 | [diff] [blame] | 109 | mp_hal_stdout_tx_char(*str++); |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 110 | } |
| 111 | } |
| 112 | |
Paul Sokolovsky | 402a743 | 2016-03-29 11:48:43 +0300 | [diff] [blame] | 113 | void mp_hal_debug_tx_strn_cooked(void *env, const char *str, uint32_t len) { |
| 114 | (void)env; |
| 115 | while (len--) { |
| 116 | if (*str == '\n') { |
| 117 | uart_tx_one_char(UART0, '\r'); |
| 118 | } |
| 119 | uart_tx_one_char(UART0, *str++); |
| 120 | } |
| 121 | } |
| 122 | |
Paul Sokolovsky | 6a09e7d | 2015-10-29 19:35:04 +0300 | [diff] [blame] | 123 | uint32_t mp_hal_ticks_ms(void) { |
Josef Gajdusek | 800d5cd | 2015-05-10 17:54:09 +0200 | [diff] [blame] | 124 | return system_get_time() / 1000; |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 125 | } |
| 126 | |
Damien George | b41a14a | 2015-12-28 17:27:44 +0000 | [diff] [blame] | 127 | uint32_t mp_hal_ticks_us(void) { |
| 128 | return system_get_time(); |
| 129 | } |
| 130 | |
Paul Sokolovsky | ebd9f55 | 2015-10-29 13:03:44 +0300 | [diff] [blame] | 131 | void mp_hal_delay_ms(uint32_t delay) { |
| 132 | mp_hal_delay_us(delay * 1000); |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 133 | } |
| 134 | |
| 135 | void mp_hal_set_interrupt_char(int c) { |
Paul Sokolovsky | d3a4d39 | 2015-12-20 13:58:58 +0200 | [diff] [blame] | 136 | if (c != -1) { |
| 137 | mp_obj_exception_clear_traceback(MP_STATE_PORT(mp_kbd_exception)); |
| 138 | } |
| 139 | extern int interrupt_char; |
| 140 | interrupt_char = c; |
Damien George | 075d597 | 2014-11-27 20:30:33 +0000 | [diff] [blame] | 141 | } |
Paul Sokolovsky | 8ab16b6 | 2015-12-29 20:44:55 +0200 | [diff] [blame] | 142 | |
Paul Sokolovsky | a099bfe | 2016-03-11 09:38:20 +0700 | [diff] [blame] | 143 | void ets_event_poll(void) { |
| 144 | ets_loop_iter(); |
| 145 | if (MP_STATE_VM(mp_pending_exception) != NULL) { |
| 146 | mp_obj_t obj = MP_STATE_VM(mp_pending_exception); |
| 147 | MP_STATE_VM(mp_pending_exception) = MP_OBJ_NULL; |
| 148 | nlr_raise(obj); |
| 149 | } |
| 150 | } |
| 151 | |
Paul Sokolovsky | 8ab16b6 | 2015-12-29 20:44:55 +0200 | [diff] [blame] | 152 | void __assert_func(const char *file, int line, const char *func, const char *expr) { |
| 153 | printf("assert:%s:%d:%s: %s\n", file, line, func, expr); |
| 154 | nlr_raise(mp_obj_new_exception_msg(&mp_type_AssertionError, |
| 155 | "C-level assert")); |
| 156 | } |
Paul Sokolovsky | 61fa7c8 | 2016-03-30 18:50:38 +0300 | [diff] [blame] | 157 | |
| 158 | void mp_hal_signal_input(void) { |
| 159 | system_os_post(UART_TASK_ID, 0, 0); |
| 160 | } |
Paul Sokolovsky | 98af891 | 2016-03-31 19:49:55 +0300 | [diff] [blame^] | 161 | |
| 162 | static int call_dupterm_read(void) { |
| 163 | if (MP_STATE_PORT(term_obj) == NULL) { |
| 164 | return -1; |
| 165 | } |
| 166 | |
| 167 | nlr_buf_t nlr; |
| 168 | if (nlr_push(&nlr) == 0) { |
| 169 | mp_obj_t read_m[3]; |
| 170 | mp_load_method(MP_STATE_PORT(term_obj), MP_QSTR_read, read_m); |
| 171 | read_m[2] = MP_OBJ_NEW_SMALL_INT(1); |
| 172 | mp_obj_t res = mp_call_method_n_kw(1, 0, read_m); |
| 173 | if (res == mp_const_none) { |
| 174 | return -2; |
| 175 | } |
| 176 | mp_buffer_info_t bufinfo; |
| 177 | mp_get_buffer_raise(res, &bufinfo, MP_BUFFER_READ); |
| 178 | if (bufinfo.len == 0) { |
| 179 | mp_printf(&mp_plat_print, "dupterm: EOF received, deactivating\n"); |
| 180 | MP_STATE_PORT(term_obj) = NULL; |
| 181 | return -1; |
| 182 | } |
| 183 | nlr_pop(); |
| 184 | return *(byte*)bufinfo.buf; |
| 185 | } else { |
| 186 | // Temporarily disable dupterm to avoid infinite recursion |
| 187 | mp_obj_t save_term = MP_STATE_PORT(term_obj); |
| 188 | MP_STATE_PORT(term_obj) = NULL; |
| 189 | mp_printf(&mp_plat_print, "dupterm: "); |
| 190 | mp_obj_print_exception(&mp_plat_print, nlr.ret_val); |
| 191 | MP_STATE_PORT(term_obj) = save_term; |
| 192 | } |
| 193 | |
| 194 | return -1; |
| 195 | } |
| 196 | |
| 197 | STATIC void dupterm_task_handler(os_event_t *evt) { |
| 198 | while (1) { |
| 199 | int c = call_dupterm_read(); |
| 200 | if (c < 0) { |
| 201 | break; |
| 202 | } |
| 203 | ringbuf_put(&input_buf, c); |
| 204 | } |
| 205 | mp_hal_signal_input(); |
| 206 | } |
| 207 | |
| 208 | STATIC os_event_t dupterm_evt_queue[4]; |
| 209 | |
| 210 | void dupterm_task_init() { |
| 211 | system_os_task(dupterm_task_handler, DUPTERM_TASK_ID, dupterm_evt_queue, MP_ARRAY_SIZE(dupterm_evt_queue)); |
| 212 | } |
| 213 | |
| 214 | void mp_hal_signal_dupterm_input(void) { |
| 215 | system_os_post(DUPTERM_TASK_ID, 0, 0); |
| 216 | } |