diff options
Diffstat (limited to 'arduino/libraries/SoftwareSerial')
6 files changed, 573 insertions, 0 deletions
diff --git a/arduino/libraries/SoftwareSerial/SoftwareSerial.cpp b/arduino/libraries/SoftwareSerial/SoftwareSerial.cpp new file mode 100755 index 0000000..cd32092 --- /dev/null +++ b/arduino/libraries/SoftwareSerial/SoftwareSerial.cpp @@ -0,0 +1,329 @@ +/* + SoftwareSerial.cpp - library for Arduino Primo + Copyright (c) 2016 Arduino. All rights reserved. + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Lesser General Public + License as published by the Free Software Foundation; either + version 2.1 of the License, or (at your option) any later version. + + This library is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + Lesser General Public License for more details. + + You should have received a copy of the GNU Lesser General Public + License along with this library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + */ + +#include <Arduino.h> +#include <SoftwareSerial.h> +#include <variant.h> +#include <WInterrupts.h> + +SoftwareSerial *SoftwareSerial::active_object = 0; +char SoftwareSerial::_receive_buffer[_SS_MAX_RX_BUFF]; +volatile uint8_t SoftwareSerial::_receive_buffer_tail = 0; +volatile uint8_t SoftwareSerial::_receive_buffer_head = 0; + + +SoftwareSerial::SoftwareSerial(uint8_t receivePin, uint8_t transmitPin, bool inverse_logic /* = false */) : + _rx_delay_centering(0), + _rx_delay_intrabit(0), + _rx_delay_stopbit(0), + _tx_delay(0), + _buffer_overflow(false), + _inverse_logic(inverse_logic) +{ + _receivePin = receivePin; + _transmitPin = transmitPin; +} + + +SoftwareSerial::~SoftwareSerial() +{ + end(); +} + +void SoftwareSerial::begin(long speed) + { + setTX(_transmitPin); + setRX(_receivePin); + // Precalculate the various delays + //Calculate the distance between bit in micro seconds + uint32_t bit_delay = (float(1)/speed)*1000000; + + _tx_delay = bit_delay; + + //Wait 1/2 bit - 2 micro seconds (time for interrupt to be served) + _rx_delay_centering = (bit_delay/2) - 2; + //Wait 1 bit - 2 micro seconds (time in each loop iteration) + _rx_delay_intrabit = bit_delay - 1;//2 + //Wait 1 bit (the stop one) + _rx_delay_stopbit = bit_delay; + + + delayMicroseconds(_tx_delay); + + listen(); +} + +bool SoftwareSerial::listen() +{ + if (!_rx_delay_stopbit) + return false; + + if (active_object != this) + { + if (active_object) + active_object->stopListening(); + + _buffer_overflow = false; + _receive_buffer_head = _receive_buffer_tail = 0; + active_object = this; + + if(_inverse_logic) + //Start bit high + _intMask = attachInterrupt(_receivePin, handle_interrupt, RISING); + else + //Start bit low + _intMask = attachInterrupt(_receivePin, handle_interrupt, FALLING); + + return true; + } + return false; +} + +bool SoftwareSerial::stopListening() +{ + if (active_object == this) + { + detachInterrupt(_receivePin); + active_object = NULL; + return true; + } + return false; +} + +void SoftwareSerial::end() +{ + stopListening(); +} + +int SoftwareSerial::read() +{ + if (!isListening()){ + return -1;} + + + // Empty buffer? + if (_receive_buffer_head == _receive_buffer_tail){ + return -1;} + + // Read from "head" + uint8_t d = _receive_buffer[_receive_buffer_head]; // grab next byte + _receive_buffer_head = (_receive_buffer_head + 1) % _SS_MAX_RX_BUFF; + return d; +} + +int SoftwareSerial::available() +{ + if (!isListening()) + return 0; + + return (_receive_buffer_tail + _SS_MAX_RX_BUFF - _receive_buffer_head) % _SS_MAX_RX_BUFF; +} + +size_t SoftwareSerial::write(uint8_t b) +{ + if (_tx_delay == 0) { + setWriteError(); + return 0; + } + + // By declaring these as local variables, the compiler will put them + // in registers _before_ disabling interrupts and entering the + // critical timing sections below, which makes it a lot easier to + // verify the cycle timings + volatile uint32_t* reg = _transmitPortRegister; + uint32_t reg_mask = _transmitBitMask; + uint32_t inv_mask = ~_transmitBitMask; + bool inv = _inverse_logic; + uint16_t delay = _tx_delay; + + if (inv) + b = ~b; + // turn off interrupts for a clean txmit + NRF_GPIOTE->INTENCLR = _intMask; + // Write the start bit + if (inv) + *reg |= reg_mask; + else + *reg &= inv_mask; + + delayMicroseconds(delay); + + + // Write each of the 8 bits + for (uint8_t i = 8; i > 0; --i) + { + if (b & 1) // choose bit + *reg |= reg_mask; // send 1 + else + *reg &= inv_mask; // send 0 + + delayMicroseconds(delay); + b >>= 1; + } + + // restore pin to natural state + if (inv) + *reg &= inv_mask; + else + *reg |= reg_mask; + + NRF_GPIOTE->INTENSET = _intMask; + + delayMicroseconds(delay); + + return 1; +} + +void SoftwareSerial::flush() +{ + if (!isListening()) + return; + + NRF_GPIOTE->INTENCLR = _intMask; + + _receive_buffer_head = _receive_buffer_tail = 0; + + NRF_GPIOTE->INTENSET = _intMask; +} + +int SoftwareSerial::peek() +{ + if (!isListening()) + return -1; + + // Empty buffer? + if (_receive_buffer_head == _receive_buffer_tail) + return -1; + + // Read from "head" + return _receive_buffer[_receive_buffer_head]; +} + + +//private methods + +void SoftwareSerial::recv() +{ + uint8_t d = 0; + + // If RX line is high, then we don't see any start bit + // so interrupt is probably not for us + if (_inverse_logic ? rx_pin_read() : !rx_pin_read()) + { + + NRF_GPIOTE->INTENCLR = _intMask; + + // Wait approximately 1/2 of a bit width to "center" the sample + delayMicroseconds(_rx_delay_centering); + + // Read each of the 8 bits + for (uint8_t i=8; i > 0; --i) + { + + delayMicroseconds(_rx_delay_intrabit); + // nRF52 needs another delay less than 1 uSec to be better synchronized + // with the highest baud rates + __ASM volatile ( + " NOP\n\t" + " NOP\n" + " NOP\n" + " NOP\n" + " NOP\n" + " NOP\n" + " NOP\n" + " NOP\n" + " NOP\n" + " NOP\n" + " NOP\n" + " NOP\n" + " NOP\n" + " NOP\n" + " NOP\n" + " NOP\n" + ); + + d >>= 1; + + if (rx_pin_read()){ + d |= 0x80; + } + + } + if (_inverse_logic){ + d = ~d; + } + + // if buffer full, set the overflow flag and return + uint8_t next = (_receive_buffer_tail + 1) % _SS_MAX_RX_BUFF; + if (next != _receive_buffer_head) + { + // save new data in buffer: tail points to where byte goes + _receive_buffer[_receive_buffer_tail] = d; // save new byte + _receive_buffer_tail = next; + } + else + { + _buffer_overflow = true; + } + + // skip the stop bit + delayMicroseconds(_rx_delay_stopbit); + + NRF_GPIOTE->INTENSET = _intMask; + } +} + +uint32_t SoftwareSerial::rx_pin_read() +{ + return *_receivePortRegister & digitalPinToBitMask(_receivePin); +} + +/* static */ +inline void SoftwareSerial::handle_interrupt() +{ + if (active_object) + { + active_object->recv(); + } +} + +void SoftwareSerial::setTX(uint8_t tx) +{ + // First write, then set output. If we do this the other way around, + // the pin would be output low for a short while before switching to + // output hihg. Now, it is input with pullup for a short while, which + // is fine. With inverse logic, either order is fine. + digitalWrite(tx, _inverse_logic ? LOW : HIGH); + pinMode(tx, OUTPUT); + _transmitBitMask = digitalPinToBitMask(tx); + NRF_GPIO_Type * port = digitalPinToPort(tx); + _transmitPortRegister = portOutputRegister(port); +} + +void SoftwareSerial::setRX(uint8_t rx) +{ + pinMode(rx, INPUT); + if (!_inverse_logic) + digitalWrite(rx, HIGH); // pullup for normal logic! + _receivePin = rx; + _receiveBitMask = digitalPinToBitMask(rx); + NRF_GPIO_Type * port = digitalPinToPort(rx); + _receivePortRegister = portInputRegister(port); +} diff --git a/arduino/libraries/SoftwareSerial/SoftwareSerial.h b/arduino/libraries/SoftwareSerial/SoftwareSerial.h new file mode 100755 index 0000000..ae1fc08 --- /dev/null +++ b/arduino/libraries/SoftwareSerial/SoftwareSerial.h @@ -0,0 +1,94 @@ +/* + SoftwareSerial.cpp - library for Arduino Primo + Copyright (c) 2016 Arduino. All rights reserved. + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Lesser General Public + License as published by the Free Software Foundation; either + version 2.1 of the License, or (at your option) any later version. + + This library is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + Lesser General Public License for more details. + + You should have received a copy of the GNU Lesser General Public + License along with this library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + */ + +#ifndef SoftwareSerial_h +#define SoftwareSerial_h + +#include <inttypes.h> +#include <Stream.h> +#include <variant.h> + +/****************************************************************************** +* Definitions +******************************************************************************/ + +#define _SS_MAX_RX_BUFF 64 // RX buffer size + +class SoftwareSerial : public Stream +{ +private: + // per object data + uint8_t _transmitPin; + uint8_t _receivePin; + uint32_t _receiveBitMask; + volatile uint32_t* _receivePortRegister; + uint32_t _transmitBitMask; + volatile uint32_t* _transmitPortRegister; + volatile uint32_t _intMask; + + // Expressed as 4-cycle delays (must never be 0!) + uint16_t _rx_delay_centering; + uint16_t _rx_delay_intrabit; + uint16_t _rx_delay_stopbit; + uint16_t _tx_delay; + + uint16_t _buffer_overflow:1; + uint16_t _inverse_logic:1; + + // static data + static char _receive_buffer[_SS_MAX_RX_BUFF]; + static volatile uint8_t _receive_buffer_tail; + static volatile uint8_t _receive_buffer_head; + static SoftwareSerial *active_object; + + // private methods + void recv() __attribute__((__always_inline__)); + uint32_t rx_pin_read(); + void tx_pin_write(uint8_t pin_state) __attribute__((__always_inline__)); + void setTX(uint8_t transmitPin); + void setRX(uint8_t receivePin); + void setRxIntMsk(bool enable) __attribute__((__always_inline__)); + + +public: + // public methods + SoftwareSerial(uint8_t receivePin, uint8_t transmitPin, bool inverse_logic = false); + ~SoftwareSerial(); + void begin(long speed); + bool listen(); + void end(); + bool isListening() { return this == active_object; } + bool stopListening(); + bool overflow() { bool ret = _buffer_overflow; if (ret) _buffer_overflow = false; return ret; } + int peek(); + + virtual size_t write(uint8_t byte); + virtual int read(); + virtual int available(); + virtual void flush(); + operator bool() { return true; } + + using Print::write; + + // public only for easy access by interrupt handlers + static inline void handle_interrupt() __attribute__((__always_inline__)); +}; + +#endif diff --git a/arduino/libraries/SoftwareSerial/examples/SoftwareSerialExample/SoftwareSerialExample.ino b/arduino/libraries/SoftwareSerial/examples/SoftwareSerialExample/SoftwareSerialExample.ino new file mode 100755 index 0000000..bf852bf --- /dev/null +++ b/arduino/libraries/SoftwareSerial/examples/SoftwareSerialExample/SoftwareSerialExample.ino @@ -0,0 +1,40 @@ +/* + Software serial multiple serial test + + Receives from the hardware serial, sends to software serial. + Receives from software serial, sends to hardware serial. + + The circuit: + * RX is digital pin A0 (connect to TX of other device) + * TX is digital pin A1 (connect to RX of other device) + + This example code is in the public domain. + + */ + +#include <SoftwareSerial.h> + +SoftwareSerial mySerial(A0, A1); // RX, TX + + +void setup() +{ + // Open serial communications and wait for port to open: + Serial.begin(9600); + while ( !Serial ) delay(10); // for nrf52840 with native usb + + Serial.println("Goodnight moon!"); + + // set the data rate for the SoftwareSerial port + mySerial.begin(9600); + mySerial.println("Hello, world?"); +} + +void loop() // run over and over// +{ + if (mySerial.available()) + Serial.write(mySerial.read()); + + if (Serial.available()) + mySerial.write(Serial.read()); +} diff --git a/arduino/libraries/SoftwareSerial/examples/TwoPortReceive/TwoPortReceive.ino b/arduino/libraries/SoftwareSerial/examples/TwoPortReceive/TwoPortReceive.ino new file mode 100755 index 0000000..88c4284 --- /dev/null +++ b/arduino/libraries/SoftwareSerial/examples/TwoPortReceive/TwoPortReceive.ino @@ -0,0 +1,72 @@ +/* + Software serial multple serial test + + Receives from the two software serial ports, + sends to the hardware serial port. + + In order to listen on a software port, you call port.listen(). + When using two software serial ports, you have to switch ports + by listen()ing on each one in turn. Pick a logical time to switch + ports, like the end of an expected transmission, or when the + buffer is empty. This example switches ports when there is nothing + more to read from a port + + The circuit: + Two devices which communicate serially are needed. + * First serial device's TX attached to digital pin A0, RX to pin A1 + * Second serial device's TX attached to digital pin A2, RX to pin A3 + + This example code is in the public domain. + + */ + +#include <SoftwareSerial.h> +// software serial #1: TX = digital pin A1, RX = digital pin A0 +SoftwareSerial portOne(A0, A1); + +// software serial #2: TX = digital pin A3, RX = digital pin A2 +SoftwareSerial portTwo(A2, A3); + +void setup() +{ + // Open serial communications and wait for port to open: + Serial.begin(115200); + while ( !Serial ) delay(10); // for nrf52840 with native usb + + // Start each software serial port + portOne.begin(9600); + portTwo.begin(9600); +} + +void loop() +{ + // By default, the last intialized port is listening. + // when you want to listen on a port, explicitly select it: + portOne.listen(); + + Serial.println("Data from port one:"); + // while there is data coming in, read it + // and send to the hardware serial port: + while (portOne.available() > 0) { + char inByte = portOne.read(); + Serial.write(inByte); + } + + // blank line to separate data from the two ports: + Serial.println(""); + + // Now listen on the second port + portTwo.listen(); + // while there is data coming in, read it + // and send to the hardware serial port: + + Serial.println("Data from port two:"); + while (portTwo.available() > 0) { + char inByte = portTwo.read(); + Serial.write(inByte); + } + + // blank line to separate data from the two ports: + Serial.println(); +} + diff --git a/arduino/libraries/SoftwareSerial/keywords.txt b/arduino/libraries/SoftwareSerial/keywords.txt new file mode 100755 index 0000000..89a7e4b --- /dev/null +++ b/arduino/libraries/SoftwareSerial/keywords.txt @@ -0,0 +1,29 @@ +####################################### +# Syntax Coloring Map for SoftwareSerial +####################################### + +####################################### +# Datatypes (KEYWORD1) +####################################### + +SoftwareSerial KEYWORD1 + +####################################### +# Methods and Functions (KEYWORD2) +####################################### + +begin KEYWORD2 +end KEYWORD2 +read KEYWORD2 +write KEYWORD2 +available KEYWORD2 +isListening KEYWORD2 +overflow KEYWORD2 +flush KEYWORD2 +listen KEYWORD2 +peek KEYWORD2 + +####################################### +# Constants (LITERAL1) +####################################### + diff --git a/arduino/libraries/SoftwareSerial/library.properties b/arduino/libraries/SoftwareSerial/library.properties new file mode 100755 index 0000000..9a6a10e --- /dev/null +++ b/arduino/libraries/SoftwareSerial/library.properties @@ -0,0 +1,9 @@ +name=SoftwareSerial +version=1.0.0 +author=Arduino +maintainer=Arduino <swdev@arduino.org> +sentence=Enables serial communication on digital pins. +paragraph=The SoftwareSerial library has been developed to allow serial communication on any digital pin of the board, using software to replicate the functionality of the hardware UART. It is possible to have multiple software serial ports with speeds up to 115200 bps. +category=Communication +url=http://arduino.cc/en/Reference/SoftwareSerial +architectures=nrf52 |