/** * 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 . * */ #ifdef TARGET_LPC1768 #include "HardwareSerial.h" #if SERIAL_PORT == 0 || SERIAL_PORT_2 == 0 HardwareSerial Serial = HardwareSerial(LPC_UART0); #elif SERIAL_PORT == 1 || SERIAL_PORT_2 == 1 HardwareSerial Serial1 = HardwareSerial((LPC_UART_TypeDef *) LPC_UART1); #elif SERIAL_PORT == 2 || SERIAL_PORT_2 == 2 HardwareSerial Serial2 = HardwareSerial(LPC_UART2); #elif SERIAL_PORT == 3 || SERIAL_PORT_2 == 3 HardwareSerial Serial3 = HardwareSerial(LPC_UART3); #endif void HardwareSerial::begin(uint32_t baudrate) { UART_CFG_Type UARTConfigStruct; PINSEL_CFG_Type PinCfg; UART_FIFO_CFG_Type FIFOConfig; if (Baudrate == baudrate) return; // No need to re-initialize if (UARTx == LPC_UART0) { // Initialize UART0 pin connect PinCfg.Funcnum = 1; PinCfg.OpenDrain = 0; PinCfg.Pinmode = 0; PinCfg.Pinnum = 2; PinCfg.Portnum = 0; PINSEL_ConfigPin(&PinCfg); PinCfg.Pinnum = 3; PINSEL_ConfigPin(&PinCfg); } else if ((LPC_UART1_TypeDef *) UARTx == LPC_UART1) { // Initialize UART1 pin connect PinCfg.Funcnum = 1; PinCfg.OpenDrain = 0; PinCfg.Pinmode = 0; PinCfg.Pinnum = 15; PinCfg.Portnum = 0; PINSEL_ConfigPin(&PinCfg); PinCfg.Pinnum = 16; PINSEL_ConfigPin(&PinCfg); } else if (UARTx == LPC_UART2) { // Initialize UART2 pin connect PinCfg.Funcnum = 1; PinCfg.OpenDrain = 0; PinCfg.Pinmode = 0; PinCfg.Pinnum = 10; PinCfg.Portnum = 0; PINSEL_ConfigPin(&PinCfg); PinCfg.Pinnum = 11; PINSEL_ConfigPin(&PinCfg); } else if (UARTx == LPC_UART3) { // Initialize UART2 pin connect PinCfg.Funcnum = 1; PinCfg.OpenDrain = 0; PinCfg.Pinmode = 0; PinCfg.Pinnum = 0; PinCfg.Portnum = 0; PINSEL_ConfigPin(&PinCfg); PinCfg.Pinnum = 1; PINSEL_ConfigPin(&PinCfg); } /* Initialize UART Configuration parameter structure to default state: * Baudrate = 9600bps * 8 data bit * 1 Stop bit * None parity */ UART_ConfigStructInit(&UARTConfigStruct); // Re-configure baudrate UARTConfigStruct.Baud_rate = baudrate; // Initialize eripheral with given to corresponding parameter UART_Init(UARTx, &UARTConfigStruct); // Enable and reset the TX and RX FIFOs UART_FIFOConfigStructInit(&FIFOConfig); UART_FIFOConfig(UARTx, &FIFOConfig); // Enable UART Transmit UART_TxCmd(UARTx, ENABLE); // Configure Interrupts UART_IntConfig(UARTx, UART_INTCFG_RBR, ENABLE); UART_IntConfig(UARTx, UART_INTCFG_RLS, ENABLE); // Set proper priority and enable interrupts if (UARTx == LPC_UART0) { NVIC_SetPriority(UART0_IRQn, NVIC_EncodePriority(0, 3, 0)); NVIC_EnableIRQ(UART0_IRQn); } else if ((LPC_UART1_TypeDef *) UARTx == LPC_UART1) { NVIC_SetPriority(UART1_IRQn, NVIC_EncodePriority(0, 3, 0)); NVIC_EnableIRQ(UART1_IRQn); } else if (UARTx == LPC_UART2) { NVIC_SetPriority(UART2_IRQn, NVIC_EncodePriority(0, 3, 0)); NVIC_EnableIRQ(UART2_IRQn); } else if (UARTx == LPC_UART3) { NVIC_SetPriority(UART3_IRQn, NVIC_EncodePriority(0, 3, 0)); NVIC_EnableIRQ(UART3_IRQn); } RxQueueWritePos = RxQueueReadPos = 0; #if TX_BUFFER_SIZE > 0 TxQueueWritePos = TxQueueReadPos = 0; #endif // Save the configured baudrate Baudrate = baudrate; } int16_t HardwareSerial::peek() { int16_t byte = -1; // Temporarily lock out UART receive interrupts during this read so the UART receive // interrupt won't cause problems with the index values UART_IntConfig(UARTx, UART_INTCFG_RBR, DISABLE); if (RxQueueReadPos != RxQueueWritePos) byte = RxBuffer[RxQueueReadPos]; // Re-enable UART interrupts UART_IntConfig(UARTx, UART_INTCFG_RBR, ENABLE); return byte; } int16_t HardwareSerial::read() { int16_t byte = -1; // Temporarily lock out UART receive interrupts during this read so the UART receive // interrupt won't cause problems with the index values UART_IntConfig(UARTx, UART_INTCFG_RBR, DISABLE); if (RxQueueReadPos != RxQueueWritePos) { byte = RxBuffer[RxQueueReadPos]; RxQueueReadPos = (RxQueueReadPos + 1) % RX_BUFFER_SIZE; } // Re-enable UART interrupts UART_IntConfig(UARTx, UART_INTCFG_RBR, ENABLE); return byte; } size_t HardwareSerial::write(uint8_t send) { #if TX_BUFFER_SIZE > 0 size_t bytes = 0; uint32_t fifolvl = 0; // If the Tx Buffer is full, wait for space to clear if ((TxQueueWritePos+1) % TX_BUFFER_SIZE == TxQueueReadPos) flushTX(); // Temporarily lock out UART transmit interrupts during this read so the UART transmit interrupt won't // cause problems with the index values UART_IntConfig(UARTx, UART_INTCFG_THRE, DISABLE); // LPC17xx.h incorrectly defines FIFOLVL as a uint8_t, when it's actually a 32-bit register if ((LPC_UART1_TypeDef *) UARTx == LPC_UART1) { fifolvl = *(reinterpret_cast(&((LPC_UART1_TypeDef *) UARTx)->FIFOLVL)); } else fifolvl = *(reinterpret_cast(&UARTx->FIFOLVL)); // If the queue is empty and there's space in the FIFO, immediately send the byte if (TxQueueWritePos == TxQueueReadPos && fifolvl < UART_TX_FIFO_SIZE) { bytes = UART_Send(UARTx, &send, 1, BLOCKING); } // Otherwiise, write the byte to the transmit buffer else if ((TxQueueWritePos+1) % TX_BUFFER_SIZE != TxQueueReadPos) { TxBuffer[TxQueueWritePos] = send; TxQueueWritePos = (TxQueueWritePos+1) % TX_BUFFER_SIZE; bytes++; } // Re-enable the TX Interrupt UART_IntConfig(UARTx, UART_INTCFG_THRE, ENABLE); return bytes; #else return UART_Send(UARTx, &send, 1, BLOCKING); #endif } #if TX_BUFFER_SIZE > 0 void HardwareSerial::flushTX() { // Wait for the tx buffer and FIFO to drain while (TxQueueWritePos != TxQueueReadPos && UART_CheckBusy(UARTx) == SET); } #endif size_t HardwareSerial::available() { return (RxQueueWritePos + RX_BUFFER_SIZE - RxQueueReadPos) % RX_BUFFER_SIZE; } void HardwareSerial::flush() { RxQueueWritePos = 0; RxQueueReadPos = 0; } size_t HardwareSerial::printf(const char *format, ...) { char RxBuffer[256]; va_list vArgs; va_start(vArgs, format); int length = vsnprintf(RxBuffer, 256, format, vArgs); va_end(vArgs); if (length > 0 && length < 256) { for (size_t i = 0; i < (size_t)length; ++i) write(RxBuffer[i]); } return length; } void HardwareSerial::IRQHandler() { uint32_t IIRValue; uint8_t LSRValue, byte; IIRValue = UART_GetIntId(UARTx); IIRValue &= UART_IIR_INTID_MASK; // check bit 1~3, interrupt identification // Receive Line Status if (IIRValue == UART_IIR_INTID_RLS) { LSRValue = UART_GetLineStatus(UARTx); // Receive Line Status if (LSRValue & (UART_LSR_OE | UART_LSR_PE | UART_LSR_FE | UART_LSR_RXFE | UART_LSR_BI)) { // There are errors or break interrupt // Read LSR will clear the interrupt Status = LSRValue; byte = UART_ReceiveByte(UARTx); // Dummy read on RX to clear interrupt, then bail out return; } } // Receive Data Available if (IIRValue == UART_IIR_INTID_RDA) { // Clear the FIFO while (UART_Receive(UARTx, &byte, 1, NONE_BLOCKING)) { #if ENABLED(EMERGENCY_PARSER) emergency_parser.update(emergency_state, byte); #endif if ((RxQueueWritePos + 1) % RX_BUFFER_SIZE != RxQueueReadPos) { RxBuffer[RxQueueWritePos] = byte; RxQueueWritePos = (RxQueueWritePos + 1) % RX_BUFFER_SIZE; } else break; } // Character timeout indicator } else if (IIRValue == UART_IIR_INTID_CTI) { // Character Time-out indicator Status |= 0x100; // Bit 9 as the CTI error } #if TX_BUFFER_SIZE > 0 if (IIRValue == UART_IIR_INTID_THRE) { // Disable THRE interrupt UART_IntConfig(UARTx, UART_INTCFG_THRE, DISABLE); // Wait for FIFO buffer empty while (UART_CheckBusy(UARTx) == SET); // Transfer up to UART_TX_FIFO_SIZE bytes of data for (int i = 0; i < UART_TX_FIFO_SIZE && TxQueueWritePos != TxQueueReadPos; i++) { // Move a piece of data into the transmit FIFO if (UART_Send(UARTx, &TxBuffer[TxQueueReadPos], 1, NONE_BLOCKING)) { TxQueueReadPos = (TxQueueReadPos+1) % TX_BUFFER_SIZE; } else break; } // If there is no more data to send, disable the transmit interrupt - else enable it or keep it enabled if (TxQueueWritePos == TxQueueReadPos) { UART_IntConfig(UARTx, UART_INTCFG_THRE, DISABLE); } else UART_IntConfig(UARTx, UART_INTCFG_THRE, ENABLE); } #endif } #ifdef __cplusplus extern "C" { #endif void UART0_IRQHandler(void) { #if SERIAL_PORT == 0 || SERIAL_PORT_2 == 0 Serial.IRQHandler(); #endif } void UART1_IRQHandler(void) { #if SERIAL_PORT == 1 || SERIAL_PORT_2 == 1 Serial1.IRQHandler(); #endif } void UART2_IRQHandler(void) { #if SERIAL_PORT == 2 || SERIAL_PORT_2 == 2 Serial2.IRQHandler(); #endif } void UART3_IRQHandler(void) { #if SERIAL_PORT == 3 || SERIAL_PORT_2 == 3 Serial3.IRQHandler(); #endif } #ifdef __cplusplus } #endif #endif // TARGET_LPC1768