/* * This file is free software: you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This file 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 General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program. If not, see <http://www.gnu.org/licenses/>. */ /* soft serial receive implementation, based on pulse width inputs */ #include "SoftSerial.h" #include <stdio.h> SoftSerial::SoftSerial(uint32_t _baudrate, serial_config _config) : baudrate(_baudrate), config(_config), half_bit((1000000U / baudrate)/2) { switch (config) { case SERIAL_CONFIG_8N1: case SERIAL_CONFIG_8N1I: data_width = 8; byte_width = 10; stop_mask = 0x200; break; case SERIAL_CONFIG_8E2I: data_width = 9; byte_width = 12; stop_mask = 0xC00; break; } } /* process a pulse made up of a width of values at high voltage followed by a width at low voltage */ bool SoftSerial::process_pulse(uint32_t width_high, uint32_t width_low, uint8_t &byte) { // convert to bit widths, allowing for a half bit error uint16_t bits_high = ((width_high+half_bit)*baudrate) / 1000000; uint16_t bits_low = ((width_low+half_bit)*baudrate) / 1000000; byte_timestamp_us = timestamp_us; timestamp_us += (width_high + width_low); if (bits_high == 0 || bits_low == 0) { // invalid data goto reset; } if (bits_high >= byte_width) { // if we have a start bit and a stop bit then we can have at // most 9 bits in high state for data. The rest must be idle // bits bits_high = byte_width-1; } if (state.bit_ofs == 0) { // we are in idle state, waiting for first low bit. swallow // the high bits bits_high = 0; } state.byte |= ((1U<<bits_high)-1) << state.bit_ofs; state.bit_ofs += bits_high + bits_low; if (state.bit_ofs >= byte_width) { // check start bit if ((state.byte & 1) != 0) { goto reset; } // check stop bits if ((state.byte & stop_mask) != stop_mask) { goto reset; } if (config == SERIAL_CONFIG_8E2I) { // check parity if (__builtin_parity((state.byte>>1)&0xFF) != (state.byte&0x200)>>9) { goto reset; } } byte = ((state.byte>>1) & 0xFF); state.byte >>= byte_width; state.bit_ofs -= byte_width; if (state.bit_ofs > byte_width) { state.byte = 0; state.bit_ofs = bits_low; } // swallow idle bits while (state.bit_ofs > 0 && (state.byte & 1)) { state.bit_ofs--; state.byte >>= 1; } return true; } return false; reset: state.byte = 0; state.bit_ofs = 0; return false; }