mirror of
https://github.com/RfidResearchGroup/proxmark3.git
synced 2025-08-19 21:03:48 -07:00
em4x70: Add write pin convenience function to write and verify a new pin
This commit is contained in:
parent
631d6deed3
commit
dc35f79bd2
7 changed files with 118 additions and 10 deletions
|
@ -1182,6 +1182,10 @@ static void PacketReceived(PacketCommandNG *packet) {
|
||||||
em4x70_auth((em4x70_data_t *)packet->data.asBytes);
|
em4x70_auth((em4x70_data_t *)packet->data.asBytes);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case CMD_LF_EM4X70_WRITEPIN: {
|
||||||
|
em4x70_write_pin((em4x70_data_t *)packet->data.asBytes);
|
||||||
|
break;
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef WITH_ISO15693
|
#ifdef WITH_ISO15693
|
||||||
|
|
|
@ -538,7 +538,7 @@ static bool em4x70_read_um2(void) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool find_em4x70_Tag(void) {
|
static bool find_em4x70_tag(void) {
|
||||||
// function is used to check wether a tag on the proxmark is an
|
// function is used to check wether a tag on the proxmark is an
|
||||||
// EM4170 tag or not -> speed up "lf search" process
|
// EM4170 tag or not -> speed up "lf search" process
|
||||||
return find_listen_window(false);
|
return find_listen_window(false);
|
||||||
|
@ -635,7 +635,7 @@ void em4x70_info(em4x70_data_t *etd) {
|
||||||
em4x70_setup_read();
|
em4x70_setup_read();
|
||||||
|
|
||||||
// Find the Tag
|
// Find the Tag
|
||||||
if (get_signalproperties() && find_em4x70_Tag()) {
|
if (get_signalproperties() && find_em4x70_tag()) {
|
||||||
// Read ID, UM1 and UM2
|
// Read ID, UM1 and UM2
|
||||||
status = em4x70_read_id() && em4x70_read_um1() && em4x70_read_um2();
|
status = em4x70_read_id() && em4x70_read_um1() && em4x70_read_um2();
|
||||||
}
|
}
|
||||||
|
@ -655,7 +655,7 @@ void em4x70_write(em4x70_data_t *etd) {
|
||||||
em4x70_setup_read();
|
em4x70_setup_read();
|
||||||
|
|
||||||
// Find the Tag
|
// Find the Tag
|
||||||
if (get_signalproperties() && find_em4x70_Tag()) {
|
if (get_signalproperties() && find_em4x70_tag()) {
|
||||||
|
|
||||||
// Write
|
// Write
|
||||||
status = write(etd->word, etd->address) == PM3_SUCCESS;
|
status = write(etd->word, etd->address) == PM3_SUCCESS;
|
||||||
|
@ -684,7 +684,7 @@ void em4x70_unlock(em4x70_data_t *etd) {
|
||||||
em4x70_setup_read();
|
em4x70_setup_read();
|
||||||
|
|
||||||
// Find the Tag
|
// Find the Tag
|
||||||
if (get_signalproperties() && find_em4x70_Tag()) {
|
if (get_signalproperties() && find_em4x70_tag()) {
|
||||||
|
|
||||||
// Read ID (required for send_pin command)
|
// Read ID (required for send_pin command)
|
||||||
if (em4x70_read_id()) {
|
if (em4x70_read_id()) {
|
||||||
|
@ -718,7 +718,7 @@ void em4x70_auth(em4x70_data_t *etd) {
|
||||||
em4x70_setup_read();
|
em4x70_setup_read();
|
||||||
|
|
||||||
// Find the Tag
|
// Find the Tag
|
||||||
if (get_signalproperties() && find_em4x70_Tag()) {
|
if (get_signalproperties() && find_em4x70_tag()) {
|
||||||
|
|
||||||
// Authenticate and get tag response
|
// Authenticate and get tag response
|
||||||
status = authenticate(etd->rnd, etd->frnd, response) == PM3_SUCCESS;
|
status = authenticate(etd->rnd, etd->frnd, response) == PM3_SUCCESS;
|
||||||
|
@ -729,3 +729,42 @@ void em4x70_auth(em4x70_data_t *etd) {
|
||||||
reply_ng(CMD_LF_EM4X70_AUTH, status, response, sizeof(response));
|
reply_ng(CMD_LF_EM4X70_AUTH, status, response, sizeof(response));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void em4x70_write_pin(em4x70_data_t *etd) {
|
||||||
|
|
||||||
|
uint8_t status = 0;
|
||||||
|
|
||||||
|
command_parity = etd->parity;
|
||||||
|
|
||||||
|
init_tag();
|
||||||
|
em4x70_setup_read();
|
||||||
|
|
||||||
|
// Find the Tag
|
||||||
|
if (get_signalproperties() && find_em4x70_tag()) {
|
||||||
|
|
||||||
|
// Read ID (required for send_pin command)
|
||||||
|
if (em4x70_read_id()) {
|
||||||
|
|
||||||
|
// Write new PIN
|
||||||
|
if( (write( etd->pin & 0xFFFF, EM4X70_PIN_WORD_UPPER) == PM3_SUCCESS) &&
|
||||||
|
(write((etd->pin >> 16) & 0xFFFF, EM4X70_PIN_WORD_LOWER) == PM3_SUCCESS)) {
|
||||||
|
|
||||||
|
// Now Try to authenticate using the new PIN
|
||||||
|
|
||||||
|
// 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_WRITEPIN, status, tag.data, sizeof(tag.data));
|
||||||
|
}
|
||||||
|
|
|
@ -26,5 +26,6 @@ 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);
|
void em4x70_unlock(em4x70_data_t *etd);
|
||||||
void em4x70_auth(em4x70_data_t *etd);
|
void em4x70_auth(em4x70_data_t *etd);
|
||||||
|
void em4x70_write_pin(em4x70_data_t *etd);
|
||||||
|
|
||||||
#endif /* EM4x70_H */
|
#endif /* EM4x70_H */
|
||||||
|
|
|
@ -338,12 +338,70 @@ int CmdEM4x70Auth(const char *Cmd) {
|
||||||
return PM3_ESOFT;
|
return PM3_ESOFT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int CmdEM4x70WritePIN(const char *Cmd) {
|
||||||
|
|
||||||
|
// send pin code to device, unlocking it for writing
|
||||||
|
em4x70_data_t etd = {0};
|
||||||
|
|
||||||
|
CLIParserContext *ctx;
|
||||||
|
|
||||||
|
CLIParserInit(&ctx, "lf em 4x70 writepin",
|
||||||
|
"Write PIN\n",
|
||||||
|
"lf em 4x70 writepin -p 11223344 -> Write PIN\n"
|
||||||
|
"lf em 4x70 writepin -p 11223344 --par -> Write 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_WRITEPIN, (uint8_t *)&etd, sizeof(etd));
|
||||||
|
|
||||||
|
PacketResponseNG resp;
|
||||||
|
if (!WaitForResponseTimeout(CMD_LF_EM4X70_WRITEPIN, &resp, TIMEOUT)) {
|
||||||
|
PrintAndLogEx(WARNING, "Timeout while waiting for reply.");
|
||||||
|
return PM3_ETIMEOUT;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (resp.status) {
|
||||||
|
print_info_result(resp.data.asBytes);
|
||||||
|
PrintAndLogEx(INFO, "Writing new PIN: " _GREEN_("SUCCESS"));
|
||||||
|
return PM3_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
PrintAndLogEx(FAILED, "Writing new PIN: " _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"},
|
{"unlock", CmdEM4x70Unlock, IfPm3EM4x70, "Unlock EM4x70 for writing"},
|
||||||
{"auth", CmdEM4x70Auth, IfPm3EM4x70, "Authenticate EM4x70"},
|
{"auth", CmdEM4x70Auth, IfPm3EM4x70, "Authenticate EM4x70"},
|
||||||
|
{"writepin", CmdEM4x70WritePIN, IfPm3EM4x70, "Write PIN"},
|
||||||
{NULL, NULL, NULL, NULL}
|
{NULL, NULL, NULL, NULL}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -21,6 +21,7 @@ int CmdEM4x70Info(const char *Cmd);
|
||||||
int CmdEM4x70Write(const char *Cmd);
|
int CmdEM4x70Write(const char *Cmd);
|
||||||
int CmdEM4x70Unlock(const char *Cmd);
|
int CmdEM4x70Unlock(const char *Cmd);
|
||||||
int CmdEM4x70Auth(const char *Cmd);
|
int CmdEM4x70Auth(const char *Cmd);
|
||||||
|
int CmdEM4x70WritePIN(const char *Cmd);
|
||||||
|
|
||||||
int em4x70_info(void);
|
int em4x70_info(void);
|
||||||
bool detect_4x70_block(void);
|
bool detect_4x70_block(void);
|
||||||
|
|
|
@ -13,6 +13,10 @@
|
||||||
|
|
||||||
#define EM4X70_NUM_BLOCKS 16
|
#define EM4X70_NUM_BLOCKS 16
|
||||||
|
|
||||||
|
// Common word/block addresses
|
||||||
|
#define EM4X70_PIN_WORD_LOWER 10
|
||||||
|
#define EM4X70_PIN_WORD_UPPER 11
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
bool parity;
|
bool parity;
|
||||||
|
|
||||||
|
|
|
@ -520,6 +520,7 @@ typedef struct {
|
||||||
#define CMD_LF_EM4X70_WRITE 0x0261
|
#define CMD_LF_EM4X70_WRITE 0x0261
|
||||||
#define CMD_LF_EM4X70_UNLOCK 0x0262
|
#define CMD_LF_EM4X70_UNLOCK 0x0262
|
||||||
#define CMD_LF_EM4X70_AUTH 0x0263
|
#define CMD_LF_EM4X70_AUTH 0x0263
|
||||||
|
#define CMD_LF_EM4X70_WRITEPIN 0x0264
|
||||||
// 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