An RTOS-friendly Serial interface Its primary benefit is that it never hogs the CPU. An amusing alternative to the traditional ring-bufferd interrupt-serviced systems, it uses short mbed-rtos queues to buffer characters to and from the UART, and a thread to service the transmitter. Short interrupt service routines enqueue received characters and signal the transmit thread when the transmitter is available. WARNING: Do not create RTOS-Serial objects before the RTOS is running! Put them inside your main() block or another function, not in the global initialization.
Dependents: Test_RDM880_rfid_reader
rtos_serial.cpp@10:d5adca63e94a, 2013-10-23 (annotated)
- Committer:
- altasoul
- Date:
- Wed Oct 23 21:06:59 2013 +0000
- Revision:
- 10:d5adca63e94a
- Parent:
- 8:3644d12758da
- Child:
- 11:bc067b42f8e0
Adapt to mbed-src as it exists now
Who changed what in which revision?
User | Revision | Line number | New contents of line |
---|---|---|---|
altasoul | 0:0547c8bf304f | 1 | /* |
altasoul | 0:0547c8bf304f | 2 | * Copyright (c) 2013 Tom Soulanille |
altasoul | 0:0547c8bf304f | 3 | * |
altasoul | 0:0547c8bf304f | 4 | * Permission is hereby granted, free of charge, to any person obtaining a copy |
altasoul | 0:0547c8bf304f | 5 | * of this software and associated documentation files (the "Software"), to deal |
altasoul | 0:0547c8bf304f | 6 | * in the Software without restriction, including without limitation the rights |
altasoul | 0:0547c8bf304f | 7 | * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
altasoul | 0:0547c8bf304f | 8 | * copies of the Software, and to permit persons to whom the Software is |
altasoul | 0:0547c8bf304f | 9 | * furnished to do so, subject to the following conditions: |
altasoul | 0:0547c8bf304f | 10 | * |
altasoul | 0:0547c8bf304f | 11 | * The above copyright notice and this permission notice shall be included in |
altasoul | 0:0547c8bf304f | 12 | * all copies or substantial portions of the Software. |
altasoul | 0:0547c8bf304f | 13 | * |
altasoul | 0:0547c8bf304f | 14 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
altasoul | 0:0547c8bf304f | 15 | * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
altasoul | 0:0547c8bf304f | 16 | * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
altasoul | 0:0547c8bf304f | 17 | * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
altasoul | 0:0547c8bf304f | 18 | * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
altasoul | 0:0547c8bf304f | 19 | * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |
altasoul | 0:0547c8bf304f | 20 | * SOFTWARE. |
altasoul | 0:0547c8bf304f | 21 | */ |
altasoul | 0:0547c8bf304f | 22 | |
altasoul | 10:d5adca63e94a | 23 | /* TODO: |
altasoul | 10:d5adca63e94a | 24 | * - assert rx overrun failure |
altasoul | 10:d5adca63e94a | 25 | * - fix rx overrun failure |
altasoul | 10:d5adca63e94a | 26 | * - size the tx thread stack |
altasoul | 10:d5adca63e94a | 27 | * - make an rx buffer thread |
altasoul | 10:d5adca63e94a | 28 | * - implement readable() |
altasoul | 10:d5adca63e94a | 29 | * - implement writeable() somehow |
altasoul | 10:d5adca63e94a | 30 | */ |
altasoul | 0:0547c8bf304f | 31 | #include "rtos_serial.h" |
altasoul | 0:0547c8bf304f | 32 | |
altasoul | 0:0547c8bf304f | 33 | RTOS_Serial::RTOS_Serial(PinName tx, PinName rx, const char *name) |
altasoul | 0:0547c8bf304f | 34 | : Serial(tx, rx, name) |
altasoul | 0:0547c8bf304f | 35 | { |
altasoul | 5:5d388d1d7987 | 36 | uart_number = get_index(); |
altasoul | 0:0547c8bf304f | 37 | const PinName leds[] = {LED1,LED2,LED3,LED4}; |
altasoul | 5:5d388d1d7987 | 38 | ledp = new DigitalOut(leds[uart_number]); |
altasoul | 8:3644d12758da | 39 | #ifdef RTOS_SERIAL_TX_THREAD |
altasoul | 0:0547c8bf304f | 40 | tx_emitter_threadp = new Thread(tx_emitter, (void *) this); |
altasoul | 8:3644d12758da | 41 | #endif |
altasoul | 10:d5adca63e94a | 42 | // rx_isr_pFP = attach(this, &RTOS_Serial::rx_isr, RxIrq); |
altasoul | 10:d5adca63e94a | 43 | // tx_isr_pFP = attach(this, &RTOS_Serial::tx_isr, TxIrq); |
altasoul | 10:d5adca63e94a | 44 | attach(this, &RTOS_Serial::rx_isr, RxIrq); |
altasoul | 10:d5adca63e94a | 45 | attach(this, &RTOS_Serial::tx_isr, TxIrq); |
altasoul | 8:3644d12758da | 46 | #ifdef RTOS_SERIAL_TX_THREAD |
altasoul | 2:891773cc33fd | 47 | tx_emitter_threadp->signal_set(0x01); // "prime the pump" of the tx-ready signals |
altasoul | 8:3644d12758da | 48 | #else |
altasoul | 8:3644d12758da | 49 | Serial::_putc('*'); //DEBUG |
altasoul | 8:3644d12758da | 50 | #endif |
altasoul | 1:5a66fddad7c4 | 51 | } |
altasoul | 0:0547c8bf304f | 52 | |
altasoul | 10:d5adca63e94a | 53 | #if 0 |
altasoul | 6:438a6c0acbd4 | 54 | RTOS_Serial::~RTOS_Serial() { |
altasoul | 10:d5adca63e94a | 55 | std::printf("[destroying RTOS_Serial 0x%x]", this); std::fflush(stdout); |
altasoul | 10:d5adca63e94a | 56 | bool b; |
altasoul | 10:d5adca63e94a | 57 | osDelay(200); //DEBUG |
altasoul | 10:d5adca63e94a | 58 | #if 0 |
altasoul | 10:d5adca63e94a | 59 | std::printf("[remove rx handler 0x%x", rx_isr_pFP); std::fflush(stdout); |
altasoul | 10:d5adca63e94a | 60 | // returns false as expected: b = remove_handler(NULL, RxIrq); |
altasoul | 10:d5adca63e94a | 61 | b = remove_handler(rx_isr_pFP, RxIrq); |
altasoul | 10:d5adca63e94a | 62 | std::printf("returned %d]", b); std::fflush(stdout); |
altasoul | 10:d5adca63e94a | 63 | std::printf("[remove tx handler 0x%x", tx_isr_pFP); std::fflush(stdout); |
altasoul | 10:d5adca63e94a | 64 | b = remove_handler(tx_isr_pFP, TxIrq); |
altasoul | 10:d5adca63e94a | 65 | std::printf("returned %d]", b); std::fflush(stdout); |
altasoul | 10:d5adca63e94a | 66 | #endif |
altasoul | 8:3644d12758da | 67 | #ifdef RTOS_SERIAL_TX_THREAD |
altasoul | 6:438a6c0acbd4 | 68 | tx_emitter_threadp->terminate(); |
altasoul | 10:d5adca63e94a | 69 | std::printf("[tx_emitter_thread 0x%x terminated]", tx_emitter_threadp); std::fflush(stdout); |
altasoul | 6:438a6c0acbd4 | 70 | delete tx_emitter_threadp; |
altasoul | 10:d5adca63e94a | 71 | std::printf("[tx_emitter_threadp deleted]"); std::fflush(stdout); |
altasoul | 8:3644d12758da | 72 | #endif |
altasoul | 6:438a6c0acbd4 | 73 | } |
altasoul | 10:d5adca63e94a | 74 | #endif |
altasoul | 6:438a6c0acbd4 | 75 | |
altasoul | 0:0547c8bf304f | 76 | int RTOS_Serial::get_index() { return _serial.index; } |
altasoul | 0:0547c8bf304f | 77 | |
altasoul | 0:0547c8bf304f | 78 | int RTOS_Serial::putc(int c) { |
altasoul | 2:891773cc33fd | 79 | //return Serial::putc(c); //DEBUG |
altasoul | 0:0547c8bf304f | 80 | //if (tx_q.put((int *)c, osWaitForever) == osOK) return c; else return EOF; |
altasoul | 0:0547c8bf304f | 81 | int status; |
altasoul | 0:0547c8bf304f | 82 | if ( (status = tx_q.put((int *)c, osWaitForever)) == osOK) return c; else { |
altasoul | 5:5d388d1d7987 | 83 | std::printf("\r\nRTOS_Serial::tx_q.put() returned %d\r\n", status); |
altasoul | 0:0547c8bf304f | 84 | return EOF; |
altasoul | 0:0547c8bf304f | 85 | } |
altasoul | 0:0547c8bf304f | 86 | } |
altasoul | 0:0547c8bf304f | 87 | |
altasoul | 2:891773cc33fd | 88 | int RTOS_Serial::puts(const char *s) { |
altasoul | 2:891773cc33fd | 89 | int rv = 0; |
altasoul | 2:891773cc33fd | 90 | while (*s) { |
altasoul | 2:891773cc33fd | 91 | if (putc(*s++) == EOF) { |
altasoul | 2:891773cc33fd | 92 | rv = EOF; |
altasoul | 2:891773cc33fd | 93 | break; |
altasoul | 2:891773cc33fd | 94 | } else { |
altasoul | 2:891773cc33fd | 95 | rv++; |
altasoul | 2:891773cc33fd | 96 | } |
altasoul | 2:891773cc33fd | 97 | } |
altasoul | 2:891773cc33fd | 98 | return rv; |
altasoul | 2:891773cc33fd | 99 | } |
altasoul | 2:891773cc33fd | 100 | |
altasoul | 0:0547c8bf304f | 101 | int RTOS_Serial::parent_putc(int c) { |
altasoul | 7:dd892347b524 | 102 | return Serial::_putc(c); |
altasoul | 0:0547c8bf304f | 103 | } |
altasoul | 0:0547c8bf304f | 104 | |
altasoul | 2:891773cc33fd | 105 | int RTOS_Serial::getc() { |
altasoul | 4:c7113cd0ac4b | 106 | int rv; |
altasoul | 4:c7113cd0ac4b | 107 | //return Serial::getc(); //FIXME: stand-in, which fails if we use our RX ISR |
altasoul | 4:c7113cd0ac4b | 108 | osEvent evt = rx_q.get(); |
altasoul | 4:c7113cd0ac4b | 109 | if (evt.status == osEventMessage) { |
altasoul | 4:c7113cd0ac4b | 110 | rv = (int) evt.value.v; |
altasoul | 4:c7113cd0ac4b | 111 | } else { //FIXME: find appropriate error reporting if any |
altasoul | 5:5d388d1d7987 | 112 | std::printf("\r\nRTOS_Serial::getc() evt.status %d\n", evt.status); |
altasoul | 4:c7113cd0ac4b | 113 | rv = EOF; |
altasoul | 4:c7113cd0ac4b | 114 | } |
altasoul | 4:c7113cd0ac4b | 115 | return rv; |
altasoul | 2:891773cc33fd | 116 | } |
altasoul | 2:891773cc33fd | 117 | |
altasoul | 8:3644d12758da | 118 | void RTOS_Serial::rx_isr(){ |
altasoul | 10:d5adca63e94a | 119 | rx_q.put((int *) _getc()); // FIXME: won't this wedge on overrun? |
altasoul | 1:5a66fddad7c4 | 120 | } |
altasoul | 1:5a66fddad7c4 | 121 | |
altasoul | 8:3644d12758da | 122 | void RTOS_Serial::tx_isr(){ |
altasoul | 8:3644d12758da | 123 | #ifdef RTOS_SERIAL_TX_THREAD |
altasoul | 8:3644d12758da | 124 | tx_emitter_threadp->signal_set(0x1); |
altasoul | 8:3644d12758da | 125 | #else |
altasoul | 8:3644d12758da | 126 | osEvent evt; |
altasoul | 10:d5adca63e94a | 127 | evt = tx_q.get(); // BUG: This will wait forever in ISR! |
altasoul | 8:3644d12758da | 128 | if (evt.status == osEventMessage) { |
altasoul | 8:3644d12758da | 129 | *ledp = 1; |
altasoul | 8:3644d12758da | 130 | parent_putc(evt.value.v); |
altasoul | 8:3644d12758da | 131 | *ledp = 0; |
altasoul | 8:3644d12758da | 132 | } /*else { |
altasoul | 8:3644d12758da | 133 | std::printf("\r\nRTOS_Serial::tx_emitter() evt.status %d\n", evt.status); |
altasoul | 8:3644d12758da | 134 | }*/ |
altasoul | 8:3644d12758da | 135 | #endif |
altasoul | 2:891773cc33fd | 136 | } |
altasoul | 2:891773cc33fd | 137 | |
altasoul | 8:3644d12758da | 138 | #ifdef RTOS_SERIAL_TX_THREAD |
altasoul | 0:0547c8bf304f | 139 | // tx_emitter is a class method |
altasoul | 0:0547c8bf304f | 140 | void RTOS_Serial::tx_emitter(void const *argument){ |
altasoul | 0:0547c8bf304f | 141 | RTOS_Serial *sp = (RTOS_Serial *) argument; |
altasoul | 0:0547c8bf304f | 142 | osEvent evt; |
altasoul | 2:891773cc33fd | 143 | //osStatus status; |
altasoul | 0:0547c8bf304f | 144 | while(true){ |
altasoul | 0:0547c8bf304f | 145 | evt = sp->tx_q.get(); |
altasoul | 0:0547c8bf304f | 146 | if (evt.status == osEventMessage) { |
altasoul | 2:891773cc33fd | 147 | // There is no TX interrupt until the first byte is sent out the port, |
altasoul | 2:891773cc33fd | 148 | // so we use a timeout on the signal from the interrupt service routine |
altasoul | 2:891773cc33fd | 149 | // and just proceed to transmit the character. This should happen only |
altasoul | 2:891773cc33fd | 150 | // once at most, to "prime the pump", but the timeout provides some |
altasoul | 2:891773cc33fd | 151 | // safety in case something goes wrong. |
altasoul | 2:891773cc33fd | 152 | // A first signal is sent by the RTOS_Serial constructor when the thread |
altasoul | 2:891773cc33fd | 153 | // is created, so normally this will not come into effect. |
altasoul | 2:891773cc33fd | 154 | // DEBUG: timeout omitted to search for instabilities |
altasoul | 2:891773cc33fd | 155 | Thread::signal_wait(0x1/*, 10*/); //FIXME: base the timeout on the baud rate |
altasoul | 0:0547c8bf304f | 156 | *(sp->ledp) = 1; |
altasoul | 0:0547c8bf304f | 157 | sp->parent_putc(evt.value.v); |
altasoul | 0:0547c8bf304f | 158 | *(sp->ledp) = 0; |
altasoul | 0:0547c8bf304f | 159 | } else { |
altasoul | 5:5d388d1d7987 | 160 | std::printf("\r\nRTOS_Serial::tx_emitter() evt.status %d\n", evt.status); |
altasoul | 0:0547c8bf304f | 161 | } |
altasoul | 0:0547c8bf304f | 162 | *(sp->ledp) = 0; |
altasoul | 0:0547c8bf304f | 163 | } |
altasoul | 3:5865277b7710 | 164 | } |
altasoul | 8:3644d12758da | 165 | #endif |