/** * Marlin 3D Printer Firmware * Copyright (c) 2020 MarlinFirmware [https://github.com/MarlinFirmware/Marlin] * * Based on Sprinter and grbl. * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm * * This program 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 program 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 . * */ #pragma once #include "../../../inc/MarlinConfigPre.h" #if ENABLED(EMERGENCY_PARSER) #include "../../../feature/e_parser.h" #endif #include "../../../core/serial_hook.h" #include #include /** * Generic RingBuffer * T type of the buffer array * S size of the buffer (must be power of 2) */ template class RingBuffer { public: RingBuffer() { index_read = index_write = 0; } uint32_t available() volatile { return index_write - index_read; } uint32_t free() volatile { return buffer_size - available(); } bool empty() volatile { return index_read == index_write; } bool full() volatile { return available() == buffer_size; } void clear() volatile { index_read = index_write = 0; } bool peek(T *value) volatile { if (value == 0 || available() == 0) return false; *value = buffer[mask(index_read)]; return true; } int read() volatile { if (empty()) return -1; return buffer[mask(index_read++)]; } bool write(T value) volatile { if (full()) return false; buffer[mask(index_write++)] = value; return true; } private: uint32_t mask(uint32_t val) volatile { return buffer_mask & val; } static const uint32_t buffer_size = S; static const uint32_t buffer_mask = buffer_size - 1; volatile T buffer[buffer_size]; volatile uint32_t index_write; volatile uint32_t index_read; }; struct HalSerial { HalSerial() { host_connected = true; } void begin(int32_t) {} void end() {} int peek() { uint8_t value; return receive_buffer.peek(&value) ? value : -1; } int read() { return receive_buffer.read(); } size_t write(char c) { if (!host_connected) return 0; while (!transmit_buffer.free()); return transmit_buffer.write(c); } bool connected() { return host_connected; } uint16_t available() { return (uint16_t)receive_buffer.available(); } void flush() { receive_buffer.clear(); } uint8_t availableForWrite() { return transmit_buffer.free() > 255 ? 255 : (uint8_t)transmit_buffer.free(); } void flushTX() { if (host_connected) while (transmit_buffer.available()) { /* nada */ } } volatile RingBuffer receive_buffer; volatile RingBuffer transmit_buffer; volatile bool host_connected; }; typedef Serial1Class MSerialT;