Marco Burato
5 years ago
committed by
Scott Lahteine
16 changed files with 837 additions and 667 deletions
@ -1,290 +0,0 @@ |
|||||
/*
|
|
||||
anycubic_serial.cpp --- Support for Anycubic i3 Mega TFT serial connection |
|
||||
Created by Christian Hopp on 09.12.17. |
|
||||
|
|
||||
Original file: |
|
||||
HardwareSerial.cpp - Hardware serial library for Wiring |
|
||||
Copyright (c) 2006 Nicholas Zambetti. All right 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 |
|
||||
|
|
||||
Modified 23 November 2006 by David A. Mellis |
|
||||
Modified 28 September 2010 by Mark Sproul |
|
||||
Modified 14 August 2012 by Alarus |
|
||||
*/ |
|
||||
|
|
||||
#include "../../../../inc/MarlinConfig.h" |
|
||||
|
|
||||
#if ENABLED(ANYCUBIC_TFT_MODEL) |
|
||||
|
|
||||
#include <Arduino.h> |
|
||||
|
|
||||
// This next line disables the entire anycubic_serial.cpp,
|
|
||||
// to support AtTiny series and other chips without a UART
|
|
||||
#ifdef UBRR3H |
|
||||
|
|
||||
#include "anycubic_serial.h" |
|
||||
|
|
||||
#include <stdlib.h> |
|
||||
#include <stdio.h> |
|
||||
#include <string.h> |
|
||||
#include <inttypes.h> |
|
||||
#include "wiring_private.h" |
|
||||
|
|
||||
// Define constants and variables for buffering incoming serial data. We're
|
|
||||
// using a ring buffer (I think), in which head is the index of the location
|
|
||||
// to which to write the next incoming character and tail is the index of the
|
|
||||
// location from which to read.
|
|
||||
#if (RAMEND < 1000) |
|
||||
#define SERIAL_BUFFER_SIZE 64 |
|
||||
#else |
|
||||
#define SERIAL_BUFFER_SIZE 128 |
|
||||
#endif |
|
||||
|
|
||||
struct ring_buffer { |
|
||||
unsigned char buffer[SERIAL_BUFFER_SIZE]; |
|
||||
volatile unsigned int head; |
|
||||
volatile unsigned int tail; |
|
||||
}; |
|
||||
|
|
||||
ring_buffer rx_buffer_ajg = { { 0 }, 0, 0 }; |
|
||||
ring_buffer tx_buffer_ajg = { { 0 }, 0, 0 }; |
|
||||
|
|
||||
inline void store_char(unsigned char c, ring_buffer *buffer) { |
|
||||
int i = (unsigned int)(buffer->head + 1) % SERIAL_BUFFER_SIZE; |
|
||||
|
|
||||
// if we should be storing the received character into the location
|
|
||||
// just before the tail (meaning that the head would advance to the
|
|
||||
// current location of the tail), we're about to overflow the buffer
|
|
||||
// and so we don't write the character or advance the head.
|
|
||||
if (i != buffer->tail) { |
|
||||
buffer->buffer[buffer->head] = c; |
|
||||
buffer->head = i; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
#if defined(USART3_RX_vect) && defined(UDR3) |
|
||||
void serialEvent3() __attribute__((weak)); |
|
||||
void serialEvent3() {} |
|
||||
#define serialEvent3_implemented |
|
||||
ISR(USART3_RX_vect) { |
|
||||
if (bit_is_clear(UCSR3A, UPE3)) { |
|
||||
unsigned char c = UDR3; |
|
||||
store_char(c, &rx_buffer_ajg); |
|
||||
} |
|
||||
else { |
|
||||
unsigned char c = UDR3; |
|
||||
} |
|
||||
} |
|
||||
#endif |
|
||||
|
|
||||
#ifdef USART3_UDRE_vect |
|
||||
|
|
||||
ISR(USART3_UDRE_vect) { |
|
||||
if (tx_buffer_ajg.head == tx_buffer_ajg.tail) { |
|
||||
// Buffer empty, so disable interrupts
|
|
||||
cbi(UCSR3B, UDRIE3); |
|
||||
} |
|
||||
else { |
|
||||
// There is more data in the output buffer. Send the next byte
|
|
||||
unsigned char c = tx_buffer_ajg.buffer[tx_buffer_ajg.tail]; |
|
||||
tx_buffer_ajg.tail = (tx_buffer_ajg.tail + 1) % SERIAL_BUFFER_SIZE; |
|
||||
|
|
||||
UDR3 = c; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
#endif |
|
||||
|
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
|
||||
|
|
||||
AnycubicSerialClass::AnycubicSerialClass(ring_buffer *rx_buffer, ring_buffer *tx_buffer, |
|
||||
volatile uint8_t *ubrrh, volatile uint8_t *ubrrl, |
|
||||
volatile uint8_t *ucsra, volatile uint8_t *ucsrb, |
|
||||
volatile uint8_t *ucsrc, volatile uint8_t *udr, |
|
||||
uint8_t rxen, uint8_t txen, uint8_t rxcie, uint8_t udrie, uint8_t u2x |
|
||||
) { |
|
||||
_rx_buffer = rx_buffer; |
|
||||
_tx_buffer = tx_buffer; |
|
||||
_ubrrh = ubrrh; |
|
||||
_ubrrl = ubrrl; |
|
||||
_ucsra = ucsra; |
|
||||
_ucsrb = ucsrb; |
|
||||
_ucsrc = ucsrc; |
|
||||
_udr = udr; |
|
||||
_rxen = rxen; |
|
||||
_txen = txen; |
|
||||
_rxcie = rxcie; |
|
||||
_udrie = udrie; |
|
||||
_u2x = u2x; |
|
||||
} |
|
||||
|
|
||||
// Public Methods //////////////////////////////////////////////////////////////
|
|
||||
|
|
||||
void AnycubicSerialClass::begin(unsigned long baud) { |
|
||||
uint16_t baud_setting; |
|
||||
bool use_u2x = true; |
|
||||
|
|
||||
#if F_CPU == 16000000UL |
|
||||
// hardcoded exception for compatibility with the bootloader shipped
|
|
||||
// with the Duemilanove and previous boards and the firmware on the 8U2
|
|
||||
// on the Uno and Mega 2560.
|
|
||||
if (baud == 57600) use_u2x = false; |
|
||||
#endif |
|
||||
|
|
||||
try_again: |
|
||||
|
|
||||
if (use_u2x) { |
|
||||
*_ucsra = 1 << _u2x; |
|
||||
baud_setting = (F_CPU / 4 / baud - 1) / 2; |
|
||||
} else { |
|
||||
*_ucsra = 0; |
|
||||
baud_setting = (F_CPU / 8 / baud - 1) / 2; |
|
||||
} |
|
||||
|
|
||||
if ((baud_setting > 4095) && use_u2x) { |
|
||||
use_u2x = false; |
|
||||
goto try_again; |
|
||||
} |
|
||||
|
|
||||
// assign the baud_setting, a.k.a. ubbr (USART Baud Rate Register)
|
|
||||
*_ubrrh = baud_setting >> 8; |
|
||||
*_ubrrl = baud_setting; |
|
||||
|
|
||||
transmitting = false; |
|
||||
|
|
||||
sbi(*_ucsrb, _rxen); |
|
||||
sbi(*_ucsrb, _txen); |
|
||||
sbi(*_ucsrb, _rxcie); |
|
||||
cbi(*_ucsrb, _udrie); |
|
||||
} |
|
||||
|
|
||||
void AnycubicSerialClass::begin(unsigned long baud, byte config) { |
|
||||
uint16_t baud_setting; |
|
||||
uint8_t current_config; |
|
||||
bool use_u2x = true; |
|
||||
|
|
||||
#if F_CPU == 16000000UL |
|
||||
// hardcoded exception for compatibility with the bootloader shipped
|
|
||||
// with the Duemilanove and previous boards and the firmware on the 8U2
|
|
||||
// on the Uno and Mega 2560.
|
|
||||
if (baud == 57600) use_u2x = false; |
|
||||
#endif |
|
||||
|
|
||||
try_again: |
|
||||
|
|
||||
if (use_u2x) { |
|
||||
*_ucsra = 1 << _u2x; |
|
||||
baud_setting = (F_CPU / 4 / baud - 1) / 2; |
|
||||
} |
|
||||
else { |
|
||||
*_ucsra = 0; |
|
||||
baud_setting = (F_CPU / 8 / baud - 1) / 2; |
|
||||
} |
|
||||
|
|
||||
if ((baud_setting > 4095) && use_u2x) { |
|
||||
use_u2x = false; |
|
||||
goto try_again; |
|
||||
} |
|
||||
|
|
||||
// assign the baud_setting, a.k.a. ubbr (USART Baud Rate Register)
|
|
||||
*_ubrrh = baud_setting >> 8; |
|
||||
*_ubrrl = baud_setting; |
|
||||
|
|
||||
//set the data bits, parity, and stop bits
|
|
||||
#ifdef __AVR_ATmega8__ |
|
||||
config |= 0x80; // select UCSRC register (shared with UBRRH)
|
|
||||
#endif |
|
||||
*_ucsrc = config; |
|
||||
|
|
||||
sbi(*_ucsrb, _rxen); |
|
||||
sbi(*_ucsrb, _txen); |
|
||||
sbi(*_ucsrb, _rxcie); |
|
||||
cbi(*_ucsrb, _udrie); |
|
||||
} |
|
||||
|
|
||||
void AnycubicSerialClass::end() { |
|
||||
// wait for transmission of outgoing data
|
|
||||
while (_tx_buffer->head != _tx_buffer->tail) |
|
||||
; |
|
||||
|
|
||||
cbi(*_ucsrb, _rxen); |
|
||||
cbi(*_ucsrb, _txen); |
|
||||
cbi(*_ucsrb, _rxcie); |
|
||||
cbi(*_ucsrb, _udrie); |
|
||||
|
|
||||
// clear any received data
|
|
||||
_rx_buffer->head = _rx_buffer->tail; |
|
||||
} |
|
||||
|
|
||||
int AnycubicSerialClass::available(void) { |
|
||||
return (int)(SERIAL_BUFFER_SIZE + _rx_buffer->head - _rx_buffer->tail) % SERIAL_BUFFER_SIZE; |
|
||||
} |
|
||||
|
|
||||
int AnycubicSerialClass::peek(void) { |
|
||||
if (_rx_buffer->head == _rx_buffer->tail) { |
|
||||
return -1; |
|
||||
} else { |
|
||||
return _rx_buffer->buffer[_rx_buffer->tail]; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
int AnycubicSerialClass::read(void) { |
|
||||
// if the head isn't ahead of the tail, we don't have any characters
|
|
||||
if (_rx_buffer->head == _rx_buffer->tail) { |
|
||||
return -1; |
|
||||
} else { |
|
||||
unsigned char c = _rx_buffer->buffer[_rx_buffer->tail]; |
|
||||
_rx_buffer->tail = (unsigned int)(_rx_buffer->tail + 1) % SERIAL_BUFFER_SIZE; |
|
||||
return c; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
void AnycubicSerialClass::flush() { |
|
||||
// UDR is kept full while the buffer is not empty, so TXC triggers when EMPTY && SENT
|
|
||||
while (transmitting && ! (*_ucsra & _BV(TXC0))); |
|
||||
transmitting = false; |
|
||||
} |
|
||||
|
|
||||
size_t AnycubicSerialClass::write(uint8_t c) { |
|
||||
int i = (_tx_buffer->head + 1) % SERIAL_BUFFER_SIZE; |
|
||||
|
|
||||
// If the output buffer is full, there's nothing for it other than to
|
|
||||
// wait for the interrupt handler to empty it a bit
|
|
||||
// ???: return 0 here instead?
|
|
||||
while (i == _tx_buffer->tail) |
|
||||
; |
|
||||
|
|
||||
_tx_buffer->buffer[_tx_buffer->head] = c; |
|
||||
_tx_buffer->head = i; |
|
||||
|
|
||||
sbi(*_ucsrb, _udrie); |
|
||||
// clear the TXC bit -- "can be cleared by writing a one to its bit location"
|
|
||||
transmitting = true; |
|
||||
sbi(*_ucsra, TXC0); |
|
||||
|
|
||||
return 1; |
|
||||
} |
|
||||
|
|
||||
AnycubicSerialClass::operator bool() { |
|
||||
return true; |
|
||||
} |
|
||||
|
|
||||
// Preinstantiate Objects //////////////////////////////////////////////////////
|
|
||||
|
|
||||
AnycubicSerialClass AnycubicSerial(&rx_buffer_ajg, &tx_buffer_ajg, &UBRR3H, &UBRR3L, &UCSR3A, &UCSR3B, &UCSR3C, &UDR3, RXEN3, TXEN3, RXCIE3, UDRIE3, U2X3); |
|
||||
|
|
||||
#endif // UBRR3H
|
|
||||
#endif // ANYCUBIC_TFT_MODEL
|
|
@ -1,145 +0,0 @@ |
|||||
/*
|
|
||||
anycubic_serial.h --- Support for Anycubic i3 Mega TFT serial connection |
|
||||
Created by Christian Hopp on 09.12.17. |
|
||||
|
|
||||
Original file: |
|
||||
|
|
||||
HardwareSerial.h - Hardware serial library for Wiring |
|
||||
Copyright (c) 2006 Nicholas Zambetti. All right 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 |
|
||||
|
|
||||
Modified 28 September 2010 by Mark Sproul |
|
||||
Modified 14 August 2012 by Alarus |
|
||||
*/ |
|
||||
#pragma once |
|
||||
|
|
||||
#include <inttypes.h> |
|
||||
#include <avr/pgmspace.h> |
|
||||
|
|
||||
#include "Stream.h" |
|
||||
|
|
||||
#define FORCE_INLINE __attribute__((always_inline)) inline |
|
||||
|
|
||||
struct ring_buffer; |
|
||||
|
|
||||
class AnycubicSerialClass : public Stream { |
|
||||
private: |
|
||||
ring_buffer *_rx_buffer; |
|
||||
ring_buffer *_tx_buffer; |
|
||||
volatile uint8_t *_ubrrh; |
|
||||
volatile uint8_t *_ubrrl; |
|
||||
volatile uint8_t *_ucsra; |
|
||||
volatile uint8_t *_ucsrb; |
|
||||
volatile uint8_t *_ucsrc; |
|
||||
volatile uint8_t *_udr; |
|
||||
uint8_t _rxen; |
|
||||
uint8_t _txen; |
|
||||
uint8_t _rxcie; |
|
||||
uint8_t _udrie; |
|
||||
uint8_t _u2x; |
|
||||
bool transmitting; |
|
||||
public: |
|
||||
AnycubicSerialClass(ring_buffer *rx_buffer, ring_buffer *tx_buffer, |
|
||||
volatile uint8_t *ubrrh, volatile uint8_t *ubrrl, |
|
||||
volatile uint8_t *ucsra, volatile uint8_t *ucsrb, |
|
||||
volatile uint8_t *ucsrc, volatile uint8_t *udr, |
|
||||
uint8_t rxen, uint8_t txen, uint8_t rxcie, uint8_t udrie, uint8_t u2x |
|
||||
); |
|
||||
void begin(unsigned long); |
|
||||
void begin(unsigned long, uint8_t); |
|
||||
void end(); |
|
||||
virtual int available(void); |
|
||||
virtual int peek(void); |
|
||||
virtual int read(void); |
|
||||
virtual void flush(void); |
|
||||
virtual size_t write(uint8_t); |
|
||||
inline size_t write(unsigned long n) { return write((uint8_t)n); } |
|
||||
inline size_t write(long n) { return write((uint8_t)n); } |
|
||||
inline size_t write(unsigned int n) { return write((uint8_t)n); } |
|
||||
inline size_t write(int n) { return write((uint8_t)n); } |
|
||||
using Print::write; // pull in write(str) and write(buf, size) from Print
|
|
||||
operator bool(); |
|
||||
}; |
|
||||
|
|
||||
// Define config for Serial.begin(baud, config);
|
|
||||
#define SERIAL_5N1 0x00 |
|
||||
#define SERIAL_6N1 0x02 |
|
||||
#define SERIAL_7N1 0x04 |
|
||||
#define SERIAL_8N1 0x06 |
|
||||
#define SERIAL_5N2 0x08 |
|
||||
#define SERIAL_6N2 0x0A |
|
||||
#define SERIAL_7N2 0x0C |
|
||||
#define SERIAL_8N2 0x0E |
|
||||
#define SERIAL_5E1 0x20 |
|
||||
#define SERIAL_6E1 0x22 |
|
||||
#define SERIAL_7E1 0x24 |
|
||||
#define SERIAL_8E1 0x26 |
|
||||
#define SERIAL_5E2 0x28 |
|
||||
#define SERIAL_6E2 0x2A |
|
||||
#define SERIAL_7E2 0x2C |
|
||||
#define SERIAL_8E2 0x2E |
|
||||
#define SERIAL_5O1 0x30 |
|
||||
#define SERIAL_6O1 0x32 |
|
||||
#define SERIAL_7O1 0x34 |
|
||||
#define SERIAL_8O1 0x36 |
|
||||
#define SERIAL_5O2 0x38 |
|
||||
#define SERIAL_6O2 0x3A |
|
||||
#define SERIAL_7O2 0x3C |
|
||||
#define SERIAL_8O2 0x3E |
|
||||
|
|
||||
extern void serialEventRun(void) __attribute__((weak)); |
|
||||
|
|
||||
#define ANYCUBIC_SERIAL_PROTOCOL(x) (AnycubicSerial.print(x)) |
|
||||
#define ANYCUBIC_SERIAL_PROTOCOL_F(x,y) (AnycubicSerial.print(x, y)) |
|
||||
#define ANYCUBIC_SERIAL_PROTOCOLPGM(x) (AnycubicSerialprintPGM(PSTR(x))) |
|
||||
#define ANYCUBIC_SERIAL_(x) (AnycubicSerial.print(x), AnycubicSerial.write('\n')) |
|
||||
#define ANYCUBIC_SERIAL_PROTOCOLLN(x) (AnycubicSerial.print(x), AnycubicSerial.write('\r'), AnycubicSerial.write('\n')) |
|
||||
#define ANYCUBIC_SERIAL_PROTOCOLLNPGM(x) (AnycubicSerialprintPGM(PSTR(x)), AnycubicSerial.write('\r'), AnycubicSerial.write('\n')) |
|
||||
|
|
||||
#define ANYCUBIC_SERIAL_START() (AnycubicSerial.write('\r'), AnycubicSerial.write('\n')) |
|
||||
#define ANYCUBIC_SERIAL_CMD_SEND(x) (AnycubicSerialprintPGM(PSTR(x)), AnycubicSerial.write('\r'), AnycubicSerial.write('\n')) |
|
||||
#define ANYCUBIC_SERIAL_ENTER() (AnycubicSerial.write('\r'), AnycubicSerial.write('\n')) |
|
||||
#define ANYCUBIC_SERIAL_SPACE() (AnycubicSerial.write(' ')) |
|
||||
|
|
||||
const char newErr[] PROGMEM = "ERR "; |
|
||||
const char newSucc[] PROGMEM = "OK"; |
|
||||
|
|
||||
#define ANYCUBIC_SERIAL_ERROR_START (AnycubicSerialprintPGM(newErr)) |
|
||||
#define ANYCUBIC_SERIAL_ERROR(x) ANYCUBIC_SERIAL_PROTOCOL(x) |
|
||||
#define ANYCUBIC_SERIAL_ERRORPGM(x) ANYCUBIC_SERIAL_PROTOCOLPGM(x) |
|
||||
#define ANYCUBIC_SERIAL_ERRORLN(x) ANYCUBIC_SERIAL_PROTOCOLLN(x) |
|
||||
#define ANYCUBIC_SERIAL_ERRORLNPGM(x) ANYCUBIC_SERIAL_PROTOCOLLNPGM(x) |
|
||||
|
|
||||
//##define ANYCUBIC_SERIAL_ECHO_START (AnycubicSerialprintPGM(newSucc))
|
|
||||
#define ANYCUBIC_SERIAL_ECHOLN(x) ANYCUBIC_SERIAL_PROTOCOLLN(x) |
|
||||
#define ANYCUBIC_SERIAL_SUCC_START (AnycubicSerialprintPGM(newSucc)) |
|
||||
#define ANYCUBIC_SERIAL_ECHOPAIR(name,value) (serial_echopair_P(PSTR(name),(value))) |
|
||||
#define ANYCUBIC_SERIAL_ECHOPGM(x) ANYCUBIC_SERIAL_PROTOCOLPGM(x) |
|
||||
#define ANYCUBIC_SERIAL_ECHO(x) ANYCUBIC_SERIAL_PROTOCOL(x) |
|
||||
|
|
||||
#ifdef UBRR3H |
|
||||
|
|
||||
extern AnycubicSerialClass AnycubicSerial; |
|
||||
|
|
||||
FORCE_INLINE void AnycubicSerialprintPGM(const char *str) { |
|
||||
char ch = pgm_read_byte(str); |
|
||||
while (ch) { |
|
||||
AnycubicSerial.write(ch); |
|
||||
ch = pgm_read_byte(++str); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
#endif |
|
@ -0,0 +1,536 @@ |
|||||
|
/**
|
||||
|
* 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 <http://www.gnu.org/licenses/>.
|
||||
|
* |
||||
|
*/ |
||||
|
|
||||
|
/**
|
||||
|
* extui_anycubic_chiron_lcd.cpp |
||||
|
* |
||||
|
* Anycubic Chiron TFT support for Marlin |
||||
|
*/ |
||||
|
|
||||
|
#include "../inc/MarlinConfigPre.h" |
||||
|
|
||||
|
#if ENABLED(ANYCUBIC_LCD_CHIRON) |
||||
|
|
||||
|
#include "extui/ui_api.h" |
||||
|
|
||||
|
#if ENABLED(AUTO_BED_LEVELING_BILINEAR) |
||||
|
#if GRID_MAX_POINTS_X != 5 || GRID_MAX_POINTS_Y != 5 |
||||
|
#error ANYCUBIC CHIRON LCD requires a 5x5 bed leveling grid (GRID_MAX_POINTS_X and GRID_MAX_POINTS_Y) |
||||
|
#endif |
||||
|
#else |
||||
|
#error ANYCUBIC CHIRON LCD requires AUTO_BED_LEVELING_BILINEAR enabled |
||||
|
#endif |
||||
|
|
||||
|
#if DISABLED(FILAMENT_RUNOUT_SENSOR) |
||||
|
#error ANYCUBIC CHIRON LCD requires FILAMENT_RUNOUT_SENSOR enabled |
||||
|
#endif |
||||
|
|
||||
|
#if ENABLED(POWER_LOSS_RECOVERY) |
||||
|
#error ANYCUBIC CHIRON LCD does not currently support POWER_LOSS_RECOVERY |
||||
|
#endif |
||||
|
|
||||
|
static bool is_auto_leveling = false; |
||||
|
static bool is_printing_from_sd = false; |
||||
|
static bool is_out_of_filament = false; |
||||
|
|
||||
|
static void sendNewLine(void) { |
||||
|
ANYCUBIC_LCD_SERIAL.write('\r'); |
||||
|
ANYCUBIC_LCD_SERIAL.write('\n'); |
||||
|
} |
||||
|
|
||||
|
static void send(const char *str) { |
||||
|
ANYCUBIC_LCD_SERIAL.print(str); |
||||
|
} |
||||
|
|
||||
|
static void sendLine(const char *str) { |
||||
|
send(str); |
||||
|
sendNewLine(); |
||||
|
} |
||||
|
|
||||
|
static void send_P(PGM_P str) { |
||||
|
while (const char c = pgm_read_byte(str++)) |
||||
|
ANYCUBIC_LCD_SERIAL.write(c); |
||||
|
} |
||||
|
|
||||
|
static void sendLine_P(PGM_P str) { |
||||
|
send_P(str); |
||||
|
sendNewLine(); |
||||
|
} |
||||
|
|
||||
|
static void sendValue_P(PGM_P prefix, int value) { |
||||
|
send_P(prefix); |
||||
|
ANYCUBIC_LCD_SERIAL.print(value); |
||||
|
} |
||||
|
|
||||
|
static void sendValue_P(PGM_P prefix, float value) { |
||||
|
send_P(prefix); |
||||
|
ANYCUBIC_LCD_SERIAL.print(value); |
||||
|
} |
||||
|
|
||||
|
static void sendValueLine_P(PGM_P prefix, int value) { |
||||
|
send_P(prefix); |
||||
|
ANYCUBIC_LCD_SERIAL.print(value); |
||||
|
sendNewLine(); |
||||
|
} |
||||
|
|
||||
|
static void sendValueLine_P(PGM_P prefix, float value) { |
||||
|
send_P(prefix); |
||||
|
ANYCUBIC_LCD_SERIAL.print(value); |
||||
|
sendNewLine(); |
||||
|
} |
||||
|
|
||||
|
static int parseIntArgument(const char *buffer, char letterId) { |
||||
|
char *p = strchr(buffer, letterId); |
||||
|
if (!p) |
||||
|
return -1; |
||||
|
return atoi(p+1); |
||||
|
} |
||||
|
|
||||
|
static float parseFloatArgument(const char *buffer, char letterId) { |
||||
|
char *p = strchr(buffer, letterId); |
||||
|
if (!p) |
||||
|
return NAN; |
||||
|
return strtof(p+1, nullptr); |
||||
|
} |
||||
|
|
||||
|
static int mmToHundredths(float x) { |
||||
|
// Round
|
||||
|
if (x >= 0) |
||||
|
x += 0.005f; |
||||
|
else |
||||
|
x -= 0.005f; |
||||
|
return (int)(x * 100.0f); |
||||
|
} |
||||
|
|
||||
|
static float hundredthsToMm(int x) { |
||||
|
return x / 100.0f; |
||||
|
} |
||||
|
|
||||
|
#define SEND_PGM(str) send_P(PSTR(str)) |
||||
|
#define SENDLINE_PGM(str) sendLine_P(PSTR(str)) |
||||
|
#define SENDVALUE_PGM(prefix, value) sendValue_P(PSTR(prefix), value) |
||||
|
#define SENDVALUELINE_PGM(prefix, value) sendValueLine_P(PSTR(prefix), value) |
||||
|
|
||||
|
namespace ExtUI { |
||||
|
|
||||
|
static void moveAxis(float delta, feedRate_t feedrate, axis_t axis) { |
||||
|
float pos = getAxisPosition_mm(axis); |
||||
|
pos += delta; |
||||
|
setAxisPosition_mm(pos, axis, feedrate); |
||||
|
} |
||||
|
|
||||
|
static void handleCmd(const char *rx) { |
||||
|
static FileList fileList; |
||||
|
static char selectedFileShortName[8+1+3+1]; |
||||
|
|
||||
|
if (rx[0] != 'A') { |
||||
|
SERIAL_ECHOPGM("Unexpected RX: "); |
||||
|
SERIAL_ECHOLN(rx); |
||||
|
|
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
const int cmd = atoi(&rx[1]); |
||||
|
|
||||
|
// Uncomment for debugging RX
|
||||
|
//if (cmd > 7 && cmd != 20) {
|
||||
|
// SERIAL_ECHOPGM("RX: ");
|
||||
|
// SERIAL_ECHOLN(rx);
|
||||
|
//}
|
||||
|
|
||||
|
switch (cmd) { |
||||
|
case 0: // Get Hotend Actual Temperature
|
||||
|
SENDVALUELINE_PGM("A0V ", (int)getActualTemp_celsius(E0)); |
||||
|
break; |
||||
|
case 1: // Get Hotend Target Temperature
|
||||
|
SENDVALUELINE_PGM("A1V ", (int)getTargetTemp_celsius(E0)); |
||||
|
break; |
||||
|
case 2: // Get Bed Actual Temperature
|
||||
|
SENDVALUELINE_PGM("A2V ", (int)getActualTemp_celsius(BED)); |
||||
|
break; |
||||
|
case 3: // Get Bed Target Temperature
|
||||
|
SENDVALUELINE_PGM("A3V ", (int)getTargetTemp_celsius(BED)); |
||||
|
break; |
||||
|
case 4: // Get Fan Speed
|
||||
|
SENDVALUELINE_PGM("A4V ", (int)getTargetFan_percent(FAN0)); |
||||
|
break; |
||||
|
case 5: // Get Current Coordinates
|
||||
|
SENDVALUE_PGM("A5V X: ", getAxisPosition_mm(X)); |
||||
|
SENDVALUE_PGM(" Y: ", getAxisPosition_mm(Y)); |
||||
|
SENDVALUE_PGM(" Z: ", getAxisPosition_mm(Z)); |
||||
|
sendNewLine(); |
||||
|
break; |
||||
|
case 6: // Get SD Card Print Status
|
||||
|
if (isPrintingFromMedia()) |
||||
|
SENDVALUELINE_PGM("A6V ", (int)getProgress_percent()); |
||||
|
else |
||||
|
SENDLINE_PGM("A6V ---"); |
||||
|
break; |
||||
|
case 7: // Get Printing Time
|
||||
|
if (isPrinting()) { |
||||
|
const int totalMinutes = getProgress_seconds_elapsed() / 60; |
||||
|
SENDVALUE_PGM("A7V ", (int)(totalMinutes/60)); |
||||
|
SENDVALUE_PGM(" H ", (int)(totalMinutes%60)); |
||||
|
SENDLINE_PGM(" M"); |
||||
|
} else { |
||||
|
SENDLINE_PGM("A7V 999:999"); |
||||
|
} |
||||
|
break; |
||||
|
case 8: // Get SD Card File List
|
||||
|
if (isMediaInserted()) { |
||||
|
const int startIndex = parseIntArgument(rx, 'S'); |
||||
|
SENDLINE_PGM("FN "); |
||||
|
for (int i = 0, fileIndex = 0, numFiles = 0; i < (int)fileList.count() && numFiles < 4; i++) { |
||||
|
fileList.seek(i); |
||||
|
if (!fileList.isDir()) { |
||||
|
if (fileIndex >= startIndex) { |
||||
|
sendLine(fileList.shortFilename()); |
||||
|
sendLine(fileList.longFilename()); |
||||
|
numFiles++; |
||||
|
} |
||||
|
fileIndex++; |
||||
|
} |
||||
|
} |
||||
|
SENDLINE_PGM("END"); |
||||
|
} else { |
||||
|
SENDLINE_PGM("J02"); |
||||
|
} |
||||
|
break; |
||||
|
case 9: // Pause SD Card Print
|
||||
|
if (isPrintingFromMedia()) { |
||||
|
pausePrint(); |
||||
|
is_printing_from_sd = false; |
||||
|
SENDLINE_PGM("J05"); |
||||
|
} else { |
||||
|
SENDLINE_PGM("J16"); // Print stopped
|
||||
|
} |
||||
|
break; |
||||
|
case 10: // Resume SD Card Print
|
||||
|
if (is_out_of_filament) { |
||||
|
is_out_of_filament = false; |
||||
|
// Filament change did eject the old filament automatically,
|
||||
|
// now continue and load the new one
|
||||
|
setUserConfirmed(); |
||||
|
SENDLINE_PGM("J04"); // Printing from SD card
|
||||
|
} else if (isPrintingFromMediaPaused()) { |
||||
|
resumePrint(); |
||||
|
SENDLINE_PGM("J04"); // Printing from SD card
|
||||
|
} |
||||
|
break; |
||||
|
case 11: // Stop SD Card Print
|
||||
|
if (isPrintingFromMedia()) { |
||||
|
stopPrint(); |
||||
|
is_printing_from_sd = false; |
||||
|
SENDLINE_PGM("J16"); // Print stopped
|
||||
|
} |
||||
|
break; |
||||
|
//case 12: // Kill
|
||||
|
// break;
|
||||
|
case 13: // Select File
|
||||
|
if (!isPrinting()) { |
||||
|
// Store selected file name
|
||||
|
char *p = strchr(rx, ' '); |
||||
|
if (p != nullptr && strlen(p+1) < sizeof(selectedFileShortName)) { |
||||
|
strcpy(selectedFileShortName, p+1); |
||||
|
SENDLINE_PGM("J20"); // Open succeeded
|
||||
|
} |
||||
|
else |
||||
|
SENDLINE_PGM("J21"); // Open failed
|
||||
|
} |
||||
|
break; |
||||
|
case 14: // Start Print
|
||||
|
if (!isPrinting() && strcmp(selectedFileShortName, "") != 0) { |
||||
|
printFile(selectedFileShortName); |
||||
|
is_printing_from_sd = true; |
||||
|
SENDLINE_PGM("J04"); // Printing from SD card
|
||||
|
} |
||||
|
break; |
||||
|
case 15: // Resume from power outage
|
||||
|
// This is not supported, just report print as completed
|
||||
|
SENDLINE_PGM("J16"); // Print stopped
|
||||
|
break; |
||||
|
case 16: // Set Hotend Target Temperature
|
||||
|
{ |
||||
|
int temp = parseIntArgument(rx, 'S'); |
||||
|
if (temp >= 0) |
||||
|
setTargetTemp_celsius(temp, E0); |
||||
|
} |
||||
|
break; |
||||
|
case 17: // Set Bed Target Temperature
|
||||
|
{ |
||||
|
int temp = parseIntArgument(rx, 'S'); |
||||
|
if (temp >= 0) |
||||
|
setTargetTemp_celsius(temp, BED); |
||||
|
} |
||||
|
break; |
||||
|
case 18: // Set Fan Speed
|
||||
|
{ |
||||
|
int temp = parseIntArgument(rx, 'S'); |
||||
|
if (temp >= 0) |
||||
|
setTargetFan_percent(temp, FAN0); |
||||
|
} |
||||
|
break; |
||||
|
case 19: // Disable Motors
|
||||
|
injectCommands_P(PSTR("M84")); |
||||
|
break; |
||||
|
case 20: // Get/Set Printing Speed
|
||||
|
{ |
||||
|
int newPerc = parseIntArgument(rx, 'S'); |
||||
|
if (newPerc >= 0) |
||||
|
setFeedrate_percent(newPerc); |
||||
|
else |
||||
|
SENDVALUELINE_PGM("A20V ", (int)getFeedrate_percent()); |
||||
|
} |
||||
|
break; |
||||
|
case 21: // Home axes
|
||||
|
if (!isPrinting()) { |
||||
|
const bool hasX = strchr(rx, 'X') != nullptr, |
||||
|
hasY = strchr(rx, 'Y') != nullptr, |
||||
|
hasZ = strchr(rx, 'Z') != nullptr, |
||||
|
hasC = strchr(rx, 'C') != nullptr; |
||||
|
if (hasX || hasY || hasZ) { |
||||
|
if (hasX) injectCommands_P(PSTR("G28 X")); |
||||
|
if (hasY) injectCommands_P(PSTR("G28 Y")); |
||||
|
if (hasZ) injectCommands_P(PSTR("G28 Z")); |
||||
|
} else if (hasC) { |
||||
|
injectCommands_P(PSTR("G28")); |
||||
|
} |
||||
|
} |
||||
|
break; |
||||
|
case 22: // Move axes
|
||||
|
if (!isPrinting()) { |
||||
|
const int feedrate = parseIntArgument(rx, 'F') / 60; |
||||
|
float delta; |
||||
|
if (!isnan(delta = parseFloatArgument(rx, 'X'))) |
||||
|
moveAxis(delta, feedrate, X); |
||||
|
else if (!isnan(delta = parseFloatArgument(rx, 'Y'))) |
||||
|
moveAxis(delta, feedrate, Y); |
||||
|
else if (!isnan(delta = parseFloatArgument(rx, 'Z'))) |
||||
|
moveAxis(delta, feedrate, Z); |
||||
|
} |
||||
|
break; |
||||
|
case 23: // Preheat PLA
|
||||
|
setTargetTemp_celsius(PREHEAT_1_TEMP_HOTEND, E0); |
||||
|
setTargetTemp_celsius(PREHEAT_1_TEMP_BED, BED); |
||||
|
SENDLINE_PGM("OK"); |
||||
|
break; |
||||
|
case 24: // Preheat ABS
|
||||
|
setTargetTemp_celsius(PREHEAT_2_TEMP_HOTEND, E0); |
||||
|
setTargetTemp_celsius(PREHEAT_2_TEMP_BED, BED); |
||||
|
SENDLINE_PGM("OK"); |
||||
|
break; |
||||
|
case 25: // Cool down
|
||||
|
setTargetTemp_celsius(0, E0); |
||||
|
setTargetTemp_celsius(0, BED); |
||||
|
SENDLINE_PGM("J12"); |
||||
|
break; |
||||
|
case 26: // Refresh SD Card
|
||||
|
fileList.refresh(); |
||||
|
break; |
||||
|
//case 27: // Adjust Servo Angles
|
||||
|
// break;
|
||||
|
//case 28: // Filament Test
|
||||
|
// break;
|
||||
|
case 29: // Get Bed Autolevel Grid
|
||||
|
{ |
||||
|
int x = parseIntArgument(rx, 'X'), |
||||
|
y = parseIntArgument(rx, 'Y'); |
||||
|
if (x != -1 && y != -1) { |
||||
|
xy_uint8_t coord; |
||||
|
coord.set(x, y); |
||||
|
const int value = mmToHundredths(getMeshPoint(coord)); |
||||
|
SENDVALUELINE_PGM("A29V ", value); |
||||
|
} |
||||
|
} |
||||
|
break; |
||||
|
case 30: // Autolevel
|
||||
|
if (strchr(rx, 'S')) { // Autoleveling started by clicking "PROBE" and then "OK"
|
||||
|
// Note:
|
||||
|
// We check for completion by monitoring the command queue.
|
||||
|
// Since it will become empty *while* processing the last injected command,
|
||||
|
// we enqueue an extra 10ms delay so we can the determine when all the others
|
||||
|
// have completed.
|
||||
|
if (isMachineHomed()) |
||||
|
injectCommands_P(PSTR("G29\nG4 P10")); |
||||
|
else |
||||
|
injectCommands_P(PSTR("G28\nG29\nG4 P10")); |
||||
|
is_auto_leveling = true; |
||||
|
} else { // Entering Autoleveling screen
|
||||
|
if (isPrinting()) |
||||
|
SENDLINE_PGM("J24"); // Disallow autoleveling
|
||||
|
else |
||||
|
SENDLINE_PGM("J26"); // Allow autoleveling
|
||||
|
} |
||||
|
break; |
||||
|
case 31: // Set Bed Autolevel Z offset
|
||||
|
if (strchr(rx, 'G')) { // Get
|
||||
|
SENDVALUELINE_PGM("A31V ", getZOffset_mm()); |
||||
|
} else if (strchr(rx, 'S')) { // Set
|
||||
|
float delta = parseFloatArgument(rx, 'S'); |
||||
|
delta = constrain(delta, -1.0, 1.0); |
||||
|
setZOffset_mm(getZOffset_mm() + delta); |
||||
|
|
||||
|
SENDVALUELINE_PGM("A31V ", getZOffset_mm()); |
||||
|
} else if (strchr(rx, 'D')) { // Save
|
||||
|
injectCommands_P(PSTR("M500")); |
||||
|
} |
||||
|
break; |
||||
|
//case 32: // ?
|
||||
|
// break;
|
||||
|
case 33: // Get Version Info
|
||||
|
SENDLINE_PGM("J33 " SHORT_BUILD_VERSION); |
||||
|
break; |
||||
|
case 34: // Set Bed Autolevel Grid
|
||||
|
{ |
||||
|
int x = parseIntArgument(rx, 'X'), |
||||
|
y = parseIntArgument(rx, 'Y'), |
||||
|
v = parseIntArgument(rx, 'V'); |
||||
|
if (x != -1 && y != -1 && v != -1) { // Set new value
|
||||
|
float value = hundredthsToMm(v); |
||||
|
value = constrain(value, -10, 10); |
||||
|
|
||||
|
xy_uint8_t coord; |
||||
|
coord.set(x, y); |
||||
|
setMeshPoint(coord, value); |
||||
|
} else if (strchr(rx, 'S')) { // Save (apply new values)
|
||||
|
injectCommands_P(PSTR("M500")); |
||||
|
} else if (strchr(rx, 'C')) { // Cancel (discard new values)
|
||||
|
injectCommands_P(PSTR("M501")); |
||||
|
} |
||||
|
} |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
#define RX_LEN_MAX 63 |
||||
|
static void parseSerialRx() { |
||||
|
static char rxBuffer[RX_LEN_MAX+1]; |
||||
|
static uint8_t rxLen = 0; |
||||
|
|
||||
|
while (ANYCUBIC_LCD_SERIAL.available()) { |
||||
|
const char c = ANYCUBIC_LCD_SERIAL.read(); |
||||
|
switch (c) { |
||||
|
case '\r': case '\n': |
||||
|
if (rxLen > 0 && rxLen <= RX_LEN_MAX) { |
||||
|
rxBuffer[rxLen] = '\0'; // Terminate string
|
||||
|
handleCmd(rxBuffer); |
||||
|
} |
||||
|
rxLen = 0; |
||||
|
break; |
||||
|
default: |
||||
|
if (rxLen < RX_LEN_MAX) |
||||
|
rxBuffer[rxLen++] = c; |
||||
|
else { |
||||
|
rxLen = 0xFF; // Overrun
|
||||
|
SERIAL_ECHOPGM("Warning: dropping long received line"); |
||||
|
} |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
static void detectPrintFromSdCompletion() { |
||||
|
// Note: printFile() queues some commands that actually start the print, so isPrintingFromMedia()
|
||||
|
// initially returns false
|
||||
|
if (is_printing_from_sd && !commandsInQueue() && !isPrintingFromMedia()) { |
||||
|
is_printing_from_sd = false; |
||||
|
SENDLINE_PGM("J14"); // Print done
|
||||
|
} |
||||
|
} |
||||
|
|
||||
|
static void detectAutolevelingCompletion() { |
||||
|
if (is_auto_leveling && !commandsInQueue()) { |
||||
|
is_auto_leveling = false; |
||||
|
injectCommands_P(PSTR("M500")); |
||||
|
SENDLINE_PGM("J25"); // Autoleveling done
|
||||
|
} |
||||
|
} |
||||
|
|
||||
|
void onStartup() { |
||||
|
ANYCUBIC_LCD_SERIAL.begin(115200); |
||||
|
sendNewLine(); |
||||
|
SENDLINE_PGM("J17"); // Reset
|
||||
|
delay_ms(10); |
||||
|
SENDLINE_PGM("J12"); // Ready
|
||||
|
} |
||||
|
|
||||
|
void onIdle() { |
||||
|
parseSerialRx(); |
||||
|
detectAutolevelingCompletion(); |
||||
|
detectPrintFromSdCompletion(); |
||||
|
} |
||||
|
|
||||
|
void onPrinterKilled(PGM_P const error, PGM_P const component) { } |
||||
|
|
||||
|
void onMediaInserted() { |
||||
|
SENDLINE_PGM("J00"); // SD Inserted
|
||||
|
} |
||||
|
|
||||
|
void onMediaError() { } |
||||
|
|
||||
|
void onMediaRemoved() { |
||||
|
SENDLINE_PGM("J01"); // SD Removed
|
||||
|
} |
||||
|
|
||||
|
void onPlayTone(const uint16_t frequency, const uint16_t duration) { |
||||
|
tone(BEEPER_PIN, frequency, duration); |
||||
|
} |
||||
|
|
||||
|
void onPrintTimerStarted() { } |
||||
|
|
||||
|
void onPrintTimerPaused() { } |
||||
|
|
||||
|
void onPrintTimerStopped() { } |
||||
|
|
||||
|
void onFilamentRunout(const extruder_t extruder) { |
||||
|
is_out_of_filament = true; |
||||
|
SENDLINE_PGM("J23"); // Filament runout
|
||||
|
SENDLINE_PGM("J18"); // Print paused
|
||||
|
// Note: printer will unload filament automatically
|
||||
|
} |
||||
|
|
||||
|
void onUserConfirmRequired(const char * const msg) { } |
||||
|
|
||||
|
void onStatusChanged(const char * const msg) { } |
||||
|
|
||||
|
void onFactoryReset() { } |
||||
|
|
||||
|
void onStoreSettings(char *buff) { } |
||||
|
|
||||
|
void onLoadSettings(const char *buff) { } |
||||
|
|
||||
|
void onConfigurationStoreWritten(bool success) { } |
||||
|
|
||||
|
void onConfigurationStoreRead(bool success) { } |
||||
|
|
||||
|
void onMeshUpdate(const int8_t xpos, const int8_t ypos, const float zval) { } |
||||
|
|
||||
|
#if ENABLED(POWER_LOSS_RECOVERY) |
||||
|
void onPowerLossResume() { } |
||||
|
#endif |
||||
|
|
||||
|
#if HAS_PID_HEATING |
||||
|
void onPidTuning(const result_t rst) { } |
||||
|
#endif |
||||
|
} |
||||
|
|
||||
|
#endif // ANYCUBIC_LCD_CHIRON
|
Loading…
Reference in new issue