/** * @file bluetooth.cpp * * * It looks like Bluetooth modules arrive in all kinds of initial configuration. * Sometimes we need to execute a one-time initialization including settings the baud rate. rusEFI setting uartConsoleSerialSpeed or tunerStudioSerialSpeed * has to match BT module configuration. * * * @author andreika, (c) 2017 */ #include "pch.h" #include "tunerstudio.h" #include "tunerstudio_io.h" #include "bluetooth.h" #if EFI_BLUETOOTH_SETUP static bool btProcessIsStarted = false; static const char *commands[5]; static int numCommands = 0; static int setBaudIdx = -1; static char cmdHello[30]; static char cmdBaud[25]; static char cmdName[30]; static char cmdPin[16]; // JDY-33 has 9: 128000 which we do not static const int baudRates[] = { 0, 1200, 2400, 4800, 9600, 19200, 38400, 57600, 115200, -1 }; static const int baudRateIndexList[] = { 4 /*9600*/, 6 /*38400*/, 8 /*115200*/, 7, 5, 3, 2, 1, -1 }; static const int btModuleTimeout = TIME_MS2I(1000); static SerialTsChannelBase *tsChannel; static THD_WORKING_AREA(btThreadStack, UTILITY_THREAD_STACK_SIZE); static thread_t *btThread = nullptr; static thread_reference_t btThreadRef = nullptr; // used by thread suspend/resume as a flag // Main communication code // We assume that the user has disconnected the software before starting the code. static void runCommands() { uint8_t buffer[2]; if (!btProcessIsStarted) return; efiPrintf("Sleeping..."); chThdSleepMilliseconds(1000); // safety // Store current serial port speed - we're going to change it int savedSerialSpeed = engineConfiguration->tunerStudioSerialSpeed; int prevBaudIdx = -1, baudIdx = -1, baudListIdx = 0; int cmdIdx = 0; // run all commands while (true) { if (baudIdx < 0) { // try all available baud rates, in order of preference baudIdx = baudRateIndexList[baudListIdx++]; // we've tried all baud rates? if (baudIdx < 0) break; } bool restoreAndExit = (cmdIdx >= numCommands || baudIdx < 0) || chThdShouldTerminateX(); // if the baud rate is changed, reinit the UART if (baudIdx != prevBaudIdx || restoreAndExit) { tsChannel->stop(); chThdSleepMilliseconds(10); // safety // change the port speed engineConfiguration->tunerStudioSerialSpeed = restoreAndExit ? savedSerialSpeed : baudRates[baudIdx]; efiPrintf("Restarting at %d", engineConfiguration->tunerStudioSerialSpeed); tsChannel->start(engineConfiguration->tunerStudioSerialSpeed); chThdSleepMilliseconds(10); // safety prevBaudIdx = baudIdx; } else { efiPrintf("Running at %d", engineConfiguration->tunerStudioSerialSpeed); } // exit if all commands were sent if (restoreAndExit) break; const char * command = commands[cmdIdx]; efiPrintf("Sending %s", command); // send current command tsChannel->write((uint8_t*)command, strlen(command)); // waiting for an answer bool wasAnswer = false; int received = tsChannel->readTimeout(buffer, 2, btModuleTimeout); if (received == 2) { efiPrintf("received %d byte(s) %d [%c][%c]", received, wasAnswer, buffer[0], buffer[1]); wasAnswer = (buffer[0] == 'O' && buffer[1] == 'K') || (buffer[0] == '+' && (buffer[1] >= 'A' && buffer[1] <= 'Z')); } else { efiPrintf("timeout receiving BT reply"); } // wait 1 second and skip all remaining response bytes from the bluetooth module while (true) { if (tsChannel->readTimeout(buffer, 1, btModuleTimeout) < 1) break; } if (wasAnswer) { // if we changed the baud rate if (commands[cmdIdx] == cmdBaud) { // this is what we've set baudIdx = setBaudIdx; // if we fail somehow, try all other baud rates baudListIdx = 0; } // move to the next command cmdIdx++; } else { // try other baud rate baudIdx = -1; } } // the connection is already restored to the current baud rate, so print the result if (cmdIdx == numCommands) { efiPrintf("SUCCESS! All commands (%d of %d) passed to the Bluetooth module!", cmdIdx, numCommands); } else { efiPrintf("FAIL! Only %d commands (of %d total) were passed to the Bluetooth module!", cmdIdx, numCommands); } } static THD_FUNCTION(btThreadEntryPoint, arg) { (void) arg; chRegSetThreadName("bluetooth thread"); efiPrintf("*** Bluetooth module setup procedure ***"); efiPrintf("!Warning! Please make sure you're not currently using the BT module for communication (not paired)!"); efiPrintf("TO START THE PROCEDURE: PLEASE DISCONNECT YOUR PC COM-PORT FROM THE BOARD NOW!"); efiPrintf("After that please don't turn off the board power and wait for ~15 seconds to complete. Then reconnect to the board!"); // now wait chSysLock(); msg_t msg = chThdSuspendTimeoutS(&btThreadRef, BLUETOOTH_COMMAND_TIMEOUT); chSysUnlock(); if (msg == MSG_TIMEOUT) { // timeout waiting for silence on uart... efiPrintf("The Bluetooth module init procedure is cancelled (timeout)!"); return; } else { // call this when the thread is resumed (after the disconnect) runCommands(); } // release the command btProcessIsStarted = false; } void bluetoothStart(bluetooth_module_e moduleType, const char *baudRate, const char *name, const char *pinCode) { static const char *usage = "Usage: bluetooth_hc06 "; tsChannel = getBluetoothChannel(); if (tsChannel == nullptr) { efiPrintf("No Bluetooth channel configured! Check your board config."); return; } if (btProcessIsStarted) { efiPrintf("The Bluetooth module init procedure is already started and waiting! To cancel it, run \"bluetooth_cancel\" command!"); return; } numCommands = 0; // now check the arguments and add other commands: // 1) baud rate int baud = (baudRate != NULL) ? atoi(baudRate) : 0; int i; // find a known baud rate in our list setBaudIdx = -1; for (i = 1; baudRates[i] > 0; i++) { if (baudRates[i] == baud) { setBaudIdx = i; break; } } // check the baud rate index if (setBaudIdx < 1) { if (baud == 0) efiPrintf("The parameter is set to zero! The baud rate won't be set!"); else { // unknown baud rate efiPrintf("Wrong parameter '%s'! %s", baudRate, usage); return; } } else { // ok, add command! } // 2) check name if (name == NULL || strlen(name) < 1 || strlen(name) > 20) { efiPrintf("Wrong parameter! Up to 20 characters expected! %s", usage); return; } // 3) check pin code int numDigits = 0; // check the pincode if (pinCode != NULL && strlen(pinCode) == 4) { for (i = 0; i < 4; i++) { if (pinCode[i] >= '0' && pinCode[i] <= '9') // digit numDigits++; } } if (numDigits != 4) { efiPrintf("Wrong parameter! 4 digits expected! %s", usage); return; } // ok, add commands! switch (moduleType) { case BLUETOOTH_HC_05: chsnprintf(cmdHello, sizeof(cmdHello), "AT\r\n"); chsnprintf(cmdBaud, sizeof(cmdBaud), "AT+UART=%d,0,0\r\n", baud); // baud rate, 0=(1 stop bit), 0=(no parity bits) chsnprintf(cmdName, sizeof(cmdName), "AT+NAME=%s\r\n", name); chsnprintf(cmdPin, sizeof(cmdPin), "AT+PSWD=%s\r\n", pinCode); // todo: add more commands? // AT+RMAAD // AT+ROLE=0 break; case BLUETOOTH_HC_06: chsnprintf(cmdHello, sizeof(cmdHello), "AT"); chsnprintf(cmdBaud, sizeof(cmdBaud), "AT+BAUD%c", '0' + setBaudIdx); chsnprintf(cmdName, sizeof(cmdName), "AT+NAME%s", name); chsnprintf(cmdPin, sizeof(cmdPin), "AT+PIN%s", pinCode); break; case BLUETOOTH_BK3231: chsnprintf(cmdHello, sizeof(cmdHello), "AT\r\n"); chsnprintf(cmdBaud, sizeof(cmdBaud), "AT+BAUD%c\r\n", '0' + setBaudIdx); chsnprintf(cmdName, sizeof(cmdName), "AT+NAME%s\r\n", name); chsnprintf(cmdPin, sizeof(cmdPin), "AT+PIN%s\r\n", pinCode); break; case BLUETOOTH_JDY_3x: chsnprintf(cmdHello, sizeof(cmdHello), "AT+VERSION\r\n"); chsnprintf(cmdBaud, sizeof(cmdBaud), "AT+BAUD%c\r\n", '0' + setBaudIdx); chsnprintf(cmdName, sizeof(cmdName), "AT+NAME%s\r\n", name); chsnprintf(cmdPin, sizeof(cmdPin), "AT+PIN%s\r\n", pinCode); break; default: // todo: add support for other BT module types efiPrintf("This Bluetooth module is currently not supported!"); return; } commands[numCommands++] = cmdHello; // this command is added to test a connection commands[numCommands++] = cmdBaud; commands[numCommands++] = cmdName; commands[numCommands++] = cmdPin; btProcessIsStarted = true; // create a thread to execute these commands after TS disconnected // See bluetoothSoftwareDisconnectNotify btThread = chThdCreateStatic(btThreadStack, sizeof(btThreadStack), PRIO_CONSOLE, (tfunc_t)btThreadEntryPoint, NULL); } // Called after 1S of silence on BT UART... void bluetoothSoftwareDisconnectNotify() { if (btProcessIsStarted) { // start communication with the module chThdResume(&btThreadRef, MSG_OK); // wait the thread to finish chThdWait(btThread); } } void bluetoothCancel() { if (!btProcessIsStarted) { efiPrintf("The Bluetooth module init procedure was not started! Nothing to cancel!"); return; } // terminate thread chThdTerminate(btThread); btProcessIsStarted = false; efiPrintf("The Bluetooth module init procedure is cancelled!"); } #endif /* EFI_BLUETOOTH_SETUP */