mirror of
https://github.com/RfidResearchGroup/proxmark3.git
synced 2025-08-21 05:43:48 -07:00
Merge pull request #1096 from cmolson/em4x70-unlock
EM4x70 Unlock support
This commit is contained in:
commit
6ee3a5fa10
7 changed files with 162 additions and 11 deletions
|
@ -1174,6 +1174,10 @@ static void PacketReceived(PacketCommandNG *packet) {
|
||||||
em4x70_write((em4x70_data_t *)packet->data.asBytes);
|
em4x70_write((em4x70_data_t *)packet->data.asBytes);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case CMD_LF_EM4X70_UNLOCK: {
|
||||||
|
em4x70_unlock((em4x70_data_t *)packet->data.asBytes);
|
||||||
|
break;
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef WITH_ISO15693
|
#ifdef WITH_ISO15693
|
||||||
|
|
|
@ -302,6 +302,12 @@ static void em4x70_send_nibble(uint8_t nibble, bool with_parity) {
|
||||||
em4x70_send_bit(parity);
|
em4x70_send_bit(parity);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void em4x70_send_byte(uint8_t byte) {
|
||||||
|
// Send byte msb first
|
||||||
|
for (int i = 0; i < 8; i++)
|
||||||
|
em4x70_send_bit((byte >> (7 - i)) & 1);
|
||||||
|
}
|
||||||
|
|
||||||
static void em4x70_send_word(const uint16_t word) {
|
static void em4x70_send_word(const uint16_t word) {
|
||||||
|
|
||||||
// Split into nibbles
|
// Split into nibbles
|
||||||
|
@ -354,6 +360,47 @@ static bool check_ack(void) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int send_pin(const uint32_t pin) {
|
||||||
|
|
||||||
|
// sends pin code for unlocking
|
||||||
|
if (find_listen_window(true)) {
|
||||||
|
|
||||||
|
// send PIN command
|
||||||
|
em4x70_send_nibble(EM4X70_COMMAND_PIN, true);
|
||||||
|
|
||||||
|
// --> Send TAG ID (bytes 4-7)
|
||||||
|
for(int i=0; i < 4; i++) {
|
||||||
|
em4x70_send_byte(tag.data[7-i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// --> Send PIN
|
||||||
|
for(int i=0; i < 4 ; i++) {
|
||||||
|
em4x70_send_byte((pin>>(i*8)) & 0xff);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Wait TWALB (write access lock bits)
|
||||||
|
WaitTicks(TICKS_PER_FC * EM4X70_T_TAG_TWALB);
|
||||||
|
|
||||||
|
// <-- Receive ACK
|
||||||
|
if (check_ack()) {
|
||||||
|
|
||||||
|
// <w> Writes Lock Bits
|
||||||
|
WaitTicks(TICKS_PER_FC * EM4X70_T_TAG_WEE);
|
||||||
|
// <-- Receive header + ID
|
||||||
|
uint8_t tag_id[64];
|
||||||
|
int num = em4x70_receive(tag_id);
|
||||||
|
if(num < 32) {
|
||||||
|
Dbprintf("Invalid ID Received");
|
||||||
|
return PM3_ESOFT;
|
||||||
|
}
|
||||||
|
bits2bytes(tag_id, num, &tag.data[4]);
|
||||||
|
return PM3_SUCCESS;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return PM3_ESOFT;
|
||||||
|
}
|
||||||
|
|
||||||
static int write(const uint16_t word, const uint8_t address) {
|
static int write(const uint16_t word, const uint8_t address) {
|
||||||
|
|
||||||
// writes <word> to specified <address>
|
// writes <word> to specified <address>
|
||||||
|
@ -651,4 +698,37 @@ void em4x70_write(em4x70_data_t *etd) {
|
||||||
reply_ng(CMD_LF_EM4X70_WRITE, status, tag.data, sizeof(tag.data));
|
reply_ng(CMD_LF_EM4X70_WRITE, status, tag.data, sizeof(tag.data));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void em4x70_unlock(em4x70_data_t *etd) {
|
||||||
|
|
||||||
|
uint8_t status = 0;
|
||||||
|
|
||||||
|
command_parity = etd->parity;
|
||||||
|
|
||||||
|
init_tag();
|
||||||
|
EM4170_setup_read();
|
||||||
|
|
||||||
|
// Find the Tag
|
||||||
|
if (get_signalproperties() && find_EM4X70_Tag()) {
|
||||||
|
|
||||||
|
// Read ID (required for send_pin command)
|
||||||
|
if(em4x70_read_id()) {
|
||||||
|
|
||||||
|
// Send PIN
|
||||||
|
status = send_pin(etd->pin) == PM3_SUCCESS;
|
||||||
|
|
||||||
|
// If the write succeeded, read the rest of the tag
|
||||||
|
if(status) {
|
||||||
|
// Read Tag
|
||||||
|
// ID doesn't change
|
||||||
|
em4x70_read_um1();
|
||||||
|
em4x70_read_um2();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
StopTicks();
|
||||||
|
lf_finalize();
|
||||||
|
reply_ng(CMD_LF_EM4X70_UNLOCK, status, tag.data, sizeof(tag.data));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -19,5 +19,6 @@ typedef struct {
|
||||||
|
|
||||||
void em4x70_info(em4x70_data_t *etd);
|
void em4x70_info(em4x70_data_t *etd);
|
||||||
void em4x70_write(em4x70_data_t *etd);
|
void em4x70_write(em4x70_data_t *etd);
|
||||||
|
void em4x70_unlock(em4x70_data_t *etd);
|
||||||
|
|
||||||
#endif /* EM4x70_H */
|
#endif /* EM4x70_H */
|
||||||
|
|
|
@ -80,18 +80,18 @@ int CmdEM4x70Info(const char *Cmd) {
|
||||||
|
|
||||||
CLIParserContext *ctx;
|
CLIParserContext *ctx;
|
||||||
|
|
||||||
CLIParserInit(&ctx, "lf em 4x10 info",
|
CLIParserInit(&ctx, "lf em 4x70 info",
|
||||||
"Tag Information EM4x70\n"
|
"Tag Information EM4x70\n"
|
||||||
" Tag variants include ID48 automotive transponder.\n"
|
" Tag variants include ID48 automotive transponder.\n"
|
||||||
" ID48 does not use command parity (default).\n"
|
" ID48 does not use command parity (default).\n"
|
||||||
" V4070 and EM4170 do require parity bit.",
|
" V4070 and EM4170 do require parity bit.",
|
||||||
"lf em 4x70 info\n"
|
"lf em 4x70 info\n"
|
||||||
"lf em 4x70 info -p -> adds parity bit to command\n"
|
"lf em 4x70 info --par -> adds parity bit to command\n"
|
||||||
);
|
);
|
||||||
|
|
||||||
void *argtable[] = {
|
void *argtable[] = {
|
||||||
arg_param_begin,
|
arg_param_begin,
|
||||||
arg_lit0("p", "parity", "Add parity bit when sending commands"),
|
arg_lit0(NULL, "par", "Add parity bit when sending commands"),
|
||||||
arg_param_end
|
arg_param_end
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -104,7 +104,7 @@ int CmdEM4x70Info(const char *Cmd) {
|
||||||
|
|
||||||
PacketResponseNG resp;
|
PacketResponseNG resp;
|
||||||
if (!WaitForResponseTimeout(CMD_LF_EM4X70_INFO, &resp, TIMEOUT)) {
|
if (!WaitForResponseTimeout(CMD_LF_EM4X70_INFO, &resp, TIMEOUT)) {
|
||||||
PrintAndLogEx(WARNING, "timeout while waiting for reply.");
|
PrintAndLogEx(WARNING, "Timeout while waiting for reply.");
|
||||||
return PM3_ETIMEOUT;
|
return PM3_ETIMEOUT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -113,7 +113,7 @@ int CmdEM4x70Info(const char *Cmd) {
|
||||||
return PM3_SUCCESS;
|
return PM3_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
PrintAndLogEx(FAILED, "reading tag " _RED_("failed"));
|
PrintAndLogEx(FAILED, "Reading " _RED_("Failed"));
|
||||||
return PM3_ESOFT;
|
return PM3_ESOFT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -124,15 +124,15 @@ int CmdEM4x70Write(const char *Cmd) {
|
||||||
|
|
||||||
CLIParserContext *ctx;
|
CLIParserContext *ctx;
|
||||||
|
|
||||||
CLIParserInit(&ctx, "lf em 4x10 write",
|
CLIParserInit(&ctx, "lf em 4x70 write",
|
||||||
"Write EM4x70\n",
|
"Write EM4x70\n",
|
||||||
"lf em 4x70 write -b 15 d c0de -> write 'c0de' to block 15\n"
|
"lf em 4x70 write -b 15 -d c0de -> write 'c0de' to block 15\n"
|
||||||
"lf em 4x70 write -p -b 15 -d c0de -> adds parity bit to commands\n"
|
"lf em 4x70 write -b 15 -d c0de --par -> adds parity bit to commands\n"
|
||||||
);
|
);
|
||||||
|
|
||||||
void *argtable[] = {
|
void *argtable[] = {
|
||||||
arg_param_begin,
|
arg_param_begin,
|
||||||
arg_lit0("p", "parity", "Add parity bit when sending commands"),
|
arg_lit0(NULL, "par", "Add parity bit when sending commands"),
|
||||||
arg_int1("b", "block", "<dec>", "block/word address, dec"),
|
arg_int1("b", "block", "<dec>", "block/word address, dec"),
|
||||||
arg_str1("d", "data", "<hex>", "data, 2 bytes"),
|
arg_str1("d", "data", "<hex>", "data, 2 bytes"),
|
||||||
arg_param_end
|
arg_param_end
|
||||||
|
@ -181,10 +181,70 @@ int CmdEM4x70Write(const char *Cmd) {
|
||||||
return PM3_ESOFT;
|
return PM3_ESOFT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int CmdEM4x70Unlock(const char *Cmd) {
|
||||||
|
|
||||||
|
// send pin code to device, unlocking it for writing
|
||||||
|
em4x70_data_t etd = {0};
|
||||||
|
|
||||||
|
CLIParserContext *ctx;
|
||||||
|
|
||||||
|
CLIParserInit(&ctx, "lf em 4x70 unlock",
|
||||||
|
"Unlock EM4x70 by sending PIN\n"
|
||||||
|
"Default pin may be:\n"
|
||||||
|
" AAAAAAAA\n"
|
||||||
|
" 00000000\n",
|
||||||
|
"lf em 4x70 unlock -p 11223344 -> Unlock with PIN\n"
|
||||||
|
"lf em 4x70 unlock -p 11223344 --par -> Unlock with PIN using parity commands\n"
|
||||||
|
);
|
||||||
|
|
||||||
|
void *argtable[] = {
|
||||||
|
arg_param_begin,
|
||||||
|
arg_lit0(NULL, "par", "Add parity bit when sending commands"),
|
||||||
|
arg_str1("p", "pin", "<hex>", "pin, 4 bytes"),
|
||||||
|
arg_param_end
|
||||||
|
};
|
||||||
|
|
||||||
|
CLIExecWithReturn(ctx, Cmd, argtable, true);
|
||||||
|
|
||||||
|
etd.parity = arg_get_lit(ctx, 1);
|
||||||
|
|
||||||
|
int pin_len = 0;
|
||||||
|
uint8_t pin[4] = {0x0};
|
||||||
|
|
||||||
|
CLIGetHexWithReturn(ctx, 2, pin, &pin_len);
|
||||||
|
|
||||||
|
CLIParserFree(ctx);
|
||||||
|
|
||||||
|
if (pin_len != 4) {
|
||||||
|
PrintAndLogEx(FAILED, "PIN length must be 4 bytes instead of %d", pin_len);
|
||||||
|
return PM3_EINVARG;
|
||||||
|
}
|
||||||
|
|
||||||
|
etd.pin = BYTES2UINT32(pin);
|
||||||
|
|
||||||
|
clearCommandBuffer();
|
||||||
|
SendCommandNG(CMD_LF_EM4X70_UNLOCK, (uint8_t *)&etd, sizeof(etd));
|
||||||
|
|
||||||
|
PacketResponseNG resp;
|
||||||
|
if (!WaitForResponseTimeout(CMD_LF_EM4X70_UNLOCK, &resp, TIMEOUT)) {
|
||||||
|
PrintAndLogEx(WARNING, "Timeout while waiting for reply.");
|
||||||
|
return PM3_ETIMEOUT;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (resp.status) {
|
||||||
|
print_info_result(resp.data.asBytes);
|
||||||
|
return PM3_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
PrintAndLogEx(FAILED, "Unlocking tag " _RED_("failed"));
|
||||||
|
return PM3_ESOFT;
|
||||||
|
}
|
||||||
|
|
||||||
static command_t CommandTable[] = {
|
static command_t CommandTable[] = {
|
||||||
{"help", CmdHelp, AlwaysAvailable, "This help"},
|
{"help", CmdHelp, AlwaysAvailable, "This help"},
|
||||||
{"info", CmdEM4x70Info, IfPm3EM4x70, "Tag information EM4x70"},
|
{"info", CmdEM4x70Info, IfPm3EM4x70, "Tag information EM4x70"},
|
||||||
{"write", CmdEM4x70Write, IfPm3EM4x70, "Write EM4x70"},
|
{"write", CmdEM4x70Write, IfPm3EM4x70, "Write EM4x70"},
|
||||||
|
{"unlock", CmdEM4x70Unlock, IfPm3EM4x70, "Unlock EM4x70 for writing"},
|
||||||
{NULL, NULL, NULL, NULL}
|
{NULL, NULL, NULL, NULL}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
int CmdLFEM4X70(const char *Cmd);
|
int CmdLFEM4X70(const char *Cmd);
|
||||||
int CmdEM4x70Info(const char *Cmd);
|
int CmdEM4x70Info(const char *Cmd);
|
||||||
int CmdEM4x70Write(const char *Cmd);
|
int CmdEM4x70Write(const char *Cmd);
|
||||||
|
int CmdEM4x70Unlock(const char *Cmd);
|
||||||
|
|
||||||
int em4x70_info(void);
|
int em4x70_info(void);
|
||||||
bool detect_4x70_block(void);
|
bool detect_4x70_block(void);
|
||||||
|
|
|
@ -19,6 +19,10 @@ typedef struct {
|
||||||
// Used for writing address
|
// Used for writing address
|
||||||
uint8_t address;
|
uint8_t address;
|
||||||
uint16_t word;
|
uint16_t word;
|
||||||
|
|
||||||
|
// PIN to unlock
|
||||||
|
uint32_t pin;
|
||||||
|
|
||||||
} em4x70_data_t;
|
} em4x70_data_t;
|
||||||
|
|
||||||
#endif /* EM4X70_H__ */
|
#endif /* EM4X70_H__ */
|
||||||
|
|
|
@ -518,6 +518,7 @@ typedef struct {
|
||||||
#define CMD_LF_EM4X50_CHK 0x0253
|
#define CMD_LF_EM4X50_CHK 0x0253
|
||||||
#define CMD_LF_EM4X70_INFO 0x0260
|
#define CMD_LF_EM4X70_INFO 0x0260
|
||||||
#define CMD_LF_EM4X70_WRITE 0x0261
|
#define CMD_LF_EM4X70_WRITE 0x0261
|
||||||
|
#define CMD_LF_EM4X70_UNLOCK 0x0262
|
||||||
// Sampling configuration for LF reader/sniffer
|
// Sampling configuration for LF reader/sniffer
|
||||||
#define CMD_LF_SAMPLING_SET_CONFIG 0x021D
|
#define CMD_LF_SAMPLING_SET_CONFIG 0x021D
|
||||||
#define CMD_LF_FSK_SIMULATE 0x021E
|
#define CMD_LF_FSK_SIMULATE 0x021E
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue