/* * Copyright (C) OpenTX * * Based on code named * th9x - http://code.google.com/p/th9x * er9x - http://code.google.com/p/er9x * gruvin9x - http://code.google.com/p/gruvin9x * * License GPLv2: http://www.gnu.org/licenses/gpl-2.0.html * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. * * 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. */ #include uint8_t telemetryStreaming = 0; uint8_t R9ModuleStreaming = 0; uint8_t telemetryRxBuffer[TELEMETRY_RX_PACKET_SIZE]; // Receive buffer. 9 bytes (full packet), worst case 18 bytes with byte-stuffing (+1) uint8_t telemetryRxBufferCount = 0; #if defined(WS_HOW_HIGH) uint8_t wshhStreaming = 0; #endif uint8_t telemetryState = TELEMETRY_INIT; TelemetryData telemetryData; uint8_t telemetryProtocol = 255; #if defined(PCBSKY9X) && defined(REVX) uint8_t serialInversion = 0; #endif void processTelemetryData(uint8_t data) { #if defined(CROSSFIRE) if (telemetryProtocol == PROTOCOL_TELEMETRY_CROSSFIRE) { processCrossfireTelemetryData(data); return; } #endif #if defined(MULTIMODULE) if (telemetryProtocol == PROTOCOL_TELEMETRY_SPEKTRUM) { processSpektrumTelemetryData(data); return; } if (telemetryProtocol == PROTOCOL_TELEMETRY_FLYSKY_IBUS) { processFlySkyTelemetryData(data); return; } if (telemetryProtocol == PROTOCOL_TELEMETRY_MULTIMODULE) { processMultiTelemetryData(data); return; } #endif if (telemetryProtocol == PROTOCOL_TELEMETRY_PXX2) { processFrskyPXX2Data(data); return; } processFrskyTelemetryData(data); } void processGetHardwareInfoFrame(uint8_t module, uint8_t * frame) { if (moduleSettings[module].mode != MODULE_MODE_GET_HARDWARE_INFO) { return; } uint8_t index = frame[3]; if (index == 0xFF) { reusableBuffer.hardware.modules[module].hw_version = *((uint16_t *)&frame[4]); reusableBuffer.hardware.modules[module].sw_version = *((uint16_t *)&frame[6]); } else if (index < PXX2_MAX_RECEIVERS_PER_MODULE){ reusableBuffer.hardware.modules[module].receivers[index].hw_version = *((uint16_t *)&frame[4]); reusableBuffer.hardware.modules[module].receivers[index].sw_version = *((uint16_t *)&frame[6]); } } void processReceiverSettingsFrame(uint8_t module, uint8_t * frame) { if (moduleSettings[module].mode != MODULE_MODE_RECEIVER_SETTINGS) { return; } for (uint8_t pin=0; pin<24; pin++) { reusableBuffer.receiverSetup.channelMapping[pin] = frame[5 + pin]; } reusableBuffer.receiverSetup.state = 0xFF; moduleSettings[module].mode = MODULE_MODE_NORMAL; } void processRegisterFrame(uint8_t module, uint8_t * frame) { if (moduleSettings[module].mode != MODULE_MODE_REGISTER) { return; } if (frame[3] == 0x00 && reusableBuffer.moduleSetup.pxx2.registerStep == REGISTER_START) { // RX_NAME follows, we store it for the next step str2zchar(reusableBuffer.moduleSetup.pxx2.registerRxName, (const char *)&frame[4], PXX2_LEN_RX_NAME); reusableBuffer.moduleSetup.pxx2.registerStep = REGISTER_RX_NAME_RECEIVED; } else if (frame[3] == 0x01 && reusableBuffer.moduleSetup.pxx2.registerStep == REGISTER_RX_NAME_SELECTED) { // RX_NAME + PASSWORD follow, we check they are good if (cmpStrWithZchar((char *)&frame[4], reusableBuffer.moduleSetup.pxx2.registerRxName, PXX2_LEN_RX_NAME) && cmpStrWithZchar((char *)&frame[12], reusableBuffer.moduleSetup.pxx2.registrationID, PXX2_LEN_REGISTRATION_ID)) { reusableBuffer.moduleSetup.pxx2.registerStep = REGISTER_OK; moduleSettings[module].mode = MODULE_MODE_NORMAL; POPUP_INFORMATION(STR_REG_OK); } } } void processBindFrame(uint8_t module, uint8_t * frame) { if (moduleSettings[module].mode != MODULE_MODE_BIND) { return; } switch(frame[3]) { case 0x00: if (reusableBuffer.moduleSetup.pxx2.bindStep == BIND_START) { bool found = false; for (uint8_t i=0; i 0x33) static tmr10ms_t alarmsCheckTime = 0; #define SCHEDULE_NEXT_ALARMS_CHECK(seconds) alarmsCheckTime = get_tmr10ms() + (100*(seconds)) if (int32_t(get_tmr10ms() - alarmsCheckTime) > 0) { SCHEDULE_NEXT_ALARMS_CHECK(1/*second*/); bool sensor_lost = false; for (int i=0; i TELEMETRY_VALUE_OLD_THRESHOLD) { TelemetrySensor * sensor = & g_model.telemetrySensors[i]; if (sensor->unit != UNIT_DATETIME) { item.setOld(); sensor_lost = true; } } } } if (sensor_lost && TELEMETRY_STREAMING() && !g_model.rssiAlarms.disabled) { audioEvent(AU_SENSOR_LOST); } #if defined(PCBTARANIS) || defined(PCBHORUS) if ((isModulePXX(INTERNAL_MODULE) || isModulePXX(EXTERNAL_MODULE)) && FRSKY_BAD_ANTENNA()) { AUDIO_RAS_RED(); POPUP_WARNING(STR_WARNING); const char * w = STR_ANTENNAPROBLEM; SET_WARNING_INFO(w, strlen(w), 0); SCHEDULE_NEXT_ALARMS_CHECK(10/*seconds*/); } #endif if (!g_model.rssiAlarms.disabled) { if (TELEMETRY_STREAMING()) { if (TELEMETRY_RSSI() < g_model.rssiAlarms.getCriticalRssi() ) { AUDIO_RSSI_RED(); SCHEDULE_NEXT_ALARMS_CHECK(10/*seconds*/); } else if (TELEMETRY_RSSI() < g_model.rssiAlarms.getWarningRssi() ) { AUDIO_RSSI_ORANGE(); SCHEDULE_NEXT_ALARMS_CHECK(10/*seconds*/); } } if (TELEMETRY_STREAMING()) { if (telemetryState == TELEMETRY_KO) { AUDIO_TELEMETRY_BACK(); } telemetryState = TELEMETRY_OK; } else if (telemetryState == TELEMETRY_OK) { telemetryState = TELEMETRY_KO; AUDIO_TELEMETRY_LOST(); } } } } void telemetryInterrupt10ms() { if (TELEMETRY_STREAMING()) { if (!TELEMETRY_OPENXSENSOR()) { for (int i=0; i 0) { wshhStreaming--; } #endif if (R9ModuleStreaming > 0) { R9ModuleStreaming--; } if (telemetryStreaming > 0) { telemetryStreaming--; } else { #if !defined(SIMU) telemetryData.rssi.reset(); #endif } } void telemetryReset() { memclear(&telemetryData, sizeof(telemetryData)); for (int index=0; index * luaInputTelemetryFifo = NULL; #endif