/* * Marlin 3D Printer Firmware * Copyright (C) 2016 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 <http://www.gnu.org/licenses/>. * */ #include "Marlin.h" #if ENABLED(EXPERIMENTAL_I2CBUS) #include "twibus.h" #include <Wire.h> TWIBus::TWIBus() { Wire.begin(); // We use no address so we will join the BUS as the master this->reset(); } void TWIBus::reset() { this->addr = 0; this->buffer_s = 0; this->buffer[0] = 0x00; } void TWIBus::address(uint8_t addr) { this->addr = addr; #if ENABLED(DEBUG_TWIBUS) debug(PSTR("sendto"), this->addr); #endif } void TWIBus::addbyte(char c) { if (buffer_s >= sizeof(this->buffer)) return; this->buffer[this->buffer_s++] = c; #if ENABLED(DEBUG_TWIBUS) debug(PSTR("addbyte"), this->buffer[this->buffer_s - 1]); #endif } void TWIBus::send() { if (!this->addr) return; #if ENABLED(DEBUG_TWIBUS) debug(PSTR("send()")); #endif Wire.beginTransmission(this->addr); Wire.write(this->buffer, this->buffer_s); Wire.endTransmission(); // Reset the buffer after sending the data this->reset(); } void TWIBus::reqbytes(uint8_t bytes) { if (!this->addr) return; #if ENABLED(DEBUG_TWIBUS) debug(PSTR("reqbytes"), bytes); #endif millis_t t = millis() + this->timeout; Wire.requestFrom(this->addr, bytes); // requestFrom() is a blocking function while (Wire.available() < bytes) { if (ELAPSED(millis(), t)) break; else continue; } SERIAL_ECHO_START; SERIAL_ECHOPAIR("i2c-reply: from:", this->addr); SERIAL_ECHOPAIR(" bytes:", Wire.available()); SERIAL_ECHOPGM (" data:"); // Protect against buffer overflows if the number of received bytes // is less than the number of requested bytes uint8_t wba = Wire.available(); for (int i = 0; i < wba; i++) SERIAL_CHAR(Wire.read()); SERIAL_EOL; // Reset the buffer after sending the data this->reset(); } #if ENABLED(DEBUG_TWIBUS) void TWIBus::debug(const char func[], int32_t val/*=-1*/) { if (DEBUGGING(INFO)) { SERIAL_ECHOPGM("TWIBus::"); serialprintPGM(func); if (val >= 0) SERIAL_ECHOPAIR(": ", val); SERIAL_EOL; } } #endif #endif //EXPERIMENTAL_I2CBUS