/* * 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 "opentx.h" #include "io/frsky_firmware_update.h" #include "io/multi_firmware_update.h" #include "libopenui/src/libopenui_file.h" #define NODE_TYPE(fname) fname[SD_SCREEN_FILE_LENGTH+1] #define IS_DIRECTORY(fname) ((bool)(!NODE_TYPE(fname))) #define IS_FILE(fname) ((bool)(NODE_TYPE(fname))) inline void REFRESH_FILES() { reusableBuffer.sdManager.offset = 65535; } void menuRadioSdManagerInfo(event_t event) { SIMPLE_SUBMENU(STR_SD_INFO_TITLE, 1); lcdDrawTextAlignedLeft(2*FH, STR_SD_TYPE); lcdDrawText(10*FW, 2*FH, SD_IS_HC() ? STR_SDHC_CARD : STR_SD_CARD); lcdDrawTextAlignedLeft(3*FH, STR_SD_SIZE); lcdDrawNumber(10*FW, 3*FH, sdGetSize(), LEFT); lcdDrawChar(lcdLastRightPos, 3*FH, 'M'); lcdDrawTextAlignedLeft(4*FH, STR_SD_SECTORS); #if defined(SD_GET_FREE_BLOCKNR) lcdDrawNumber(10*FW, 4*FH, SD_GET_FREE_BLOCKNR()/1000, LEFT); lcdDrawChar(lcdLastRightPos, 4*FH, '/'); lcdDrawNumber(lcdLastRightPos+FW, 4*FH, sdGetNoSectors()/1000, LEFT); #else lcdDrawNumber(10*FW, 4*FH, sdGetNoSectors()/1000, LEFT); #endif lcdDrawChar(lcdLastRightPos, 4*FH, 'k'); lcdDrawTextAlignedLeft(5*FH, STR_SD_SPEED); lcdDrawNumber(10*FW, 5*FH, SD_GET_SPEED()/1000, LEFT); lcdDrawText(lcdLastRightPos, 5*FH, "kb/s"); } inline bool isFilenameGreater(bool isfile, const char * fn, const char * line) { return (isfile && IS_DIRECTORY(line)) || (isfile==IS_FILE(line) && strcasecmp(fn, line) > 0); } inline bool isFilenameLower(bool isfile, const char * fn, const char * line) { return (!isfile && IS_FILE(line)) || (isfile==IS_FILE(line) && strcasecmp(fn, line) < 0); } void getSelectionFullPath(char * lfn) { f_getcwd(lfn, FF_MAX_LFN); strcat(lfn, "/"); strcat(lfn, reusableBuffer.sdManager.lines[menuVerticalPosition - HEADER_LINE - menuVerticalOffset]); } void onSdFormatConfirm(const char * result) { if (result == STR_OK) { showMessageBox(STR_FORMATTING); logsClose(); #if defined(PCBSKY9X) Card_state = SD_ST_DATA; #endif audioQueue.stopSD(); if (sdCardFormat()) { f_chdir(ROOT_PATH); REFRESH_FILES(); } } } #if defined(PXX2) void onUpdateConfirmation(const char * result) { if (result == STR_OK) { OtaUpdateInformation * destination = moduleState[EXTERNAL_MODULE].otaUpdateInformation; Pxx2OtaUpdate otaUpdate(EXTERNAL_MODULE, destination->candidateReceiversNames[destination->selectedReceiverIndex]); otaUpdate.flashFirmware(destination->filename, drawProgressScreen); } else { moduleState[EXTERNAL_MODULE].mode = MODULE_MODE_NORMAL; } } void onUpdateStateChanged() { if (reusableBuffer.sdManager.otaUpdateInformation.step == BIND_INFO_REQUEST) { uint8_t modelId = reusableBuffer.sdManager.otaUpdateInformation.receiverInformation.modelID; if (modelId > 0 && modelId < DIM(PXX2ReceiversNames)) { if (isPXX2ReceiverOptionAvailable(modelId, RECEIVER_OPTION_OTA)) { POPUP_CONFIRMATION(getPXX2ReceiverName(modelId), onUpdateConfirmation); char *tmp = strAppend(reusableBuffer.sdManager.otaReceiverVersion, TR_CURRENT_VERSION); tmp = strAppendUnsigned(tmp, 1 + reusableBuffer.sdManager.otaUpdateInformation.receiverInformation.swVersion.major); *tmp++ = '.'; tmp = strAppendUnsigned(tmp, reusableBuffer.sdManager.otaUpdateInformation.receiverInformation.swVersion.minor); *tmp++ = '.'; tmp = strAppendUnsigned(tmp, reusableBuffer.sdManager.otaUpdateInformation.receiverInformation.swVersion.revision); SET_WARNING_INFO(reusableBuffer.sdManager.otaReceiverVersion, tmp - reusableBuffer.sdManager.otaReceiverVersion, 0); } else { POPUP_WARNING(STR_OTA_UPDATE_ERROR); SET_WARNING_INFO(STR_UNSUPPORTED_RX, sizeof(TR_UNSUPPORTED_RX) - 1, 0); moduleState[EXTERNAL_MODULE].mode = MODULE_MODE_NORMAL; } } else { POPUP_WARNING(STR_OTA_UPDATE_ERROR); SET_WARNING_INFO(STR_UNKNOWN_RX, sizeof(TR_UNKNOWN_RX) - 1, 0); moduleState[EXTERNAL_MODULE].mode = MODULE_MODE_NORMAL; } } } #endif void onSdManagerMenu(const char * result) { TCHAR lfn[FF_MAX_LFN+1]; // TODO possible buffer overflows here! uint8_t index = menuVerticalPosition - HEADER_LINE - menuVerticalOffset; char * line = reusableBuffer.sdManager.lines[index]; if (result == STR_SD_INFO) { pushMenu(menuRadioSdManagerInfo); } else if (result == STR_SD_FORMAT) { POPUP_CONFIRMATION(STR_CONFIRM_FORMAT, onSdFormatConfirm); } else if (result == STR_COPY_FILE) { clipboard.type = CLIPBOARD_TYPE_SD_FILE; f_getcwd(clipboard.data.sd.directory, CLIPBOARD_PATH_LEN); strncpy(clipboard.data.sd.filename, line, CLIPBOARD_PATH_LEN-1); } else if (result == STR_PASTE) { f_getcwd(lfn, FF_MAX_LFN); // if destination is dir, copy into that dir if (IS_DIRECTORY(line)) { strcat(lfn, "/"); strcat(lfn, line); } if (strcmp(clipboard.data.sd.directory, lfn)) { // prevent copying to the same directory POPUP_WARNING(sdCopyFile(clipboard.data.sd.filename, clipboard.data.sd.directory, clipboard.data.sd.filename, lfn)); REFRESH_FILES(); } } else if (result == STR_RENAME_FILE) { memcpy(reusableBuffer.sdManager.originalName, line, sizeof(reusableBuffer.sdManager.originalName)); uint8_t fnlen = 0, extlen = 0; getFileExtension(line, 0, LEN_FILE_EXTENSION_MAX, &fnlen, &extlen); // write spaces to allow extending the length of a filename memset(line + fnlen - extlen, ' ', SD_SCREEN_FILE_LENGTH - fnlen + extlen); line[SD_SCREEN_FILE_LENGTH-extlen] = '\0'; s_editMode = EDIT_MODIFY_STRING; editNameCursorPos = 0; } else if (result == STR_DELETE_FILE) { getSelectionFullPath(lfn); f_unlink(lfn); strncpy(statusLineMsg, line, 13); strcpy(statusLineMsg+min((uint8_t)strlen(statusLineMsg), (uint8_t)13), STR_REMOVED); showStatusLine(); REFRESH_FILES(); } else if (result == STR_PLAY_FILE) { getSelectionFullPath(lfn); audioQueue.stopAll(); audioQueue.playFile(lfn, 0, ID_PLAY_FROM_SD_MANAGER); } #if LCD_DEPTH > 1 else if (result == STR_ASSIGN_BITMAP) { strAppendFilename(g_model.header.bitmap, line, sizeof(g_model.header.bitmap)); memcpy(modelHeaders[g_eeGeneral.currModel].bitmap, g_model.header.bitmap, sizeof(g_model.header.bitmap)); storageDirty(EE_MODEL); } #endif else if (result == STR_VIEW_TEXT) { getSelectionFullPath(lfn); pushMenuTextView(lfn); } #if defined(PCBTARANIS) else if (result == STR_FLASH_BOOTLOADER) { getSelectionFullPath(lfn); bootloaderFlash(lfn); } else if (result == STR_FLASH_INTERNAL_MODULE) { getSelectionFullPath(lfn); FrskyDeviceFirmwareUpdate device(INTERNAL_MODULE); device.flashFirmware(lfn, drawProgressScreen); } else if (result == STR_FLASH_EXTERNAL_MODULE) { // needed on X-Lite (as the R9M needs 2S while the external device flashing port only provides 5V) getSelectionFullPath(lfn); FrskyDeviceFirmwareUpdate device(EXTERNAL_MODULE); device.flashFirmware(lfn, drawProgressScreen); } else if (result == STR_FLASH_EXTERNAL_DEVICE) { getSelectionFullPath(lfn); FrskyDeviceFirmwareUpdate device(SPORT_MODULE); device.flashFirmware(lfn, drawProgressScreen); } #if defined(MULTIMODULE) #if defined(INTERNAL_MODULE_MULTI) else if (result == STR_FLASH_INTERNAL_MULTI) { getSelectionFullPath(lfn); multiFlashFirmware(INTERNAL_MODULE, lfn); } #endif else if (result == STR_FLASH_EXTERNAL_MULTI) { getSelectionFullPath(lfn); multiFlashFirmware(EXTERNAL_MODULE, lfn); } #endif #if defined(BLUETOOTH) else if (result == STR_FLASH_BLUETOOTH_MODULE) { getSelectionFullPath(lfn); bluetooth.flashFirmware(lfn, drawProgressScreen); } #endif #if defined(HARDWARE_POWER_MANAGEMENT_UNIT) else if (result == STR_FLASH_POWER_MANAGEMENT_UNIT) { getSelectionFullPath(lfn); FrskyChipFirmwareUpdate device; device.flashFirmware(lfn, drawProgressScreen); } #endif #if defined(PXX2) else if (result == STR_FLASH_RECEIVER_OTA) { memclear(&reusableBuffer.sdManager.otaUpdateInformation, sizeof(OtaUpdateInformation)); getSelectionFullPath(reusableBuffer.sdManager.otaUpdateInformation.filename); moduleState[EXTERNAL_MODULE].startBind(&reusableBuffer.sdManager.otaUpdateInformation, onUpdateStateChanged); } #endif #endif #if defined(LUA) else if (result == STR_EXECUTE_FILE) { getSelectionFullPath(lfn); luaExec(lfn); } #endif } #if defined(PXX2) void onUpdateReceiverSelection(const char * result) { if (result != STR_EXIT) { reusableBuffer.sdManager.otaUpdateInformation.selectedReceiverIndex = (result - reusableBuffer.sdManager.otaUpdateInformation.candidateReceiversNames[0]) / sizeof(reusableBuffer.sdManager.otaUpdateInformation.candidateReceiversNames[0]); reusableBuffer.sdManager.otaUpdateInformation.step = BIND_INFO_REQUEST; #if defined(SIMU) reusableBuffer.sdManager.otaUpdateInformation.receiverInformation.modelID = 0x01; onUpdateStateChanged(); #endif } else { // the user pressed [Exit] moduleState[EXTERNAL_MODULE].mode = MODULE_MODE_NORMAL; } } #endif void menuRadioSdManager(event_t _event) { #if LCD_DEPTH > 1 int lastPos = menuVerticalPosition; #endif if (moduleState[EXTERNAL_MODULE].mode == MODULE_MODE_BIND && EVT_KEY_MASK(_event) == KEY_EXIT) { moduleState[EXTERNAL_MODULE].mode = MODULE_MODE_NORMAL; CLEAR_POPUP(); killEvents(KEY_EXIT); _event = 0; } event_t event = (EVT_KEY_MASK(_event) == KEY_ENTER ? 0 : _event); SIMPLE_MENU(SD_IS_HC() ? STR_SDHC_CARD : STR_SD_CARD, menuTabGeneral, MENU_RADIO_SD_MANAGER, HEADER_LINE + reusableBuffer.sdManager.count); switch (_event) { case EVT_ENTRY: f_chdir(ROOT_PATH); REFRESH_FILES(); #if LCD_DEPTH > 1 lastPos = -1; #endif break; case EVT_ENTRY_UP: REFRESH_FILES(); break; #if defined(PCBX9) || defined(RADIO_X7) // TODO NO_MENU_KEY case EVT_KEY_LONG(KEY_MENU): if (SD_CARD_PRESENT() && !READ_ONLY() && s_editMode == 0) { killEvents(_event); POPUP_MENU_ADD_ITEM(STR_SD_INFO); POPUP_MENU_ADD_ITEM(STR_SD_FORMAT); POPUP_MENU_START(onSdManagerMenu); } break; #endif case EVT_KEY_BREAK(KEY_EXIT): REFRESH_FILES(); break; #if !defined(PCBTARANIS) case EVT_KEY_FIRST(KEY_RIGHT): #endif case EVT_KEY_BREAK(KEY_ENTER): if (s_editMode > 0) { break; } else { int index = menuVerticalPosition - HEADER_LINE - menuVerticalOffset; if (IS_DIRECTORY(reusableBuffer.sdManager.lines[index])) { f_chdir(reusableBuffer.sdManager.lines[index]); menuVerticalOffset = 0; menuVerticalPosition = HEADER_LINE; REFRESH_FILES(); killEvents(_event); return; } } break; case EVT_KEY_LONG(KEY_ENTER): #if !defined(PCBX9) && !defined(RADIO_X7) // TODO NO_HEADER_LINE if (menuVerticalPosition < HEADER_LINE) { killEvents(_event); POPUP_MENU_ADD_ITEM(STR_SD_INFO); POPUP_MENU_ADD_ITEM(STR_SD_FORMAT); POPUP_MENU_START(onSdManagerMenu); break; } #endif TCHAR lfn[FF_MAX_LFN + 1]; getSelectionFullPath(lfn); if (SD_CARD_PRESENT() && s_editMode <= 0) { killEvents(_event); int index = menuVerticalPosition - HEADER_LINE - menuVerticalOffset; char * line = reusableBuffer.sdManager.lines[index]; if (!strcmp(line, "..")) { break; // no menu for parent dir } const char * ext = getFileExtension(line); if (ext) { if (!strcasecmp(ext, SOUNDS_EXT)) { POPUP_MENU_ADD_ITEM(STR_PLAY_FILE); } #if LCD_DEPTH > 1 else if (isExtensionMatching(ext, BITMAPS_EXT)) { if (!READ_ONLY() && (ext-line) <= (int)sizeof(g_model.header.bitmap)) { POPUP_MENU_ADD_ITEM(STR_ASSIGN_BITMAP); } } #endif #if defined(LUA) else if (isExtensionMatching(ext, SCRIPTS_EXT)) { POPUP_MENU_ADD_ITEM(STR_EXECUTE_FILE); } #endif #if defined(MULTIMODULE) && !defined(DISABLE_MULTI_UPDATE) if (!READ_ONLY() && !strcasecmp(ext, MULTI_FIRMWARE_EXT)) { MultiFirmwareInformation information; if (information.readMultiFirmwareInformation(line) == nullptr) { #if defined(INTERNAL_MODULE_MULTI) POPUP_MENU_ADD_ITEM(STR_FLASH_INTERNAL_MULTI); #endif POPUP_MENU_ADD_ITEM(STR_FLASH_EXTERNAL_MULTI); } } #endif #if defined(PCBTARANIS) if (!READ_ONLY() && !strcasecmp(ext, FIRMWARE_EXT)) { if (isBootloader(lfn)) { POPUP_MENU_ADD_ITEM(STR_FLASH_BOOTLOADER); } } else if (!READ_ONLY() && !strcasecmp(ext, SPORT_FIRMWARE_EXT)) { if (HAS_SPORT_UPDATE_CONNECTOR()) POPUP_MENU_ADD_ITEM(STR_FLASH_EXTERNAL_DEVICE); POPUP_MENU_ADD_ITEM(STR_FLASH_INTERNAL_MODULE); POPUP_MENU_ADD_ITEM(STR_FLASH_EXTERNAL_MODULE); } else if (!READ_ONLY() && !strcasecmp(ext, FRSKY_FIRMWARE_EXT)) { FrSkyFirmwareInformation information; if (readFrSkyFirmwareInformation(line, information) == nullptr) { if (information.productFamily == FIRMWARE_FAMILY_INTERNAL_MODULE) POPUP_MENU_ADD_ITEM(STR_FLASH_INTERNAL_MODULE); if (information.productFamily == FIRMWARE_FAMILY_EXTERNAL_MODULE) POPUP_MENU_ADD_ITEM(STR_FLASH_EXTERNAL_MODULE); if (information.productFamily == FIRMWARE_FAMILY_RECEIVER || information.productFamily == FIRMWARE_FAMILY_SENSOR) { if (HAS_SPORT_UPDATE_CONNECTOR()) POPUP_MENU_ADD_ITEM(STR_FLASH_EXTERNAL_DEVICE); else POPUP_MENU_ADD_ITEM(STR_FLASH_EXTERNAL_MODULE); } #if defined(PXX2) if (information.productFamily == FIRMWARE_FAMILY_RECEIVER) POPUP_MENU_ADD_ITEM(STR_FLASH_RECEIVER_OTA); #endif #if defined(BLUETOOTH) if (information.productFamily == FIRMWARE_FAMILY_BLUETOOTH_CHIP) POPUP_MENU_ADD_ITEM(STR_FLASH_BLUETOOTH_MODULE); #endif #if defined(HARDWARE_POWER_MANAGEMENT_UNIT) if (information.productFamily == FIRMWARE_FAMILY_POWER_MANAGEMENT_UNIT) POPUP_MENU_ADD_ITEM(STR_FLASH_POWER_MANAGEMENT_UNIT); #endif } } #endif if (isExtensionMatching(ext, TEXT_EXT) || isExtensionMatching(ext, SCRIPTS_EXT)) { POPUP_MENU_ADD_ITEM(STR_VIEW_TEXT); } } if (!READ_ONLY()) { if (IS_FILE(line)) POPUP_MENU_ADD_ITEM(STR_COPY_FILE); if (clipboard.type == CLIPBOARD_TYPE_SD_FILE) POPUP_MENU_ADD_ITEM(STR_PASTE); POPUP_MENU_ADD_ITEM(STR_RENAME_FILE); if (IS_FILE(line)) POPUP_MENU_ADD_ITEM(STR_DELETE_FILE); } POPUP_MENU_START(onSdManagerMenu); } break; } if (SD_CARD_PRESENT()) { if (reusableBuffer.sdManager.offset != menuVerticalOffset) { FILINFO fno; DIR dir; if (menuVerticalOffset == reusableBuffer.sdManager.offset + 1) { memmove(reusableBuffer.sdManager.lines[0], reusableBuffer.sdManager.lines[1], (NUM_BODY_LINES-1)*sizeof(reusableBuffer.sdManager.lines[0])); memset(reusableBuffer.sdManager.lines[NUM_BODY_LINES-1], 0xff, SD_SCREEN_FILE_LENGTH); NODE_TYPE(reusableBuffer.sdManager.lines[NUM_BODY_LINES-1]) = 1; } else if (menuVerticalOffset == reusableBuffer.sdManager.offset - 1) { memmove(reusableBuffer.sdManager.lines[1], reusableBuffer.sdManager.lines[0], (NUM_BODY_LINES-1)*sizeof(reusableBuffer.sdManager.lines[0])); memset(reusableBuffer.sdManager.lines[0], 0, sizeof(reusableBuffer.sdManager.lines[0])); } else { reusableBuffer.sdManager.offset = menuVerticalOffset; memset(reusableBuffer.sdManager.lines, 0, sizeof(reusableBuffer.sdManager.lines)); } reusableBuffer.sdManager.count = 0; FRESULT res = f_opendir(&dir, "."); // Open the directory if (res == FR_OK) { bool firstTime = true; for (;;) { res = sdReadDir(&dir, &fno, firstTime); if (res != FR_OK || fno.fname[0] == 0) break; /* Break on error or end of dir */ if (strlen(fno.fname) > SD_SCREEN_FILE_LENGTH) continue; if (fno.fattrib & AM_HID) continue; /* Ignore Windows hidden files */ if (fno.fname[0] == '.' && fno.fname[1] != '.') continue; /* Ignore UNIX hidden files, but not .. */ reusableBuffer.sdManager.count++; bool isfile = !(fno.fattrib & AM_DIR); if (menuVerticalOffset == 0) { for (uint8_t i=0; i=0; i--) { char * line = reusableBuffer.sdManager.lines[i]; if (line[0] == '\0' || isFilenameGreater(isfile, fno.fname, line)) { if (i > 0) memmove(reusableBuffer.sdManager.lines[0], reusableBuffer.sdManager.lines[1], sizeof(reusableBuffer.sdManager.lines[0]) * i); memset(line, 0, sizeof(reusableBuffer.sdManager.lines[0])); strcpy(line, fno.fname); NODE_TYPE(line) = isfile; break; } } } else if (menuVerticalOffset > reusableBuffer.sdManager.offset) { if (isFilenameGreater(isfile, fno.fname, reusableBuffer.sdManager.lines[NUM_BODY_LINES-2]) && isFilenameLower(isfile, fno.fname, reusableBuffer.sdManager.lines[NUM_BODY_LINES-1])) { memset(reusableBuffer.sdManager.lines[NUM_BODY_LINES-1], 0, sizeof(reusableBuffer.sdManager.lines[0])); strcpy(reusableBuffer.sdManager.lines[NUM_BODY_LINES-1], fno.fname); NODE_TYPE(reusableBuffer.sdManager.lines[NUM_BODY_LINES-1]) = isfile; } } else { if (isFilenameLower(isfile, fno.fname, reusableBuffer.sdManager.lines[1]) && isFilenameGreater(isfile, fno.fname, reusableBuffer.sdManager.lines[0])) { memset(reusableBuffer.sdManager.lines[0], 0, sizeof(reusableBuffer.sdManager.lines[0])); strcpy(reusableBuffer.sdManager.lines[0], fno.fname); NODE_TYPE(reusableBuffer.sdManager.lines[0]) = isfile; } } } f_closedir(&dir); } } reusableBuffer.sdManager.offset = menuVerticalOffset; int index = menuVerticalPosition - HEADER_LINE - menuVerticalOffset; for (uint8_t i=0; i 0) { if(reusableBuffer.sdManager.otaUpdateInformation.candidateReceiversCount != popupMenuItemsCount) { CLEAR_POPUP(); popupMenuItemsCount = min(reusableBuffer.sdManager.otaUpdateInformation.candidateReceiversCount,PXX2_MAX_RECEIVERS_PER_MODULE); for (auto rx = 0; rx < popupMenuItemsCount; rx++) { popupMenuItems[rx] = reusableBuffer.sdManager.otaUpdateInformation.candidateReceiversNames[rx]; } popupMenuTitle = STR_PXX2_SELECT_RX; POPUP_MENU_START(onUpdateReceiverSelection); } } else { POPUP_WAIT(STR_WAITING_FOR_RX); } } } #endif #if LCD_DEPTH > 1 const char * ext = getFileExtension(reusableBuffer.sdManager.lines[index]); if (ext && isExtensionMatching(ext, BITMAPS_EXT)) { if (lastPos != menuVerticalPosition) { if (!lcdLoadBitmap(modelBitmap, reusableBuffer.sdManager.lines[index], MODEL_BITMAP_WIDTH, MODEL_BITMAP_HEIGHT)) { memcpy(modelBitmap, logo_taranis, MODEL_BITMAP_SIZE); } } lcdDrawBitmap(22*FW+2, 2*FH+FH/2, modelBitmap); } #endif } else { lcdDrawCenteredText(LCD_H/2, STR_NO_SDCARD); REFRESH_FILES(); } }