]> git.zerfleddert.de Git - proxmark3-svn/blame_incremental - client/cmdlfem4x.c
lfem4x cleanup/add/fix
[proxmark3-svn] / client / cmdlfem4x.c
... / ...
CommitLineData
1//-----------------------------------------------------------------------------
2// Copyright (C) 2010 iZsh <izsh at fail0verflow.com>
3//
4// This code is licensed to you under the terms of the GNU GPL, version 2 or,
5// at your option, any later version. See the LICENSE.txt file for the text of
6// the license.
7//-----------------------------------------------------------------------------
8// Low frequency EM4x commands
9//-----------------------------------------------------------------------------
10
11#include <stdio.h>
12#include <string.h>
13#include <inttypes.h>
14#include "proxmark3.h"
15#include "ui.h"
16#include "util.h"
17#include "graph.h"
18#include "cmdparser.h"
19#include "cmddata.h"
20#include "cmdlf.h"
21#include "cmdlfem4x.h"
22#include "lfdemod.h"
23char *global_em410xId;
24
25static int CmdHelp(const char *Cmd);
26
27int CmdEMdemodASK(const char *Cmd)
28{
29 char cmdp = param_getchar(Cmd, 0);
30 int findone = (cmdp == '1') ? 1 : 0;
31 UsbCommand c={CMD_EM410X_DEMOD};
32 c.arg[0]=findone;
33 SendCommand(&c);
34 return 0;
35}
36
37/* Read the ID of an EM410x tag.
38 * Format:
39 * 1111 1111 1 <-- standard non-repeatable header
40 * XXXX [row parity bit] <-- 10 rows of 5 bits for our 40 bit tag ID
41 * ....
42 * CCCC <-- each bit here is parity for the 10 bits above in corresponding column
43 * 0 <-- stop bit, end of tag
44 */
45int CmdEM410xRead(const char *Cmd)
46{
47 uint32_t hi=0;
48 uint64_t lo=0;
49
50 if(!AskEm410xDemod("", &hi, &lo)) return 0;
51 PrintAndLog("EM410x pattern found: ");
52 printEM410x(hi, lo);
53 if (hi){
54 PrintAndLog ("EM410x XL pattern found");
55 return 0;
56 }
57 char id[12] = {0x00};
58 sprintf(id, "%010llx",lo);
59
60 global_em410xId = id;
61 return 1;
62}
63
64// emulate an EM410X tag
65int CmdEM410xSim(const char *Cmd)
66{
67 int i, n, j, binary[4], parity[4];
68
69 char cmdp = param_getchar(Cmd, 0);
70 uint8_t uid[5] = {0x00};
71
72 if (cmdp == 'h' || cmdp == 'H') {
73 PrintAndLog("Usage: lf em4x 410xsim <UID>");
74 PrintAndLog("");
75 PrintAndLog(" sample: lf em4x 410xsim 0F0368568B");
76 return 0;
77 }
78
79 if (param_gethex(Cmd, 0, uid, 10)) {
80 PrintAndLog("UID must include 10 HEX symbols");
81 return 0;
82 }
83
84 PrintAndLog("Starting simulating UID %02X%02X%02X%02X%02X", uid[0],uid[1],uid[2],uid[3],uid[4]);
85 PrintAndLog("Press pm3-button to about simulation");
86
87 /* clock is 64 in EM410x tags */
88 int clock = 64;
89
90 /* clear our graph */
91 ClearGraph(0);
92
93 /* write 9 start bits */
94 for (i = 0; i < 9; i++)
95 AppendGraph(0, clock, 1);
96
97 /* for each hex char */
98 parity[0] = parity[1] = parity[2] = parity[3] = 0;
99 for (i = 0; i < 10; i++)
100 {
101 /* read each hex char */
102 sscanf(&Cmd[i], "%1x", &n);
103 for (j = 3; j >= 0; j--, n/= 2)
104 binary[j] = n % 2;
105
106 /* append each bit */
107 AppendGraph(0, clock, binary[0]);
108 AppendGraph(0, clock, binary[1]);
109 AppendGraph(0, clock, binary[2]);
110 AppendGraph(0, clock, binary[3]);
111
112 /* append parity bit */
113 AppendGraph(0, clock, binary[0] ^ binary[1] ^ binary[2] ^ binary[3]);
114
115 /* keep track of column parity */
116 parity[0] ^= binary[0];
117 parity[1] ^= binary[1];
118 parity[2] ^= binary[2];
119 parity[3] ^= binary[3];
120 }
121
122 /* parity columns */
123 AppendGraph(0, clock, parity[0]);
124 AppendGraph(0, clock, parity[1]);
125 AppendGraph(0, clock, parity[2]);
126 AppendGraph(0, clock, parity[3]);
127
128 /* stop bit */
129 AppendGraph(1, clock, 0);
130
131 CmdLFSim("0"); //240 start_gap.
132 return 0;
133}
134
135/* Function is equivalent of lf read + data samples + em410xread
136 * looped until an EM410x tag is detected
137 *
138 * Why is CmdSamples("16000")?
139 * TBD: Auto-grow sample size based on detected sample rate. IE: If the
140 * rate gets lower, then grow the number of samples
141 * Changed by martin, 4000 x 4 = 16000,
142 * see http://www.proxmark.org/forum/viewtopic.php?pid=7235#p7235
143
144*/
145int CmdEM410xWatch(const char *Cmd)
146{
147 do {
148 if (ukbhit()) {
149 printf("\naborted via keyboard!\n");
150 break;
151 }
152
153 CmdLFRead("s");
154 getSamples("8192",true); //capture enough to get 2 full messages
155 } while (!CmdEM410xRead(""));
156
157 return 0;
158}
159
160//currently only supports manchester modulations
161int CmdEM410xWatchnSpoof(const char *Cmd)
162{
163 CmdEM410xWatch(Cmd);
164 PrintAndLog("# Replaying captured ID: %s",global_em410xId);
165 CmdLFaskSim("");
166 return 0;
167}
168
169bool EM_EndParityTest(uint8_t *BitStream, size_t size, uint8_t rows, uint8_t cols, uint8_t pType)
170{
171 if (rows*cols>size) return false;
172 uint8_t colP=0;
173 //assume last row is a parity row and do not test
174 for (uint8_t colNum = 0; colNum < cols-1; colNum++) {
175 for (uint8_t rowNum = 0; rowNum < rows; rowNum++) {
176 colP ^= BitStream[(rowNum*cols)+colNum];
177 }
178 if (colP != pType) return false;
179 }
180 return true;
181}
182
183bool EM_ByteParityTest(uint8_t *BitStream, size_t size, uint8_t rows, uint8_t cols, uint8_t pType)
184{
185 if (rows*cols>size) return false;
186 uint8_t rowP=0;
187 //assume last row is a parity row and do not test
188 for (uint8_t rowNum = 0; rowNum < rows-1; rowNum++) {
189 for (uint8_t colNum = 0; colNum < cols; colNum++) {
190 rowP ^= BitStream[(rowNum*cols)+colNum];
191 }
192 if (rowP != pType) return false;
193 }
194 return true;
195}
196
197uint32_t OutputEM4x50_Block(uint8_t *BitStream, size_t size, bool verbose, bool pTest)
198{
199 if (size<45) return 0;
200 uint32_t code = bytebits_to_byte(BitStream,8);
201 code = code<<8 | bytebits_to_byte(BitStream+9,8);
202 code = code<<8 | bytebits_to_byte(BitStream+18,8);
203 code = code<<8 | bytebits_to_byte(BitStream+27,8);
204 if (verbose || g_debugMode){
205 for (uint8_t i = 0; i<5; i++){
206 if (i == 4) PrintAndLog("");
207 PrintAndLog("%d%d%d%d%d%d%d%d %d -> 0x%02x",
208 BitStream[i*9],
209 BitStream[i*9+1],
210 BitStream[i*9+2],
211 BitStream[i*9+3],
212 BitStream[i*9+4],
213 BitStream[i*9+5],
214 BitStream[i*9+6],
215 BitStream[i*9+7],
216 BitStream[i*9+8],
217 bytebits_to_byte(BitStream+i*9,8)
218 );
219 }
220 if (pTest)
221 PrintAndLog("Parity Passed");
222 else
223 PrintAndLog("Parity Failed");
224 }
225 //PrintAndLog("Code: %08x",code);
226 return code;
227}
228/* Read the transmitted data of an EM4x50 tag
229 * Format:
230 *
231 * XXXXXXXX [row parity bit (even)] <- 8 bits plus parity
232 * XXXXXXXX [row parity bit (even)] <- 8 bits plus parity
233 * XXXXXXXX [row parity bit (even)] <- 8 bits plus parity
234 * XXXXXXXX [row parity bit (even)] <- 8 bits plus parity
235 * CCCCCCCC <- column parity bits
236 * 0 <- stop bit
237 * LW <- Listen Window
238 *
239 * This pattern repeats for every block of data being transmitted.
240 * Transmission starts with two Listen Windows (LW - a modulated
241 * pattern of 320 cycles each (32/32/128/64/64)).
242 *
243 * Note that this data may or may not be the UID. It is whatever data
244 * is stored in the blocks defined in the control word First and Last
245 * Word Read values. UID is stored in block 32.
246 */
247int EM4x50Read(const char *Cmd, bool verbose)
248{
249 uint8_t fndClk[]={0,8,16,32,40,50,64};
250 int clk = 0;
251 int invert = 0;
252 sscanf(Cmd, "%i %i", &clk, &invert);
253 int tol = 0;
254 int i, j, startblock, skip, block, start, end, low, high, minClk;
255 bool complete= false;
256 int tmpbuff[MAX_GRAPH_TRACE_LEN / 64];
257 save_restoreGB(1);
258 uint32_t Code[6];
259 char tmp[6];
260
261 char tmp2[20];
262 high= low= 0;
263 memset(tmpbuff, 0, MAX_GRAPH_TRACE_LEN / 64);
264
265 // first get high and low values
266 for (i = 0; i < GraphTraceLen; i++)
267 {
268 if (GraphBuffer[i] > high)
269 high = GraphBuffer[i];
270 else if (GraphBuffer[i] < low)
271 low = GraphBuffer[i];
272 }
273
274 // populate a buffer with pulse lengths
275 i= 0;
276 j= 0;
277 minClk= 255;
278 while (i < GraphTraceLen)
279 {
280 // measure from low to low
281 while ((GraphBuffer[i] > low) && (i<GraphTraceLen))
282 ++i;
283 start= i;
284 while ((GraphBuffer[i] < high) && (i<GraphTraceLen))
285 ++i;
286 while ((GraphBuffer[i] > low) && (i<GraphTraceLen))
287 ++i;
288 if (j>=(MAX_GRAPH_TRACE_LEN/64)) {
289 break;
290 }
291 tmpbuff[j++]= i - start;
292 if (i-start < minClk) minClk = i-start;
293 }
294 // set clock
295 if (!clk){
296 for (uint8_t clkCnt = 0; clkCnt<7; clkCnt++) {
297 tol = fndClk[clkCnt]/8;
298 if (fndClk[clkCnt]-tol >= minClk) {
299 clk=fndClk[clkCnt];
300 break;
301 }
302 }
303 }
304
305 // look for data start - should be 2 pairs of LW (pulses of clk*3,clk*2)
306 start= -1;
307 skip= 0;
308 for (i= 0; i < j - 4 ; ++i)
309 {
310 skip += tmpbuff[i];
311 if (tmpbuff[i] >= clk*3-tol && tmpbuff[i] <= clk*3+tol)
312 if (tmpbuff[i+1] >= clk*2-tol && tmpbuff[i+1] <= clk*2+tol)
313 if (tmpbuff[i+2] >= clk*3-tol && tmpbuff[i+2] <= clk*3+tol)
314 if (tmpbuff[i+3] >= clk-tol)
315 {
316 start= i + 4;
317 break;
318 }
319 }
320 startblock= i + 4;
321
322 // skip over the remainder of LW
323 skip += tmpbuff[i+1] + tmpbuff[i+2] + clk + clk/8;
324
325 int phaseoff = tmpbuff[i+3]-clk;
326
327 // now do it again to find the end
328 end = skip;
329 for (i += 3; i < j - 4 ; ++i)
330 {
331 end += tmpbuff[i];
332 if (tmpbuff[i] >= clk*3-tol && tmpbuff[i] <= clk*3 + tol)
333 if (tmpbuff[i+1] >= clk*2-tol && tmpbuff[i+1] <= clk*2 + tol)
334 if (tmpbuff[i+2] >= clk*3-tol && tmpbuff[i+2] <= clk*3 + tol)
335 if (tmpbuff[i+3] >= clk-tol)
336 {
337 complete= true;
338 break;
339 }
340 }
341 end = i;
342 // report back
343 if (verbose || g_debugMode) {
344 if (start >= 0) {
345 PrintAndLog("\nNote: should print 45 bits then 0177 (end of block)");
346 PrintAndLog(" for each block");
347 PrintAndLog(" Also, sometimes the demod gets out of sync and ");
348 PrintAndLog(" inverts the output - when this happens the 0177");
349 PrintAndLog(" will be 3 extra 1's at the end");
350 PrintAndLog(" 'data askedge' command may fix that");
351 } else {
352 PrintAndLog("No data found!");
353 PrintAndLog("Try again with more samples.");
354 return 0;
355 }
356 if (!complete)
357 {
358 PrintAndLog("*** Warning!");
359 PrintAndLog("Partial data - no end found!");
360 PrintAndLog("Try again with more samples.");
361 }
362 } else if (start < 0) return 0;
363 start=skip;
364 snprintf(tmp2, sizeof(tmp2),"%d %d 1000 %d", clk, invert, clk*47);
365 // get rid of leading crap
366 snprintf(tmp, sizeof(tmp),"%i",skip);
367 CmdLtrim(tmp);
368 bool pTest;
369 bool AllPTest=true;
370 // now work through remaining buffer printing out data blocks
371 block = 0;
372 i = startblock;
373 while (block < 6)
374 {
375 if (verbose || g_debugMode) PrintAndLog("\nBlock %i:", block);
376 skip = phaseoff;
377
378 // look for LW before start of next block
379 for ( ; i < j - 4 ; ++i)
380 {
381 skip += tmpbuff[i];
382 if (tmpbuff[i] >= clk*3-tol && tmpbuff[i] <= clk*3+tol)
383 if (tmpbuff[i+1] >= clk-tol)
384 break;
385 }
386 skip += clk;
387 phaseoff = tmpbuff[i+1]-clk;
388 i += 2;
389 if (ASKmanDemod(tmp2, false, false)<1) return 0;
390 //set DemodBufferLen to just one block
391 DemodBufferLen = skip/clk;
392 //test parities
393 pTest = EM_ByteParityTest(DemodBuffer,DemodBufferLen,5,9,0);
394 pTest &= EM_EndParityTest(DemodBuffer,DemodBufferLen,5,9,0);
395 AllPTest &= pTest;
396 //get output
397 Code[block]=OutputEM4x50_Block(DemodBuffer,DemodBufferLen,verbose, pTest);
398 if (g_debugMode) PrintAndLog("\nskipping %d samples, bits:%d",start, skip/clk);
399 //skip to start of next block
400 snprintf(tmp,sizeof(tmp),"%i",skip);
401 CmdLtrim(tmp);
402 block++;
403 if (i>=end) break; //in case chip doesn't output 6 blocks
404 }
405 //print full code:
406 if (verbose || g_debugMode || AllPTest){
407 PrintAndLog("Found data at sample: %i - using clock: %i",skip,clk);
408 //PrintAndLog("\nSummary:");
409 end=block;
410 for (block=0; block<end; block++){
411 PrintAndLog("Block %d: %08x",block,Code[block]);
412 }
413 if (AllPTest)
414 PrintAndLog("Parities Passed");
415 else
416 PrintAndLog("Parities Failed");
417 }
418
419 //restore GraphBuffer
420 save_restoreGB(0);
421 return (int)AllPTest;
422}
423
424int CmdEM4x50Read(const char *Cmd)
425{
426 return EM4x50Read(Cmd, true);
427}
428
429int CmdEM410xWrite(const char *Cmd)
430{
431 uint64_t id = 0xFFFFFFFFFFFFFFFF; // invalid id value
432 int card = 0xFF; // invalid card value
433 unsigned int clock = 0; // invalid clock value
434
435 sscanf(Cmd, "%" PRIx64 " %d %d", &id, &card, &clock);
436
437 // Check ID
438 if (id == 0xFFFFFFFFFFFFFFFF) {
439 PrintAndLog("Error! ID is required.\n");
440 return 0;
441 }
442 if (id >= 0x10000000000) {
443 PrintAndLog("Error! Given EM410x ID is longer than 40 bits.\n");
444 return 0;
445 }
446
447 // Check Card
448 if (card == 0xFF) {
449 PrintAndLog("Error! Card type required.\n");
450 return 0;
451 }
452 if (card < 0) {
453 PrintAndLog("Error! Bad card type selected.\n");
454 return 0;
455 }
456
457 // Check Clock
458 if (card == 1)
459 {
460 // Default: 64
461 if (clock == 0)
462 clock = 64;
463
464 // Allowed clock rates: 16, 32 and 64
465 if ((clock != 16) && (clock != 32) && (clock != 64)) {
466 PrintAndLog("Error! Clock rate %d not valid. Supported clock rates are 16, 32 and 64.\n", clock);
467 return 0;
468 }
469 }
470 else if (clock != 0)
471 {
472 PrintAndLog("Error! Clock rate is only supported on T55x7 tags.\n");
473 return 0;
474 }
475
476 if (card == 1) {
477 PrintAndLog("Writing %s tag with UID 0x%010" PRIx64 " (clock rate: %d)", "T55x7", id, clock);
478 // NOTE: We really should pass the clock in as a separate argument, but to
479 // provide for backwards-compatibility for older firmware, and to avoid
480 // having to add another argument to CMD_EM410X_WRITE_TAG, we just store
481 // the clock rate in bits 8-15 of the card value
482 card = (card & 0xFF) | (((uint64_t)clock << 8) & 0xFF00);
483 }
484 else if (card == 0)
485 PrintAndLog("Writing %s tag with UID 0x%010" PRIx64, "T5555", id, clock);
486 else {
487 PrintAndLog("Error! Bad card type selected.\n");
488 return 0;
489 }
490
491 UsbCommand c = {CMD_EM410X_WRITE_TAG, {card, (uint32_t)(id >> 32), (uint32_t)id}};
492 SendCommand(&c);
493
494 return 0;
495}
496
497int CmdReadWord(const char *Cmd)
498{
499 int Word = -1; //default to invalid word
500 UsbCommand c;
501
502 sscanf(Cmd, "%d", &Word);
503
504 if ( (Word > 15) | (Word < 0) ) {
505 PrintAndLog("Word must be between 0 and 15");
506 return 1;
507 }
508
509 PrintAndLog("Reading word %d", Word);
510
511 c.cmd = CMD_EM4X_READ_WORD;
512 c.d.asBytes[0] = 0x0; //Normal mode
513 c.arg[0] = 0;
514 c.arg[1] = Word;
515 c.arg[2] = 0;
516 SendCommand(&c);
517 return 0;
518}
519
520int CmdReadWordPWD(const char *Cmd)
521{
522 int Word = -1; //default to invalid word
523 int Password = 0xFFFFFFFF; //default to blank password
524 UsbCommand c;
525
526 sscanf(Cmd, "%d %x", &Word, &Password);
527
528 if ( (Word > 15) | (Word < 0) ) {
529 PrintAndLog("Word must be between 0 and 15");
530 return 1;
531 }
532
533 PrintAndLog("Reading word %d with password %08X", Word, Password);
534
535 c.cmd = CMD_EM4X_READ_WORD;
536 c.d.asBytes[0] = 0x1; //Password mode
537 c.arg[0] = 0;
538 c.arg[1] = Word;
539 c.arg[2] = Password;
540 SendCommand(&c);
541 return 0;
542}
543
544int CmdWriteWord(const char *Cmd)
545{
546 int Word = 16; //default to invalid block
547 int Data = 0xFFFFFFFF; //default to blank data
548 UsbCommand c;
549
550 sscanf(Cmd, "%x %d", &Data, &Word);
551
552 if (Word > 15) {
553 PrintAndLog("Word must be between 0 and 15");
554 return 1;
555 }
556
557 PrintAndLog("Writing word %d with data %08X", Word, Data);
558
559 c.cmd = CMD_EM4X_WRITE_WORD;
560 c.d.asBytes[0] = 0x0; //Normal mode
561 c.arg[0] = Data;
562 c.arg[1] = Word;
563 c.arg[2] = 0;
564 SendCommand(&c);
565 return 0;
566}
567
568int CmdWriteWordPWD(const char *Cmd)
569{
570 int Word = 16; //default to invalid word
571 int Data = 0xFFFFFFFF; //default to blank data
572 int Password = 0xFFFFFFFF; //default to blank password
573 UsbCommand c;
574
575 sscanf(Cmd, "%x %d %x", &Data, &Word, &Password);
576
577 if (Word > 15) {
578 PrintAndLog("Word must be between 0 and 15");
579 return 1;
580 }
581
582 PrintAndLog("Writing word %d with data %08X and password %08X", Word, Data, Password);
583
584 c.cmd = CMD_EM4X_WRITE_WORD;
585 c.d.asBytes[0] = 0x1; //Password mode
586 c.arg[0] = Data;
587 c.arg[1] = Word;
588 c.arg[2] = Password;
589 SendCommand(&c);
590 return 0;
591}
592
593static command_t CommandTable[] =
594{
595 {"help", CmdHelp, 1, "This help"},
596 {"em410xdemod", CmdEMdemodASK, 0, "[findone] -- Extract ID from EM410x tag (option 0 for continuous loop, 1 for only 1 tag)"},
597 {"em410xread", CmdEM410xRead, 1, "[clock rate] -- Extract ID from EM410x tag"},
598 {"em410xsim", CmdEM410xSim, 0, "<UID> -- Simulate EM410x tag"},
599 {"em410xwatch", CmdEM410xWatch, 0, "['h'] -- Watches for EM410x 125/134 kHz tags (option 'h' for 134)"},
600 {"em410xspoof", CmdEM410xWatchnSpoof, 0, "['h'] --- Watches for EM410x 125/134 kHz tags, and replays them. (option 'h' for 134)" },
601 {"em410xwrite", CmdEM410xWrite, 1, "<UID> <'0' T5555> <'1' T55x7> [clock rate] -- Write EM410x UID to T5555(Q5) or T55x7 tag, optionally setting clock rate"},
602 {"em4x50read", CmdEM4x50Read, 1, "Extract data from EM4x50 tag"},
603 {"readword", CmdReadWord, 1, "<Word> -- Read EM4xxx word data"},
604 {"readwordPWD", CmdReadWordPWD, 1, "<Word> <Password> -- Read EM4xxx word data in password mode"},
605 {"writeword", CmdWriteWord, 1, "<Data> <Word> -- Write EM4xxx word data"},
606 {"writewordPWD", CmdWriteWordPWD, 1, "<Data> <Word> <Password> -- Write EM4xxx word data in password mode"},
607 {NULL, NULL, 0, NULL}
608};
609
610int CmdLFEM4X(const char *Cmd)
611{
612 CmdsParse(CommandTable, Cmd);
613 return 0;
614}
615
616int CmdHelp(const char *Cmd)
617{
618 CmdsHelp(CommandTable);
619 return 0;
620}
Impressum, Datenschutz