2014-08-29 07:52:33 -07:00
|
|
|
/**
|
|
|
|
* @file console_io.c
|
|
|
|
*
|
|
|
|
* @date Dec 29, 2012
|
2015-01-12 15:04:10 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2015
|
2014-08-29 07:52:33 -07:00
|
|
|
*
|
|
|
|
* This file is part of rusEfi - see http://rusefi.com
|
|
|
|
*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "main.h"
|
|
|
|
#include "console_io.h"
|
2015-01-08 09:03:39 -08:00
|
|
|
#include "rfiutil.h"
|
2015-03-08 20:04:55 -07:00
|
|
|
#include "tunerstudio.h"
|
2014-08-29 07:52:33 -07:00
|
|
|
|
2014-12-23 20:03:31 -08:00
|
|
|
#if HAL_USE_SERIAL_USB || defined(__DOXYGEN__)
|
2015-02-24 14:04:22 -08:00
|
|
|
#include "usbcfg.h"
|
|
|
|
#include "usbconsole.h"
|
|
|
|
extern SerialUSBDriver SDU1;
|
2014-12-23 20:03:31 -08:00
|
|
|
#endif
|
2014-08-29 07:52:33 -07:00
|
|
|
|
|
|
|
int lastWriteSize;
|
|
|
|
int lastWriteActual;
|
|
|
|
|
|
|
|
static bool isSerialConsoleStarted = false;
|
|
|
|
|
|
|
|
static EventListener consoleEventListener;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Reads a whole line from the input channel.
|
|
|
|
*
|
|
|
|
* @param[in] chp pointer to a @p BaseChannel object
|
|
|
|
* @param[in] line pointer to the line buffer
|
|
|
|
* @param[in] size buffer maximum length
|
|
|
|
* @return The operation status.
|
|
|
|
* @retval TRUE the channel was reset or CTRL-D pressed.
|
|
|
|
* @retval FALSE operation successful.
|
|
|
|
*/
|
|
|
|
static bool getConsoleLine(BaseSequentialStream *chp, char *line, unsigned size) {
|
|
|
|
char *p = line;
|
|
|
|
|
2015-03-21 21:04:52 -07:00
|
|
|
while (true) {
|
2014-08-29 07:52:33 -07:00
|
|
|
if (!isConsoleReady()) {
|
|
|
|
// we better do not read from USB serial before it is ready
|
|
|
|
chThdSleepMilliseconds(10);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
short c = (short) chSequentialStreamGet(chp);
|
|
|
|
|
|
|
|
if (isSerialOverUart()) {
|
|
|
|
uint32_t flags;
|
|
|
|
chSysLock()
|
|
|
|
;
|
|
|
|
|
|
|
|
flags = chEvtGetAndClearFlagsI(&consoleEventListener);
|
|
|
|
chSysUnlock()
|
|
|
|
;
|
|
|
|
|
|
|
|
if (flags & SD_OVERRUN_ERROR) {
|
|
|
|
// firmwareError("serial overrun");
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#if EFI_UART_ECHO_TEST_MODE
|
|
|
|
/**
|
|
|
|
* That's test code - let's test connectivity
|
|
|
|
*/
|
|
|
|
consolePutChar((uint8_t) c);
|
|
|
|
continue;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (c < 0 || c == 4) {
|
2015-03-21 21:04:52 -07:00
|
|
|
return true;
|
2015-02-24 14:04:22 -08:00
|
|
|
}
|
2014-08-29 07:52:33 -07:00
|
|
|
if (c == 8) {
|
|
|
|
if (p != line) {
|
|
|
|
// backspace
|
|
|
|
consolePutChar((uint8_t) c);
|
|
|
|
consolePutChar(0x20);
|
|
|
|
consolePutChar((uint8_t) c);
|
|
|
|
p--;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (c == '\r') {
|
|
|
|
consolePutChar('\r');
|
|
|
|
consolePutChar('\n');
|
|
|
|
*p = 0;
|
|
|
|
return false;
|
|
|
|
}
|
2015-03-08 20:04:55 -07:00
|
|
|
if (c == '\n') {
|
|
|
|
consolePutChar('\n');
|
|
|
|
*p = 0;
|
|
|
|
return false;
|
|
|
|
}
|
2014-08-29 07:52:33 -07:00
|
|
|
if (c < 0x20) {
|
|
|
|
continue;
|
2015-02-24 14:04:22 -08:00
|
|
|
}
|
2014-08-29 07:52:33 -07:00
|
|
|
if (p < line + size - 1) {
|
|
|
|
consolePutChar((uint8_t) c);
|
|
|
|
*p++ = (char) c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// todo: this is ugly as hell!
|
|
|
|
static char consoleInput[] = " ";
|
|
|
|
|
2015-03-08 15:07:31 -07:00
|
|
|
CommandHandler console_line_callback;
|
2014-08-29 07:52:33 -07:00
|
|
|
|
|
|
|
static bool is_serial_over_uart;
|
|
|
|
|
|
|
|
bool isSerialOverUart(void) {
|
|
|
|
return is_serial_over_uart;
|
|
|
|
}
|
|
|
|
|
2015-01-12 05:04:11 -08:00
|
|
|
#if (defined(EFI_CONSOLE_UART_DEVICE) && ! EFI_SIMULATOR ) || defined(__DOXYGEN__)
|
2014-08-29 07:52:33 -07:00
|
|
|
static SerialConfig serialConfig = { SERIAL_SPEED, 0, USART_CR2_STOP1_BITS | USART_CR2_LINEN, 0 };
|
2015-01-08 09:03:39 -08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if EFI_PROD_CODE || EFI_EGT || defined(__DOXYGEN__)
|
2014-08-29 07:52:33 -07:00
|
|
|
|
|
|
|
SerialDriver * getConsoleChannel(void) {
|
2015-01-08 09:03:39 -08:00
|
|
|
#if defined(EFI_CONSOLE_UART_DEVICE) || defined(__DOXYGEN__)
|
2014-08-29 07:52:33 -07:00
|
|
|
if (isSerialOverUart()) {
|
|
|
|
return (SerialDriver *) EFI_CONSOLE_UART_DEVICE;
|
|
|
|
}
|
2015-01-08 09:03:39 -08:00
|
|
|
#endif /* EFI_CONSOLE_UART_DEVICE */
|
|
|
|
|
|
|
|
#if HAL_USE_SERIAL_USB || defined(__DOXYGEN__)
|
|
|
|
return (SerialDriver *) &SDU1;
|
2014-12-23 20:03:31 -08:00
|
|
|
#else
|
2015-01-08 09:03:39 -08:00
|
|
|
return NULL;
|
2014-12-23 20:03:31 -08:00
|
|
|
#endif
|
2014-08-29 07:52:33 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
bool isConsoleReady(void) {
|
|
|
|
if (isSerialOverUart()) {
|
|
|
|
return isSerialConsoleStarted;
|
|
|
|
} else {
|
|
|
|
return is_usb_serial_ready();
|
|
|
|
}
|
|
|
|
}
|
2015-01-08 10:04:23 -08:00
|
|
|
#endif /* EFI_PROD_CODE || EFI_EGT */
|
2014-12-31 17:04:33 -08:00
|
|
|
|
2015-03-08 17:05:03 -07:00
|
|
|
bool_t consoleInBinaryMode = false;
|
|
|
|
|
2015-03-08 20:04:55 -07:00
|
|
|
ts_channel_s binaryConsole;
|
|
|
|
|
|
|
|
uint8_t buffer[DL_OUTPUT_BUFFER];
|
|
|
|
|
2014-12-07 16:03:14 -08:00
|
|
|
static THD_WORKING_AREA(consoleThreadStack, 2 * UTILITY_THREAD_STACK_SIZE);
|
|
|
|
static msg_t consoleThreadThreadEntryPoint(void *arg) {
|
|
|
|
(void) arg;
|
|
|
|
chRegSetThreadName("console thread");
|
|
|
|
|
|
|
|
#if EFI_PROD_CODE
|
|
|
|
if (!isSerialOverUart()) {
|
|
|
|
/**
|
|
|
|
* This method contains a long delay, that's the reason why this is not done on the main thread
|
|
|
|
*/
|
|
|
|
usb_serial_start();
|
|
|
|
}
|
|
|
|
#endif /* EFI_PROD_CODE */
|
|
|
|
|
2015-03-08 20:04:55 -07:00
|
|
|
binaryConsole.channel = (BaseChannel *) getConsoleChannel();
|
|
|
|
// todo: clean this spot!
|
|
|
|
binaryConsole.writeBuffer = buffer;
|
|
|
|
|
2014-12-07 16:03:14 -08:00
|
|
|
while (true) {
|
|
|
|
efiAssert(getRemainingStack(chThdSelf()) > 256, "lowstck#9e", 0);
|
|
|
|
bool end = getConsoleLine((BaseSequentialStream*) getConsoleChannel(), consoleInput, sizeof(consoleInput));
|
|
|
|
if (end) {
|
|
|
|
// firmware simulator is the only case when this happens
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-03-19 19:04:54 -07:00
|
|
|
char *trimmed = efiTrim(consoleInput);
|
|
|
|
|
|
|
|
(console_line_callback)(trimmed);
|
2015-03-08 20:04:55 -07:00
|
|
|
|
|
|
|
if (consoleInBinaryMode) {
|
|
|
|
// switch to binary protocol
|
2015-03-17 19:07:24 -07:00
|
|
|
runBinaryProtocolLoop(&binaryConsole, true);
|
2015-03-08 20:04:55 -07:00
|
|
|
}
|
2014-12-07 16:03:14 -08:00
|
|
|
}
|
|
|
|
#if defined __GNUC__
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-08-29 07:52:33 -07:00
|
|
|
void consolePutChar(int x) {
|
|
|
|
chSequentialStreamPut(getConsoleChannel(), (uint8_t )(x));
|
|
|
|
}
|
|
|
|
|
|
|
|
// 10 seconds
|
|
|
|
#define CONSOLE_WRITE_TIMEOUT 10000
|
|
|
|
|
|
|
|
void consoleOutputBuffer(const uint8_t *buf, int size) {
|
|
|
|
lastWriteSize = size;
|
|
|
|
#if !EFI_UART_ECHO_TEST_MODE
|
|
|
|
lastWriteActual = chnWriteTimeout(getConsoleChannel(), buf, size, CONSOLE_WRITE_TIMEOUT);
|
|
|
|
// if (r != size)
|
|
|
|
// firmwareError("Partial console write");
|
|
|
|
#endif /* EFI_UART_ECHO_TEST_MODE */
|
|
|
|
}
|
|
|
|
|
2015-03-08 14:04:54 -07:00
|
|
|
static Logging *logger;
|
|
|
|
|
|
|
|
static void switchToBinaryProtocol(void) {
|
|
|
|
scheduleMsg(logger, "switching to binary protocol");
|
2015-03-08 17:05:03 -07:00
|
|
|
consoleInBinaryMode = true;
|
2015-03-08 14:04:54 -07:00
|
|
|
}
|
|
|
|
|
2015-03-08 15:07:31 -07:00
|
|
|
void startConsole(Logging *sharedLogger, CommandHandler console_line_callback_p) {
|
2015-03-08 14:04:54 -07:00
|
|
|
logger = sharedLogger;
|
2014-08-29 07:52:33 -07:00
|
|
|
console_line_callback = console_line_callback_p;
|
|
|
|
|
2015-01-12 05:04:11 -08:00
|
|
|
#if (defined(EFI_CONSOLE_UART_DEVICE) && ! EFI_SIMULATOR) || defined(__DOXYGEN__)
|
2014-08-29 07:52:33 -07:00
|
|
|
|
|
|
|
palSetPadMode(CONSOLE_MODE_SWITCH_PORT, CONSOLE_MODE_SWITCH_PIN, PAL_MODE_INPUT_PULLUP);
|
|
|
|
|
|
|
|
is_serial_over_uart = GET_CONSOLE_MODE_VALUE() == EFI_USE_UART_FOR_CONSOLE;
|
|
|
|
|
|
|
|
if (isSerialOverUart()) {
|
|
|
|
/*
|
|
|
|
* Activates the serial using the driver default configuration (that's 38400)
|
|
|
|
* it is important to set 'NONE' as flow control! in terminal application on the PC
|
|
|
|
*/
|
|
|
|
sdStart(EFI_CONSOLE_UART_DEVICE, &serialConfig);
|
|
|
|
|
|
|
|
// cannot use pin repository here because pin repository prints to console
|
|
|
|
palSetPadMode(EFI_CONSOLE_RX_PORT, EFI_CONSOLE_RX_PIN, PAL_MODE_ALTERNATE(EFI_CONSOLE_AF));
|
|
|
|
palSetPadMode(EFI_CONSOLE_TX_PORT, EFI_CONSOLE_TX_PIN, PAL_MODE_ALTERNATE(EFI_CONSOLE_AF));
|
|
|
|
|
2014-12-07 16:03:14 -08:00
|
|
|
isSerialConsoleStarted = true;
|
2014-08-29 07:52:33 -07:00
|
|
|
|
|
|
|
chEvtRegisterMask((EventSource *) chnGetEventSource(EFI_CONSOLE_UART_DEVICE), &consoleEventListener, 1);
|
|
|
|
}
|
2015-01-08 09:03:39 -08:00
|
|
|
#else
|
|
|
|
is_serial_over_uart = false;
|
2014-08-29 07:52:33 -07:00
|
|
|
#endif /* EFI_PROD_CODE */
|
2014-12-07 16:03:14 -08:00
|
|
|
|
2014-08-29 07:52:33 -07:00
|
|
|
chThdCreateStatic(consoleThreadStack, sizeof(consoleThreadStack), NORMALPRIO, consoleThreadThreadEntryPoint, NULL);
|
2015-03-08 20:04:55 -07:00
|
|
|
addConsoleAction(BINARY_COMMAND, switchToBinaryProtocol);
|
2014-08-29 07:52:33 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return TRUE if already in locked context
|
|
|
|
*/
|
|
|
|
bool lockAnyContext(void) {
|
|
|
|
int alreadyLocked = isLocked();
|
|
|
|
if (alreadyLocked)
|
2015-01-14 15:04:00 -08:00
|
|
|
return true;
|
2014-08-29 07:52:33 -07:00
|
|
|
if (isIsrContext()) {
|
|
|
|
chSysLockFromIsr()
|
|
|
|
;
|
|
|
|
} else {
|
|
|
|
chSysLock()
|
|
|
|
;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool lockOutputBuffer(void) {
|
|
|
|
return lockAnyContext();
|
|
|
|
}
|
|
|
|
|
|
|
|
void unlockAnyContext(void) {
|
|
|
|
if (isIsrContext()) {
|
|
|
|
chSysUnlockFromIsr()
|
|
|
|
;
|
|
|
|
} else {
|
|
|
|
chSysUnlock()
|
|
|
|
;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void unlockOutputBuffer(void) {
|
|
|
|
unlockAnyContext();
|
|
|
|
}
|