@@ -34,6 +34,58 @@ | |||
#include "mk20dx128.h" | |||
#include <inttypes.h> | |||
// uncomment to enable 9 bit formats | |||
//#define SERIAL_9BIT_SUPPORT | |||
#define SERIAL_7E1 0x02 | |||
#define SERIAL_7O1 0x03 | |||
#define SERIAL_8N1 0x00 | |||
#define SERIAL_8N2 0x04 | |||
#define SERIAL_8E1 0x06 | |||
#define SERIAL_8O1 0x07 | |||
#define SERIAL_7E1_RXINV 0x12 | |||
#define SERIAL_7O1_RXINV 0x13 | |||
#define SERIAL_8N1_RXINV 0x10 | |||
#define SERIAL_8N2_RXINV 0x14 | |||
#define SERIAL_8E1_RXINV 0x16 | |||
#define SERIAL_8O1_RXINV 0x17 | |||
#define SERIAL_7E1_TXINV 0x22 | |||
#define SERIAL_7O1_TXINV 0x23 | |||
#define SERIAL_8N1_TXINV 0x20 | |||
#define SERIAL_8N2_TXINV 0x24 | |||
#define SERIAL_8E1_TXINV 0x26 | |||
#define SERIAL_8O1_TXINV 0x27 | |||
#define SERIAL_7E1_RXINV_TXINV 0x32 | |||
#define SERIAL_7O1_RXINV_TXINV 0x33 | |||
#define SERIAL_8N1_RXINV_TXINV 0x30 | |||
#define SERIAL_8N2_RXINV_TXINV 0x34 | |||
#define SERIAL_8E1_RXINV_TXINV 0x36 | |||
#define SERIAL_8O1_RXINV_TXINV 0x37 | |||
#ifdef SERIAL_9BIT_SUPPORT | |||
#define SERIAL_9N1 0x84 | |||
#define SERIAL_9E1 0x8E | |||
#define SERIAL_9O1 0x8F | |||
#define SERIAL_9N1_RXINV 0x94 | |||
#define SERIAL_9E1_RXINV 0x9E | |||
#define SERIAL_9O1_RXINV 0x9F | |||
#define SERIAL_9N1_TXINV 0xA4 | |||
#define SERIAL_9E1_TXINV 0xAE | |||
#define SERIAL_9O1_TXINV 0xAF | |||
#define SERIAL_9N1_RXINV_TXINV 0xB4 | |||
#define SERIAL_9E1_RXINV_TXINV 0xBE | |||
#define SERIAL_9O1_RXINV_TXINV 0xBF | |||
#endif | |||
// bit0: parity, 0=even, 1=odd | |||
// bit1: parity, 0=disable, 1=enable | |||
// bit2: mode, 1=9bit, 0=8bit | |||
// bit3: mode10: 1=10bit, 0=8bit | |||
// bit4: rxinv, 0=normal, 1=inverted | |||
// bit5: txinv, 0=normal, 1=inverted | |||
// bit6: unused | |||
// bit7: actual data goes into 9th bit | |||
#define BAUD2DIV(baud) (((F_CPU * 2) + ((baud) >> 1)) / (baud)) | |||
#define BAUD2DIV3(baud) (((F_BUS * 2) + ((baud) >> 1)) / (baud)) | |||
@@ -43,8 +95,9 @@ | |||
extern "C" { | |||
#endif | |||
void serial_begin(uint32_t divisor); | |||
void serial_format(uint32_t format); | |||
void serial_end(void); | |||
void serial_putchar(uint8_t c); | |||
void serial_putchar(uint32_t c); | |||
void serial_write(const void *buf, unsigned int count); | |||
void serial_flush(void); | |||
int serial_available(void); | |||
@@ -57,8 +110,9 @@ void serial_phex16(uint32_t n); | |||
void serial_phex32(uint32_t n); | |||
void serial2_begin(uint32_t divisor); | |||
void serial2_format(uint32_t format); | |||
void serial2_end(void); | |||
void serial2_putchar(uint8_t c); | |||
void serial2_putchar(uint32_t c); | |||
void serial2_write(const void *buf, unsigned int count); | |||
void serial2_flush(void); | |||
int serial2_available(void); | |||
@@ -67,8 +121,9 @@ int serial2_peek(void); | |||
void serial2_clear(void); | |||
void serial3_begin(uint32_t divisor); | |||
void serial3_format(uint32_t format); | |||
void serial3_end(void); | |||
void serial3_putchar(uint8_t c); | |||
void serial3_putchar(uint32_t c); | |||
void serial3_write(const void *buf, unsigned int count); | |||
void serial3_flush(void); | |||
int serial3_available(void); | |||
@@ -88,13 +143,16 @@ void serial3_clear(void); | |||
class HardwareSerial : public Stream | |||
{ | |||
public: | |||
void begin(uint32_t baud) { serial_begin(BAUD2DIV(baud)); } | |||
void end(void) { serial_end(); } | |||
virtual void begin(uint32_t baud) { serial_begin(BAUD2DIV(baud)); } | |||
virtual void begin(uint32_t baud, uint32_t format) { | |||
serial_begin(BAUD2DIV(baud)); | |||
serial_format(format); } | |||
virtual void end(void) { serial_end(); } | |||
virtual int available(void) { return serial_available(); } | |||
virtual int peek(void) { return serial_peek(); } | |||
virtual int read(void) { return serial_getchar(); } | |||
virtual void flush(void) { serial_flush(); } | |||
void clear(void) { serial_clear(); } | |||
virtual void clear(void) { serial_clear(); } | |||
virtual size_t write(uint8_t c) { serial_putchar(c); return 1; } | |||
size_t write(unsigned long n) { return write((uint8_t)n); } | |||
size_t write(long n) { return write((uint8_t)n); } | |||
@@ -105,19 +163,23 @@ public: | |||
size_t write(const char *str) { size_t len = strlen(str); | |||
serial_write((const uint8_t *)str, len); | |||
return len; } | |||
size_t write9bit(uint32_t c) { serial_putchar(c); return 1; } | |||
}; | |||
extern HardwareSerial Serial1; | |||
class HardwareSerial2 : public HardwareSerial | |||
{ | |||
public: | |||
void begin(uint32_t baud) { serial2_begin(BAUD2DIV(baud)); } | |||
void end(void) { serial2_end(); } | |||
virtual void begin(uint32_t baud) { serial2_begin(BAUD2DIV(baud)); } | |||
virtual void begin(uint32_t baud, uint32_t format) { | |||
serial2_begin(BAUD2DIV(baud)); | |||
serial2_format(format); } | |||
virtual void end(void) { serial2_end(); } | |||
virtual int available(void) { return serial2_available(); } | |||
virtual int peek(void) { return serial2_peek(); } | |||
virtual int read(void) { return serial2_getchar(); } | |||
virtual void flush(void) { serial2_flush(); } | |||
void clear(void) { serial2_clear(); } | |||
virtual void clear(void) { serial2_clear(); } | |||
virtual size_t write(uint8_t c) { serial2_putchar(c); return 1; } | |||
size_t write(unsigned long n) { return write((uint8_t)n); } | |||
size_t write(long n) { return write((uint8_t)n); } | |||
@@ -128,19 +190,23 @@ public: | |||
size_t write(const char *str) { size_t len = strlen(str); | |||
serial2_write((const uint8_t *)str, len); | |||
return len; } | |||
size_t write9bit(uint32_t c) { serial2_putchar(c); return 1; } | |||
}; | |||
extern HardwareSerial2 Serial2; | |||
class HardwareSerial3 : public HardwareSerial | |||
{ | |||
public: | |||
void begin(uint32_t baud) { serial3_begin(BAUD2DIV3(baud)); } | |||
void end(void) { serial3_end(); } | |||
virtual void begin(uint32_t baud) { serial3_begin(BAUD2DIV3(baud)); } | |||
virtual void begin(uint32_t baud, uint32_t format) { | |||
serial3_begin(BAUD2DIV3(baud)); | |||
serial3_format(format); } | |||
virtual void end(void) { serial3_end(); } | |||
virtual int available(void) { return serial3_available(); } | |||
virtual int peek(void) { return serial3_peek(); } | |||
virtual int read(void) { return serial3_getchar(); } | |||
virtual void flush(void) { serial3_flush(); } | |||
void clear(void) { serial3_clear(); } | |||
virtual void clear(void) { serial3_clear(); } | |||
virtual size_t write(uint8_t c) { serial3_putchar(c); return 1; } | |||
size_t write(unsigned long n) { return write((uint8_t)n); } | |||
size_t write(long n) { return write((uint8_t)n); } | |||
@@ -151,6 +217,7 @@ public: | |||
size_t write(const char *str) { size_t len = strlen(str); | |||
serial3_write((const uint8_t *)str, len); | |||
return len; } | |||
size_t write9bit(uint32_t c) { serial3_putchar(c); return 1; } | |||
}; | |||
extern HardwareSerial3 Serial3; | |||
@@ -45,8 +45,16 @@ | |||
// changes not recommended below this point.... | |||
//////////////////////////////////////////////////////////////// | |||
static volatile uint8_t tx_buffer[TX_BUFFER_SIZE]; | |||
static volatile uint8_t rx_buffer[RX_BUFFER_SIZE]; | |||
#ifdef SERIAL_9BIT_SUPPORT | |||
static uint8_t use9Bits = 0; | |||
#define BUFTYPE uint16_t | |||
#else | |||
#define BUFTYPE uint8_t | |||
#define use9Bits 0 | |||
#endif | |||
static volatile BUFTYPE tx_buffer[TX_BUFFER_SIZE]; | |||
static volatile BUFTYPE rx_buffer[RX_BUFFER_SIZE]; | |||
static volatile uint8_t transmitting = 0; | |||
#if TX_BUFFER_SIZE > 255 | |||
static volatile uint16_t tx_buffer_head = 0; | |||
@@ -94,6 +102,29 @@ void serial_begin(uint32_t divisor) | |||
NVIC_ENABLE_IRQ(IRQ_UART0_STATUS); | |||
} | |||
void serial_format(uint32_t format) | |||
{ | |||
uint8_t c; | |||
c = UART0_C1; | |||
c = (c & ~0x13) | (format & 0x03); // configure parity | |||
if (format & 0x04) c |= 0x10; // 9 bits (might include parity) | |||
UART0_C1 = c; | |||
if ((format & 0x0F) == 0x04) UART0_C3 |= 0x40; // 8N2 is 9 bit with 9th bit always 1 | |||
c = UART0_S2 & ~0x10; | |||
if (format & 0x10) c |= 0x10; // rx invert | |||
UART0_S2 = c; | |||
c = UART0_C3 & ~0x10; | |||
if (format & 0x20) c |= 0x10; // tx invert | |||
UART0_C3 = c; | |||
#ifdef SERIAL_9BIT_SUPPORT | |||
c = UART0_C4 & 0x1F; | |||
if (format & 0x08) c |= 0x20; // 9 bit mode with parity (requires 10 bits) | |||
UART0_C4 = c; | |||
use9Bits = format & 0x80; | |||
#endif | |||
} | |||
void serial_end(void) | |||
{ | |||
if (!(SIM_SCGC4 & SIM_SCGC4_UART0)) return; | |||
@@ -129,7 +160,7 @@ static int get_nvic_execution_priority(void) | |||
} | |||
void serial_putchar(uint8_t c) | |||
void serial_putchar(uint32_t c) | |||
{ | |||
uint32_t head; | |||
@@ -245,7 +276,7 @@ void serial_clear(void) | |||
void uart0_status_isr(void) | |||
{ | |||
uint32_t head, newhead, tail; | |||
uint32_t head, newhead, tail, n; | |||
uint8_t avail, c; | |||
if (UART0_S1 & (UART_S1_RDRF | UART_S1_IDLE)) { | |||
@@ -275,12 +306,13 @@ void uart0_status_isr(void) | |||
head = rx_buffer_head; | |||
tail = rx_buffer_tail; | |||
do { | |||
c = UART0_D; | |||
n = UART0_D; | |||
if (use9Bits && (UART0_C3 & 0x80)) n |= 0x100; | |||
newhead = head + 1; | |||
if (newhead >= RX_BUFFER_SIZE) newhead = 0; | |||
if (newhead != tail) { | |||
head = newhead; | |||
rx_buffer[head] = c; | |||
rx_buffer[head] = n; | |||
} | |||
} while (--avail > 0); | |||
rx_buffer_head = head; | |||
@@ -294,7 +326,9 @@ void uart0_status_isr(void) | |||
if (tail == head) break; | |||
if (++tail >= TX_BUFFER_SIZE) tail = 0; | |||
avail = UART0_S1; | |||
UART0_D = tx_buffer[tail]; | |||
n = tx_buffer[tail]; | |||
if (use9Bits) UART0_C3 = (UART0_C3 & ~0x40) | ((n & 0x100) >> 2); | |||
UART0_D = n; | |||
} while (UART0_TCFIFO < 8); | |||
tx_buffer_tail = tail; | |||
if (UART0_S1 & UART_S1_TDRE) UART0_C2 = C2_TX_COMPLETING; |
@@ -35,8 +35,16 @@ | |||
// UART0 and UART1 are clocked by F_CPU, UART2 is clocked by F_BUS | |||
// UART0 has 8 byte fifo, UART1 and UART2 have 1 byte buffer | |||
#ifdef SERIAL_9BIT_SUPPORT | |||
static uint8_t use9Bits = 0; | |||
#define BUFTYPE uint16_t | |||
#else | |||
#define BUFTYPE uint8_t | |||
#define use9Bits 0 | |||
#endif | |||
#define TX_BUFFER_SIZE 40 | |||
static volatile uint8_t tx_buffer[TX_BUFFER_SIZE]; | |||
static volatile BUFTYPE tx_buffer[TX_BUFFER_SIZE]; | |||
static volatile uint8_t transmitting = 0; | |||
#if TX_BUFFER_SIZE > 255 | |||
static volatile uint16_t tx_buffer_head = 0; | |||
@@ -47,7 +55,7 @@ static volatile uint8_t tx_buffer_tail = 0; | |||
#endif | |||
#define RX_BUFFER_SIZE 64 | |||
static volatile uint8_t rx_buffer[RX_BUFFER_SIZE]; | |||
static volatile BUFTYPE rx_buffer[RX_BUFFER_SIZE]; | |||
static volatile uint8_t rx_buffer_head = 0; | |||
static volatile uint8_t rx_buffer_tail = 0; | |||
@@ -76,6 +84,35 @@ void serial2_begin(uint32_t divisor) | |||
NVIC_ENABLE_IRQ(IRQ_UART1_STATUS); | |||
} | |||
void serial2_format(uint32_t format) | |||
{ | |||
uint8_t c; | |||
c = UART1_C1; | |||
c = (c & ~0x13) | (format & 0x03); // configure parity | |||
if (format & 0x04) c |= 0x10; // 9 bits (might include parity) | |||
UART1_C1 = c; | |||
if ((format & 0x0F) == 0x04) UART1_C3 |= 0x40; // 8N2 is 9 bit with 9th bit always 1 | |||
c = UART1_S2 & ~0x10; | |||
if (format & 0x10) c |= 0x10; // rx invert | |||
UART1_S2 = c; | |||
c = UART1_C3 & ~0x10; | |||
if (format & 0x20) c |= 0x10; // tx invert | |||
UART1_C3 = c; | |||
#ifdef SERIAL_9BIT_SUPPORT | |||
c = UART1_C4 & 0x1F; | |||
if (format & 0x08) c |= 0x20; // 9 bit mode with parity (requires 10 bits) | |||
UART1_C4 = c; | |||
use9Bits = format & 0x80; | |||
#endif | |||
// UART1_C1.0 = parity, 0=even, 1=odd | |||
// UART1_C1.1 = parity, 0=disable, 1=enable | |||
// UART1_C1.4 = mode, 1=9bit, 0=8bit | |||
// UART1_C4.5 = mode, 1=10bit, 0=8bit | |||
// UART1_C3.4 = txinv, 0=normal, 1=inverted | |||
// UART1_S2.4 = rxinv, 0=normal, 1=inverted | |||
} | |||
void serial2_end(void) | |||
{ | |||
if (!(SIM_SCGC4 & SIM_SCGC4_UART1)) return; | |||
@@ -88,7 +125,7 @@ void serial2_end(void) | |||
rx_buffer_tail = 0; | |||
} | |||
void serial2_putchar(uint8_t c) | |||
void serial2_putchar(uint32_t c) | |||
{ | |||
uint32_t head; | |||
@@ -165,17 +202,18 @@ void serial2_clear(void) | |||
void uart1_status_isr(void) | |||
{ | |||
uint32_t head, tail; | |||
uint32_t head, tail, n; | |||
uint8_t c; | |||
//digitalWriteFast(4, HIGH); | |||
if (UART1_S1 & UART_S1_RDRF) { | |||
//digitalWriteFast(5, HIGH); | |||
c = UART1_D; | |||
n = UART1_D; | |||
if (use9Bits && (UART1_C3 & 0x80)) n |= 0x100; | |||
head = rx_buffer_head + 1; | |||
if (head >= RX_BUFFER_SIZE) head = 0; | |||
if (head != rx_buffer_tail) { | |||
rx_buffer[head] = c; | |||
rx_buffer[head] = n; | |||
rx_buffer_head = head; | |||
} | |||
//digitalWriteFast(5, LOW); | |||
@@ -189,7 +227,9 @@ void uart1_status_isr(void) | |||
UART1_C2 = C2_TX_COMPLETING; | |||
} else { | |||
if (++tail >= TX_BUFFER_SIZE) tail = 0; | |||
UART1_D = tx_buffer[tail]; | |||
n = tx_buffer[tail]; | |||
if (use9Bits) UART1_C3 = (UART1_C3 & ~0x40) | ((n & 0x100) >> 2); | |||
UART1_D = n; | |||
tx_buffer_tail = tail; | |||
} | |||
//digitalWriteFast(5, LOW); |
@@ -35,8 +35,16 @@ | |||
// UART0 and UART1 are clocked by F_CPU, UART2 is clocked by F_BUS | |||
// UART0 has 8 byte fifo, UART1 and UART2 have 1 byte buffer | |||
#ifdef SERIAL_9BIT_SUPPORT | |||
static uint8_t use9Bits = 0; | |||
#define BUFTYPE uint16_t | |||
#else | |||
#define BUFTYPE uint8_t | |||
#define use9Bits 0 | |||
#endif | |||
#define TX_BUFFER_SIZE 40 | |||
static volatile uint8_t tx_buffer[TX_BUFFER_SIZE]; | |||
static volatile BUFTYPE tx_buffer[TX_BUFFER_SIZE]; | |||
static volatile uint8_t transmitting = 0; | |||
#if TX_BUFFER_SIZE > 255 | |||
static volatile uint16_t tx_buffer_head = 0; | |||
@@ -47,7 +55,7 @@ static volatile uint8_t tx_buffer_tail = 0; | |||
#endif | |||
#define RX_BUFFER_SIZE 64 | |||
static volatile uint8_t rx_buffer[RX_BUFFER_SIZE]; | |||
static volatile BUFTYPE rx_buffer[RX_BUFFER_SIZE]; | |||
static volatile uint8_t rx_buffer_head = 0; | |||
static volatile uint8_t rx_buffer_tail = 0; | |||
@@ -76,6 +84,29 @@ void serial3_begin(uint32_t divisor) | |||
NVIC_ENABLE_IRQ(IRQ_UART2_STATUS); | |||
} | |||
void serial3_format(uint32_t format) | |||
{ | |||
uint8_t c; | |||
c = UART2_C1; | |||
c = (c & ~0x13) | (format & 0x03); // configure parity | |||
if (format & 0x04) c |= 0x10; // 9 bits (might include parity) | |||
UART2_C1 = c; | |||
if ((format & 0x0F) == 0x04) UART2_C3 |= 0x40; // 8N2 is 9 bit with 9th bit always 1 | |||
c = UART2_S2 & ~0x10; | |||
if (format & 0x10) c |= 0x10; // rx invert | |||
UART2_S2 = c; | |||
c = UART2_C3 & ~0x10; | |||
if (format & 0x20) c |= 0x10; // tx invert | |||
UART2_C3 = c; | |||
#ifdef SERIAL_9BIT_SUPPORT | |||
c = UART2_C4 & 0x1F; | |||
if (format & 0x08) c |= 0x20; // 9 bit mode with parity (requires 10 bits) | |||
UART2_C4 = c; | |||
use9Bits = format & 0x80; | |||
#endif | |||
} | |||
void serial3_end(void) | |||
{ | |||
if (!(SIM_SCGC4 & SIM_SCGC4_UART2)) return; | |||
@@ -88,7 +119,7 @@ void serial3_end(void) | |||
rx_buffer_tail = 0; | |||
} | |||
void serial3_putchar(uint8_t c) | |||
void serial3_putchar(uint32_t c) | |||
{ | |||
uint32_t head; | |||
@@ -165,17 +196,18 @@ void serial3_clear(void) | |||
void uart2_status_isr(void) | |||
{ | |||
uint32_t head, tail; | |||
uint32_t head, tail, n; | |||
uint8_t c; | |||
//digitalWriteFast(4, HIGH); | |||
if (UART2_S1 & UART_S1_RDRF) { | |||
//digitalWriteFast(5, HIGH); | |||
c = UART2_D; | |||
n = UART2_D; | |||
if (use9Bits && (UART2_C3 & 0x80)) n |= 0x100; | |||
head = rx_buffer_head + 1; | |||
if (head >= RX_BUFFER_SIZE) head = 0; | |||
if (head != rx_buffer_tail) { | |||
rx_buffer[head] = c; | |||
rx_buffer[head] = n; | |||
rx_buffer_head = head; | |||
} | |||
//digitalWriteFast(5, LOW); | |||
@@ -189,7 +221,9 @@ void uart2_status_isr(void) | |||
UART2_C2 = C2_TX_COMPLETING; | |||
} else { | |||
if (++tail >= TX_BUFFER_SIZE) tail = 0; | |||
UART2_D = tx_buffer[tail]; | |||
n = tx_buffer[tail]; | |||
if (use9Bits) UART2_C3 = (UART2_C3 & ~0x40) | ((n & 0x100) >> 2); | |||
UART2_D = n; | |||
tx_buffer_tail = tail; | |||
} | |||
//digitalWriteFast(5, LOW); |