mirror of
https://github.com/RfidResearchGroup/proxmark3.git
synced 2025-01-24 00:50:27 +08:00
460 lines
15 KiB
C
460 lines
15 KiB
C
//-----------------------------------------------------------------------------
|
|
// Copyright (C) 2018 iceman
|
|
//
|
|
// This code is licensed to you under the terms of the GNU GPL, version 2 or,
|
|
// at your option, any later version. See the LICENSE.txt file for the text of
|
|
// the license.
|
|
//-----------------------------------------------------------------------------
|
|
// Proxmark3 RDV40 Flash memory commands
|
|
//-----------------------------------------------------------------------------
|
|
#include "cmdflashmemspiffs.h"
|
|
|
|
#include "mbedtls/base64.h"
|
|
#include "mbedtls/rsa.h"
|
|
#include "mbedtls/sha1.h"
|
|
|
|
static int CmdHelp(const char *Cmd);
|
|
|
|
static int CmdFlashMemSpiFFSMount(const char *Cmd) {
|
|
(void)Cmd; // Cmd is not used so far
|
|
clearCommandBuffer();
|
|
SendCommandNG(CMD_SPIFFS_MOUNT, NULL, 0);
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int CmdFlashMemSpiFFSUnmount(const char *Cmd) {
|
|
(void)Cmd; // Cmd is not used so far
|
|
clearCommandBuffer();
|
|
SendCommandNG(CMD_SPIFFS_UNMOUNT, NULL, 0);
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int CmdFlashMemSpiFFSTest(const char *Cmd) {
|
|
(void)Cmd; // Cmd is not used so far
|
|
clearCommandBuffer();
|
|
SendCommandNG(CMD_SPIFFS_TEST, NULL, 0);
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int CmdFlashMemSpiFFSTree(const char *Cmd) {
|
|
(void)Cmd; // Cmd is not used so far
|
|
clearCommandBuffer();
|
|
SendCommandNG(CMD_SPIFFS_PRINT_TREE, NULL, 0);
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int CmdFlashMemSpiFFSInfo(const char *Cmd) {
|
|
(void)Cmd; // Cmd is not used so far
|
|
clearCommandBuffer();
|
|
SendCommandNG(CMD_SPIFFS_PRINT_FSINFO, NULL, 0);
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int usage_flashmemspiffs_remove(void) {
|
|
PrintAndLogEx(NORMAL, "Remove a file from spiffs filesystem");
|
|
PrintAndLogEx(NORMAL, " Usage: mem spiffs remove <filename>");
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int usage_flashmemspiffs_rename(void) {
|
|
PrintAndLogEx(NORMAL, "Rename/move a file in spiffs filesystem");
|
|
PrintAndLogEx(NORMAL, " Usage: mem spiffs rename <source> <destination>");
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int usage_flashmemspiffs_copy(void) {
|
|
PrintAndLogEx(NORMAL, "Copy a file to another (destructively) in spiffs filesystem");
|
|
PrintAndLogEx(NORMAL, " Usage: mem spiffs copy <source> <destination>");
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int usage_flashmemspiffs_dump(void) {
|
|
PrintAndLogEx(NORMAL, "Dumps flash memory on device into a file or in console");
|
|
PrintAndLogEx(NORMAL, "Size is handled by first sending a STAT command against file existence");
|
|
PrintAndLogEx(NORMAL, " Usage: mem spiffs dump o <filename> [f <file name> [e]] [p]");
|
|
PrintAndLogEx(NORMAL, " o <filename> : filename in SPIFFS");
|
|
PrintAndLogEx(NORMAL, " f <filename> : file name to save to");
|
|
PrintAndLogEx(NORMAL, " p : print dump in console");
|
|
PrintAndLogEx(NORMAL, " e : also save in EML format (good for tags save and dictonnary files)");
|
|
PrintAndLogEx(NORMAL, " You must specify at lease option f or option p, both if you wish");
|
|
PrintAndLogEx(NORMAL, "");
|
|
PrintAndLogEx(NORMAL, "Examples:");
|
|
PrintAndLogEx(NORMAL, " mem spiffs dump o hf_colin/lasttag f lasttag e");
|
|
PrintAndLogEx(NORMAL, " mem spiffs dump o hf_colin/lasttag p");
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int usage_flashmemspiffs_load(void) {
|
|
PrintAndLogEx(NORMAL, "Uploads binary-wise file into device filesystem");
|
|
PrintAndLogEx(NORMAL, "Usage: mem spiffs load o <filename> f <filename>");
|
|
PrintAndLogEx(NORMAL, "Warning: mem area to be written must have been wiped first");
|
|
PrintAndLogEx(NORMAL, "(this is already taken care when loading dictionaries)");
|
|
PrintAndLogEx(NORMAL, " o <filename> : destination filename");
|
|
PrintAndLogEx(NORMAL, " f <filename> : local filename");
|
|
PrintAndLogEx(NORMAL, "");
|
|
PrintAndLogEx(NORMAL, "Examples:");
|
|
PrintAndLogEx(NORMAL, " mem spiffs load f myfile o myapp.conf");
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int CmdFlashMemSpiFFSRemove(const char *Cmd) {
|
|
|
|
char filename[32] = {0};
|
|
bool errors = false;
|
|
|
|
/*char ctmp = tolower(param_getchar(Cmd, 0));
|
|
if (strlen(Cmd) < 1 || ctmp == 'h') {
|
|
return usage_flashmemspiffs_remove();
|
|
}*/
|
|
|
|
if (param_getstr(Cmd, 0, filename, 32) >= 32) {
|
|
PrintAndLogEx(FAILED, "Filename too long");
|
|
errors = true;
|
|
}
|
|
|
|
// check null filename ?
|
|
if (errors) {
|
|
usage_flashmemspiffs_remove();
|
|
return PM3_EINVARG;
|
|
}
|
|
|
|
SendCommandMIX(CMD_SPIFFS_REMOVE, 0, 0, 0, (uint8_t *)filename, 32);
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int CmdFlashMemSpiFFSRename(const char *Cmd) {
|
|
|
|
char srcfilename[32] = {0};
|
|
char destfilename[32] = {0};
|
|
bool errors = false;
|
|
|
|
/*char ctmp = tolower(param_getchar(Cmd, 0));
|
|
if (strlen(Cmd) < 1 || ctmp == 'h') {
|
|
return usage_flashmemspiffs_rename();
|
|
}*/
|
|
|
|
if (param_getstr(Cmd, 0, srcfilename, 32) >= 32) {
|
|
PrintAndLogEx(FAILED, "Source Filename too long");
|
|
errors = true;
|
|
}
|
|
if (srcfilename[0] == '\0') {
|
|
PrintAndLogEx(FAILED, "Source Filename missing or invalid");
|
|
errors = true;
|
|
}
|
|
|
|
if (param_getstr(Cmd, 1, destfilename, 32) >= 32) {
|
|
PrintAndLogEx(FAILED, "Source Filename too long");
|
|
errors = true;
|
|
}
|
|
if (destfilename[0] == '\0') {
|
|
PrintAndLogEx(FAILED, "Source Filename missing or invalid");
|
|
errors = true;
|
|
}
|
|
|
|
// check null filename ?
|
|
if (errors) {
|
|
usage_flashmemspiffs_rename();
|
|
return PM3_EINVARG;
|
|
}
|
|
|
|
char data[65];
|
|
sprintf(data, "%s,%s", srcfilename, destfilename);
|
|
SendCommandMIX(CMD_SPIFFS_RENAME, 0, 0, 0, (uint8_t *)data, 65);
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int CmdFlashMemSpiFFSCopy(const char *Cmd) {
|
|
|
|
char srcfilename[32] = {0};
|
|
char destfilename[32] = {0};
|
|
bool errors = false;
|
|
|
|
/*char ctmp = tolower(param_getchar(Cmd, 0));
|
|
if (strlen(Cmd) < 1 || ctmp == 'h') {
|
|
return usage_flashmemspiffs_copy();
|
|
}*/
|
|
|
|
if (param_getstr(Cmd, 0, srcfilename, 32) >= 32) {
|
|
PrintAndLogEx(FAILED, "Source Filename too long");
|
|
errors = true;
|
|
}
|
|
if (srcfilename[0] == '\0') {
|
|
PrintAndLogEx(FAILED, "Source Filename missing or invalid");
|
|
errors = true;
|
|
}
|
|
|
|
if (param_getstr(Cmd, 1, destfilename, 32) >= 32) {
|
|
PrintAndLogEx(FAILED, "Source Filename too long");
|
|
errors = true;
|
|
}
|
|
if (destfilename[0] == '\0') {
|
|
PrintAndLogEx(FAILED, "Source Filename missing or invalid");
|
|
errors = true;
|
|
}
|
|
|
|
// check null filename ?
|
|
if (errors) {
|
|
usage_flashmemspiffs_copy();
|
|
return PM3_EINVARG;
|
|
}
|
|
|
|
char data[65];
|
|
sprintf(data, "%s,%s", srcfilename, destfilename);
|
|
SendCommandMIX(CMD_SPIFFS_COPY, 0, 0, 0, (uint8_t *)data, 65);
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int CmdFlashMemSpiFFSDump(const char *Cmd) {
|
|
|
|
char filename[FILE_PATH_SIZE] = {0};
|
|
uint8_t cmdp = 0;
|
|
bool errors = false;
|
|
bool print = false;
|
|
uint32_t start_index = 0, len = FLASH_MEM_MAX_SIZE;
|
|
char destfilename[32] = {0};
|
|
bool eml = false;
|
|
|
|
while (param_getchar(Cmd, cmdp) != 0x00 && !errors) {
|
|
switch (tolower(param_getchar(Cmd, cmdp))) {
|
|
case 'h':
|
|
return usage_flashmemspiffs_dump();
|
|
/*case 'l':
|
|
len = param_get32ex(Cmd, cmdp + 1, FLASH_MEM_MAX_SIZE, 10);
|
|
cmdp += 2;
|
|
break;*/
|
|
case 'o':
|
|
param_getstr(Cmd, cmdp + 1, destfilename, 32);
|
|
cmdp += 2;
|
|
break;
|
|
case 'p':
|
|
print = true;
|
|
cmdp += 1;
|
|
break;
|
|
case 'e':
|
|
eml = true;
|
|
cmdp += 1;
|
|
break;
|
|
case 'f':
|
|
// File handling
|
|
if (param_getstr(Cmd, cmdp + 1, filename, FILE_PATH_SIZE) >= FILE_PATH_SIZE) {
|
|
PrintAndLogEx(FAILED, "Filename too long");
|
|
errors = true;
|
|
break;
|
|
}
|
|
cmdp += 2;
|
|
break;
|
|
default:
|
|
PrintAndLogEx(WARNING, "Unknown parameter '%c'", param_getchar(Cmd, cmdp));
|
|
errors = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ((filename[0] == '\0') && (!print)) {
|
|
PrintAndLogEx(FAILED, "No print asked and Local dump Filename missing or invalid");
|
|
errors = true;
|
|
}
|
|
|
|
if (destfilename[0] == '\0') {
|
|
PrintAndLogEx(FAILED, "SPIFFS Filename missing or invalid");
|
|
errors = true;
|
|
}
|
|
|
|
// Validations
|
|
if (errors || cmdp == 0) {
|
|
usage_flashmemspiffs_dump();
|
|
return PM3_EINVARG;
|
|
}
|
|
|
|
// get size from spiffs itself !
|
|
SendCommandMIX(CMD_SPIFFS_STAT, 0, 0, 0, (uint8_t *)destfilename, 32);
|
|
PacketResponseNG resp;
|
|
if (!WaitForResponseTimeout(CMD_ACK, &resp, 2000)) {
|
|
PrintAndLogEx(WARNING, "timeout while waiting for reply.");
|
|
return PM3_ETIMEOUT;
|
|
}
|
|
|
|
len = resp.oldarg[0];
|
|
|
|
uint8_t *dump = calloc(len, sizeof(uint8_t));
|
|
if (!dump) {
|
|
PrintAndLogDevice(ERR, "error, cannot allocate memory ");
|
|
return PM3_EMALLOC;
|
|
}
|
|
|
|
PrintAndLogEx(INFO, "downloading "_YELLOW_("%u") "bytes from spiffs (flashmem)", len);
|
|
if (!GetFromDevice(SPIFFS, dump, len, start_index, (uint8_t *)destfilename, 32, NULL, -1, true)) {
|
|
PrintAndLogEx(FAILED, "ERROR; downloading from spiffs(flashmemory)");
|
|
free(dump);
|
|
return PM3_EFLASH;
|
|
}
|
|
|
|
if (print) {
|
|
print_hex_break(dump, len, 32);
|
|
}
|
|
|
|
if (filename[0] != '\0') {
|
|
saveFile(filename, "", dump, len);
|
|
if (eml) {
|
|
saveFileEML(filename, dump, len, 16);
|
|
}
|
|
}
|
|
|
|
free(dump);
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static int CmdFlashMemSpiFFSLoad(const char *Cmd) {
|
|
|
|
uint32_t append = 0;
|
|
char filename[FILE_PATH_SIZE] = {0};
|
|
char destfilename[32] = {0};
|
|
bool errors = false;
|
|
uint8_t cmdp = 0;
|
|
|
|
while (param_getchar(Cmd, cmdp) != 0x00 && !errors) {
|
|
switch (tolower(param_getchar(Cmd, cmdp))) {
|
|
case 'h':
|
|
return usage_flashmemspiffs_load();
|
|
case 'f':
|
|
if (param_getstr(Cmd, cmdp + 1, filename, FILE_PATH_SIZE) >= FILE_PATH_SIZE) {
|
|
PrintAndLogEx(FAILED, "Filename too long");
|
|
errors = true;
|
|
break;
|
|
}
|
|
cmdp += 2;
|
|
break;
|
|
case 'o':
|
|
param_getstr(Cmd, cmdp + 1, destfilename, 32);
|
|
cmdp += 2;
|
|
break;
|
|
default:
|
|
PrintAndLogEx(WARNING, "Unknown parameter '%c'", param_getchar(Cmd, cmdp));
|
|
errors = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (destfilename[0] == '\0') {
|
|
PrintAndLogEx(FAILED, "Destination Filename missing or invalid");
|
|
errors = true;
|
|
}
|
|
|
|
// Validations
|
|
if (errors || cmdp == 0) {
|
|
usage_flashmemspiffs_load();
|
|
return PM3_EINVARG;
|
|
}
|
|
|
|
size_t datalen = 0;
|
|
int res = 0;
|
|
uint8_t *data = calloc(FLASH_MEM_MAX_SIZE, sizeof(uint8_t));
|
|
|
|
res = loadFile(filename, "", data, FLASH_MEM_MAX_SIZE, &datalen);
|
|
// int res = loadFileEML( filename, data, &datalen);
|
|
if (res) {
|
|
free(data);
|
|
return PM3_EFILE;
|
|
}
|
|
|
|
if (datalen > FLASH_MEM_MAX_SIZE) {
|
|
PrintAndLogDevice(ERR, "error, filesize is larger than available memory");
|
|
free(data);
|
|
return PM3_EOVFLOW;
|
|
}
|
|
|
|
uint8_t *newdata = realloc(data, datalen);
|
|
if (newdata == NULL) {
|
|
free(data);
|
|
return PM3_EMALLOC;
|
|
} else {
|
|
data = newdata;
|
|
}
|
|
|
|
// We want to mount before multiple operation so the lazy writes/append will not
|
|
// trigger a mount + umount each loop iteration (lazy ops device side)
|
|
SendCommandNG(CMD_SPIFFS_MOUNT, NULL, 0);
|
|
|
|
// Send to device
|
|
uint32_t bytes_sent = 0;
|
|
uint32_t bytes_remaining = datalen;
|
|
|
|
// fast push mode
|
|
conn.block_after_ACK = true;
|
|
|
|
// SendCommandMIX(CMD_SPIFFS_COPY, 0, 0, 0, (uint8_t *)data, 65);
|
|
|
|
while (bytes_remaining > 0) {
|
|
uint32_t bytes_in_packet = MIN(FLASH_MEM_BLOCK_SIZE, bytes_remaining);
|
|
|
|
clearCommandBuffer();
|
|
|
|
char fdata[32 + bytes_in_packet];
|
|
memset(fdata, 0, sizeof(fdata));
|
|
memcpy(fdata, destfilename, 32);
|
|
memcpy(fdata + 32, data + bytes_sent, bytes_in_packet);
|
|
// sprintf(fdata, "%s%s", destfilename, data + bytes_sent);
|
|
|
|
if (bytes_sent > 0)
|
|
append = 1;
|
|
|
|
SendCommandOLD(CMD_SPIFFS_WRITE, append, bytes_in_packet, 0, fdata, 32 + bytes_in_packet);
|
|
|
|
bytes_remaining -= bytes_in_packet;
|
|
bytes_sent += bytes_in_packet;
|
|
|
|
PacketResponseNG resp;
|
|
if (!WaitForResponseTimeout(CMD_ACK, &resp, 2000)) {
|
|
PrintAndLogEx(WARNING, "timeout while waiting for reply.");
|
|
conn.block_after_ACK = false;
|
|
free(data);
|
|
return PM3_ETIMEOUT;
|
|
}
|
|
|
|
uint8_t isok = resp.oldarg[0] & 0xFF;
|
|
if (!isok) {
|
|
conn.block_after_ACK = false;
|
|
PrintAndLogEx(FAILED, "Flash write fail [offset %u]", bytes_sent);
|
|
return PM3_EFLASH;
|
|
}
|
|
}
|
|
|
|
conn.block_after_ACK = false;
|
|
free(data);
|
|
PrintAndLogEx(SUCCESS, "Wrote "_GREEN_("%u") "bytes to file "_GREEN_("%s"), datalen, destfilename);
|
|
|
|
// We want to unmount after these to set things back to normal but more than this
|
|
// unmouting ensure that SPIFFS CACHES are all flushed so our file is actually written on memory
|
|
SendCommandNG(CMD_SPIFFS_UNMOUNT, NULL, 0);
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
static command_t CommandTable[] = {
|
|
|
|
{"help", CmdHelp, AlwaysAvailable, "This help"},
|
|
{
|
|
"copy", CmdFlashMemSpiFFSCopy, IfPm3Flash,
|
|
"Copy a file to another (destructively) in SPIFFS FileSystem in FlashMEM (spiffs)"
|
|
},
|
|
{"dump", CmdFlashMemSpiFFSDump, IfPm3Flash, "Dump a file from SPIFFS FileSystem in FlashMEM (spiffs)"},
|
|
{"info", CmdFlashMemSpiFFSInfo, IfPm3Flash, "Print filesystem info and usage statistics (spiffs)"},
|
|
{"load", CmdFlashMemSpiFFSLoad, IfPm3Flash, "Upload file into SPIFFS Filesystem (spiffs)"},
|
|
{"mount", CmdFlashMemSpiFFSMount, IfPm3Flash, "Mount the SPIFFS Filesystem if not already mounted (spiffs)"},
|
|
{"remove", CmdFlashMemSpiFFSRemove, IfPm3Flash, "Remove a file from SPIFFS FileSystem in FlashMEM (spiffs)"},
|
|
{"rename", CmdFlashMemSpiFFSRename, IfPm3Flash, "Rename/move a file in SPIFFS FileSystem in FlashMEM (spiffs)"},
|
|
{"test", CmdFlashMemSpiFFSTest, IfPm3Flash, "Test SPIFFS Operations (require wiping pages 0 and 1)"},
|
|
{"tree", CmdFlashMemSpiFFSTree, IfPm3Flash, "Print the Flash Memory FileSystem Tree (spiffs)"},
|
|
{"unmount", CmdFlashMemSpiFFSUnmount, IfPm3Flash, "Un-mount the SPIFFS Filesystem if not already mounted (spiffs)"},
|
|
{NULL, NULL, NULL, NULL}
|
|
};
|
|
|
|
static int CmdHelp(const char *Cmd) {
|
|
(void)Cmd; // Cmd is not used so far
|
|
CmdsHelp(CommandTable);
|
|
return PM3_SUCCESS;
|
|
}
|
|
|
|
int CmdFlashMemSpiFFS(const char *Cmd) {
|
|
clearCommandBuffer();
|
|
return CmdsParse(CommandTable, Cmd);
|
|
}
|