2020-09-11 04:19:04 -07:00
|
|
|
/**
|
2015-07-10 06:01:56 -07:00
|
|
|
* @file console_io.cpp
|
|
|
|
*
|
|
|
|
* @date Dec 29, 2012
|
2020-01-13 18:57:43 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2020
|
2015-07-10 06:01:56 -07:00
|
|
|
*
|
|
|
|
* This file is part of rusEfi - see http://rusefi.com
|
|
|
|
*
|
2020-05-04 10:28:00 -07:00
|
|
|
* rusEFI can communicate with external universe via native USB or some sort of TTL mode
|
|
|
|
* We have an interesting situation with TTL communication channels, we have
|
|
|
|
* 1) SERIAL - this one was implemented first simply because the code was readily available (works on stm32)
|
|
|
|
* this one is most suitable for streaming HAL API
|
|
|
|
* this one is not great since each byte requires an IRQ and with enough IRQ delay we have a risk of data loss
|
|
|
|
* 2) UART DMA - the best one since FIFO buffer reduces data loss (works on stm32)
|
|
|
|
* We have two halves of DMA buffer - one is used for TTL while rusEFI prepares next batch of data in the other side.
|
|
|
|
* We need idle support in order to not wait for the complete buffer to get full in order to recieve a message.
|
|
|
|
* Back when we were implementing this STM32_DMA_CR_HTIE was not available in ChibiOS driver so we have added it.
|
|
|
|
* we have custom rusEFI changes to ChibiOS HAL driver v1
|
|
|
|
* F7 uses driver v2 which currently does not have rusEFI changes.
|
|
|
|
* open question if fresh ChibiOS is better in this regard.
|
|
|
|
* 3) UART this one is useful on platforms with hardware FIFO buffer like Kinetis.
|
|
|
|
* stm32 does not have such buffer so for stm32 UART without DMA has no advantages
|
|
|
|
*
|
|
|
|
*
|
2015-07-10 06:01:56 -07:00
|
|
|
* rusEfi 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.
|
|
|
|
*
|
|
|
|
* rusEfi 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/>.
|
|
|
|
*/
|
|
|
|
|
2019-03-29 07:29:01 -07:00
|
|
|
#include "engine.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
#include "console_io.h"
|
2019-07-06 17:15:49 -07:00
|
|
|
#include "os_util.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
#include "tunerstudio.h"
|
2020-06-21 16:48:55 -07:00
|
|
|
#include "connector_uart_dma.h"
|
2021-02-28 04:30:45 -08:00
|
|
|
#include "thread_priority.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_SIMULATOR
|
2015-07-10 06:01:56 -07:00
|
|
|
#include "rusEfiFunctionalTest.h"
|
2017-01-05 01:03:02 -08:00
|
|
|
#endif /*EFI_SIMULATOR */
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2016-02-14 09:03:48 -08:00
|
|
|
EXTERN_ENGINE;
|
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if HAL_USE_SERIAL_USB
|
2015-07-10 06:01:56 -07:00
|
|
|
#include "usbcfg.h"
|
|
|
|
#include "usbconsole.h"
|
2020-09-11 02:50:48 -07:00
|
|
|
#define EFI_CONSOLE_USB_DEVICE SDU1
|
|
|
|
#define SERIAL_USB_DRIVER SerialUSBDriver
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2020-09-11 02:50:48 -07:00
|
|
|
#ifdef EFI_CONSOLE_USB_DEVICE
|
|
|
|
extern SERIAL_USB_DRIVER EFI_CONSOLE_USB_DEVICE;
|
|
|
|
#endif /* EFI_CONSOLE_USB_DEVICE */
|
|
|
|
|
2020-09-11 05:25:42 -07:00
|
|
|
#endif /* HAL_USE_SERIAL_USB */
|
2020-09-11 02:50:48 -07:00
|
|
|
|
2019-05-05 08:06:27 -07:00
|
|
|
// 10 seconds
|
|
|
|
#define CONSOLE_WRITE_TIMEOUT 10000
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
static bool isSerialConsoleStarted = false;
|
|
|
|
|
2020-12-05 08:54:09 -08:00
|
|
|
#if (defined(EFI_CONSOLE_SERIAL_DEVICE) && ! EFI_SIMULATOR)
|
|
|
|
static event_listener_t consoleEventListener;
|
|
|
|
#endif
|
|
|
|
|
2016-02-13 18:02:14 -08:00
|
|
|
bool consoleByteArrived = false;
|
|
|
|
|
|
|
|
void onDataArrived(void) {
|
|
|
|
consoleByteArrived = true;
|
|
|
|
}
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
CommandHandler console_line_callback;
|
|
|
|
|
2019-05-05 08:06:27 -07:00
|
|
|
#if (defined(EFI_CONSOLE_SERIAL_DEVICE) && ! EFI_SIMULATOR )
|
2021-01-09 15:55:58 -08:00
|
|
|
SerialConfig serialConfig = {
|
|
|
|
.speed = 0,
|
|
|
|
.cr1 = 0,
|
|
|
|
.cr2 = USART_CR2_STOP1_BITS | USART_CR2_LINEN,
|
|
|
|
.cr3 = 0
|
|
|
|
};
|
2015-07-10 06:01:56 -07:00
|
|
|
#endif
|
|
|
|
|
2019-05-05 08:06:27 -07:00
|
|
|
#if (defined(EFI_CONSOLE_UART_DEVICE) && ! EFI_SIMULATOR )
|
|
|
|
/* Note: This structure is modified from the default ChibiOS layout! */
|
2020-05-03 21:10:20 -07:00
|
|
|
UARTConfig uartConfig = {
|
2021-01-09 15:55:58 -08:00
|
|
|
.txend1_cb = NULL,
|
|
|
|
.txend2_cb = NULL,
|
|
|
|
.rxend_cb = NULL,
|
|
|
|
.rxchar_cb = NULL,
|
|
|
|
.rxerr_cb = NULL,
|
2021-01-19 12:20:35 -08:00
|
|
|
.timeout_cb = NULL,
|
2021-01-09 15:55:58 -08:00
|
|
|
.speed = 0,
|
|
|
|
.cr1 = 0,
|
|
|
|
.cr2 = 0/*USART_CR2_STOP1_BITS*/ | USART_CR2_LINEN,
|
|
|
|
.cr3 = 0,
|
|
|
|
.rxhalf_cb = NULL
|
2019-05-05 08:06:27 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
// To use UART driver instead of Serial, we need to imitate "BaseChannel" streaming functionality
|
|
|
|
static msg_t _putt(void *, uint8_t b, sysinterval_t timeout) {
|
|
|
|
int n = 1;
|
|
|
|
uartSendTimeout(EFI_CONSOLE_UART_DEVICE, (size_t *)&n, &b, timeout);
|
|
|
|
return MSG_OK;
|
|
|
|
}
|
|
|
|
static size_t _writet(void *, const uint8_t *bp, size_t n, sysinterval_t timeout) {
|
|
|
|
uartSendTimeout(EFI_CONSOLE_UART_DEVICE, (size_t *)&n, bp, timeout);
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
static msg_t _put(void *ip, uint8_t b) {
|
|
|
|
#ifdef UART_USE_BLOCKING_SEND
|
|
|
|
// this version can be called from the locked state (no interrupts)
|
|
|
|
uart_lld_blocking_send(EFI_CONSOLE_UART_DEVICE, 1, (void *)&b);
|
|
|
|
#else
|
|
|
|
// uartSendTimeout() needs interrupts to wait for the end of transfer, so we have to unlock them temporary
|
|
|
|
bool wasLocked = isLocked();
|
2020-11-19 03:56:02 -08:00
|
|
|
if (wasLocked) {
|
|
|
|
if (isIsrContext()) {
|
|
|
|
chSysUnlockFromISR()
|
|
|
|
;
|
|
|
|
} else {
|
|
|
|
chSysUnlock()
|
|
|
|
;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-05 08:06:27 -07:00
|
|
|
_putt(ip, b, CONSOLE_WRITE_TIMEOUT);
|
2020-11-19 03:56:02 -08:00
|
|
|
|
|
|
|
// Relock if we were locked before
|
|
|
|
if (wasLocked) {
|
|
|
|
if (isIsrContext()) {
|
|
|
|
chSysLockFromISR();
|
|
|
|
} else {
|
|
|
|
chSysLock();
|
|
|
|
}
|
|
|
|
}
|
2019-05-05 08:06:27 -07:00
|
|
|
#endif /* UART_USE_BLOCKING_WRITE */
|
|
|
|
return MSG_OK;
|
|
|
|
}
|
|
|
|
static size_t _write(void *ip, const uint8_t *bp, size_t n) {
|
|
|
|
return _writet(ip, bp, n, CONSOLE_WRITE_TIMEOUT);
|
|
|
|
}
|
|
|
|
static msg_t _gett(void *, sysinterval_t /*timeout*/) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
static size_t _readt(void *, uint8_t */*bp*/, size_t /*n*/, sysinterval_t /*timeout*/) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
static msg_t _get(void *) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
static size_t _read(void *, uint8_t */*bp*/, size_t /*n*/) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
static msg_t _ctl(void *, unsigned int /*operation*/, void */*arg*/) {
|
|
|
|
return MSG_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is a "fake" channel for getConsoleChannel() filled with our handlers
|
|
|
|
static const struct BaseChannelVMT uartChannelVmt = {
|
|
|
|
.instance_offset = (size_t)0, .write = _write, .read = _read, .put = _put, .get = _get,
|
|
|
|
.putt = _putt, .gett = _gett, .writet = _writet, .readt = _readt, .ctl = _ctl
|
|
|
|
};
|
|
|
|
static const BaseChannel uartChannel = { .vmt = &uartChannelVmt };
|
|
|
|
#endif /* EFI_CONSOLE_UART_DEVICE */
|
|
|
|
|
2020-06-21 22:18:58 -07:00
|
|
|
ts_channel_s primaryChannel;
|
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_PROD_CODE || EFI_EGT
|
2017-01-05 01:03:02 -08:00
|
|
|
|
2019-04-01 11:18:21 -07:00
|
|
|
bool isUsbSerial(BaseChannel * channel) {
|
2020-09-11 04:51:49 -07:00
|
|
|
#if HAL_USE_SERIAL_USB
|
2020-09-11 02:50:48 -07:00
|
|
|
return channel == (BaseChannel *) &EFI_CONSOLE_USB_DEVICE;
|
2019-04-01 11:18:21 -07:00
|
|
|
#else
|
|
|
|
return false;
|
2020-09-11 02:50:48 -07:00
|
|
|
#endif /* EFI_CONSOLE_USB_DEVICE */
|
2019-04-01 11:18:21 -07:00
|
|
|
}
|
2017-05-23 10:10:43 -07:00
|
|
|
BaseChannel * getConsoleChannel(void) {
|
2020-06-21 21:25:19 -07:00
|
|
|
#if PRIMARY_UART_DMA_MODE
|
|
|
|
if (primaryChannel.uartp != nullptr) {
|
|
|
|
// primary channel is in DMA mode - we do not have a stream implementation for this.
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-05-05 08:06:27 -07:00
|
|
|
#if defined(EFI_CONSOLE_SERIAL_DEVICE)
|
|
|
|
return (BaseChannel *) EFI_CONSOLE_SERIAL_DEVICE;
|
|
|
|
#endif /* EFI_CONSOLE_SERIAL_DEVICE */
|
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if defined(EFI_CONSOLE_UART_DEVICE)
|
2019-05-05 08:06:27 -07:00
|
|
|
return (BaseChannel *) &uartChannel;
|
2015-07-10 06:01:56 -07:00
|
|
|
#endif /* EFI_CONSOLE_UART_DEVICE */
|
|
|
|
|
2020-09-11 04:19:04 -07:00
|
|
|
#if HAL_USE_SERIAL_USB
|
2020-09-11 02:50:48 -07:00
|
|
|
return (BaseChannel *) &EFI_CONSOLE_USB_DEVICE;
|
2015-07-10 06:01:56 -07:00
|
|
|
#else
|
2020-06-21 16:48:55 -07:00
|
|
|
return nullptr;
|
2017-01-05 01:03:02 -08:00
|
|
|
#endif /* HAL_USE_SERIAL_USB */
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2017-01-05 01:03:02 -08:00
|
|
|
bool isCommandLineConsoleReady(void) {
|
2019-04-01 08:39:25 -07:00
|
|
|
return isSerialConsoleStarted;
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
#endif /* EFI_PROD_CODE || EFI_EGT */
|
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if !defined(EFI_CONSOLE_NO_THREAD)
|
2019-03-30 14:41:46 -07:00
|
|
|
|
2020-08-02 07:52:52 -07:00
|
|
|
static THD_WORKING_AREA(consoleThreadStack, CONNECTIVITY_THREAD_STACK);
|
2019-04-01 14:04:49 -07:00
|
|
|
static THD_FUNCTION(consoleThreadEntryPoint, arg) {
|
2015-07-10 06:01:56 -07:00
|
|
|
(void) arg;
|
|
|
|
chRegSetThreadName("console thread");
|
|
|
|
|
2020-06-21 16:48:55 -07:00
|
|
|
#if !PRIMARY_UART_DMA_MODE
|
2020-05-02 21:20:54 -07:00
|
|
|
primaryChannel.channel = (BaseChannel *) getConsoleChannel();
|
2020-06-21 16:48:55 -07:00
|
|
|
#endif
|
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_TUNER_STUDIO
|
2020-06-21 17:47:46 -07:00
|
|
|
runBinaryProtocolLoop(&primaryChannel);
|
2019-04-01 09:11:57 -07:00
|
|
|
#endif /* EFI_TUNER_STUDIO */
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2019-03-30 14:41:46 -07:00
|
|
|
#endif /* EFI_CONSOLE_NO_THREAD */
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
void consoleOutputBuffer(const uint8_t *buf, int size) {
|
|
|
|
#if !EFI_UART_ECHO_TEST_MODE
|
2020-06-21 18:17:48 -07:00
|
|
|
BaseChannel * channel = getConsoleChannel();
|
|
|
|
if (channel != nullptr) {
|
|
|
|
chnWriteTimeout(channel, buf, size, CONSOLE_WRITE_TIMEOUT);
|
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
#endif /* EFI_UART_ECHO_TEST_MODE */
|
|
|
|
}
|
|
|
|
|
|
|
|
static Logging *logger;
|
|
|
|
|
|
|
|
void startConsole(Logging *sharedLogger, CommandHandler console_line_callback_p) {
|
|
|
|
logger = sharedLogger;
|
|
|
|
console_line_callback = console_line_callback_p;
|
|
|
|
|
2020-06-22 06:46:56 -07:00
|
|
|
#if (defined(EFI_CONSOLE_SERIAL_DEVICE) || defined(EFI_CONSOLE_UART_DEVICE)) && ! EFI_SIMULATOR
|
2021-02-12 17:27:43 -08:00
|
|
|
efiSetPadMode("console RX", EFI_CONSOLE_RX_BRAIN_PIN, PAL_MODE_ALTERNATE(EFI_CONSOLE_AF));
|
|
|
|
efiSetPadMode("console TX", EFI_CONSOLE_TX_BRAIN_PIN, PAL_MODE_ALTERNATE(EFI_CONSOLE_AF));
|
2020-06-21 13:37:33 -07:00
|
|
|
#endif
|
|
|
|
|
2020-06-21 21:46:17 -07:00
|
|
|
#if PRIMARY_UART_DMA_MODE && ! EFI_SIMULATOR
|
|
|
|
primaryChannel.uartp = EFI_CONSOLE_UART_DEVICE;
|
2020-06-21 16:48:55 -07:00
|
|
|
startUartDmaConnector(primaryChannel.uartp PASS_CONFIG_PARAMETER_SUFFIX);
|
2020-06-21 18:17:48 -07:00
|
|
|
isSerialConsoleStarted = true;
|
2020-06-21 16:48:55 -07:00
|
|
|
#elif (defined(EFI_CONSOLE_SERIAL_DEVICE) && ! EFI_SIMULATOR)
|
2015-07-10 06:01:56 -07:00
|
|
|
/*
|
2017-06-04 08:31:20 -07:00
|
|
|
* Activates the serial
|
2015-07-10 06:01:56 -07:00
|
|
|
* it is important to set 'NONE' as flow control! in terminal application on the PC
|
|
|
|
*/
|
2017-06-04 08:31:20 -07:00
|
|
|
serialConfig.speed = engineConfiguration->uartConsoleSerialSpeed;
|
2019-05-05 08:06:27 -07:00
|
|
|
sdStart(EFI_CONSOLE_SERIAL_DEVICE, &serialConfig);
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-05-05 08:06:27 -07:00
|
|
|
chEvtRegisterMask((event_source_t *) chnGetEventSource(EFI_CONSOLE_SERIAL_DEVICE), &consoleEventListener, 1);
|
2020-06-21 18:17:48 -07:00
|
|
|
isSerialConsoleStarted = true;
|
2019-05-05 08:06:27 -07:00
|
|
|
#elif (defined(EFI_CONSOLE_UART_DEVICE) && ! EFI_SIMULATOR)
|
|
|
|
uartConfig.speed = engineConfiguration->uartConsoleSerialSpeed;
|
|
|
|
uartStart(EFI_CONSOLE_UART_DEVICE, &uartConfig);
|
2020-06-21 18:17:48 -07:00
|
|
|
isSerialConsoleStarted = true;
|
2019-05-05 08:06:27 -07:00
|
|
|
#endif /* EFI_CONSOLE_SERIAL_DEVICE || EFI_CONSOLE_UART_DEVICE */
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if !defined(EFI_CONSOLE_NO_THREAD)
|
2021-02-28 04:30:45 -08:00
|
|
|
chThdCreateStatic(consoleThreadStack, sizeof(consoleThreadStack), PRIO_CONSOLE, (tfunc_t)consoleThreadEntryPoint, NULL);
|
2019-03-30 14:41:46 -07:00
|
|
|
#endif /* EFI_CONSOLE_NO_THREAD */
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|