2018-10-17 03:50:18 +08:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// 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.
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// main code for standalone HF Sniff (and ULC/NTAG/ULEV1 pwd storing)
|
|
|
|
//-----------------------------------------------------------------------------
|
2018-10-28 21:54:38 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
This can actually be used in two separate ways.
|
2019-03-09 15:59:13 +08:00
|
|
|
It can either be used to just HF 14a sniff on the go and/or grab the
|
2018-10-28 21:54:38 +08:00
|
|
|
authentication attempts for ULC/NTAG/ULEV1 into the flash mem (RDV4).
|
|
|
|
|
|
|
|
The retrieved sniffing session can be acquired by connecting the device
|
|
|
|
to a client that supports the reconnect capability and issue 'hf 14a list'.
|
|
|
|
|
|
|
|
In order to view the grabbed authentication attempts in the flash mem,
|
2019-07-15 18:48:43 +08:00
|
|
|
you can simply run 'script run read_pwd_mem' or just 'mem dump p l 256'
|
2018-11-06 21:36:00 +08:00
|
|
|
from the client to view the stored quadlets.
|
2018-10-28 21:54:38 +08:00
|
|
|
*/
|
|
|
|
|
2019-08-08 22:57:33 +08:00
|
|
|
#include "standalone.h" // standalone definitions
|
|
|
|
#include "proxmark3_arm.h"
|
|
|
|
#include "iso14443a.h"
|
|
|
|
#include "protocols.h"
|
|
|
|
#include "util.h"
|
|
|
|
#include "spiffs.h"
|
|
|
|
#include "appmain.h"
|
|
|
|
#include "fpgaloader.h"
|
|
|
|
#include "dbprint.h"
|
|
|
|
#include "ticks.h"
|
|
|
|
#include "BigBuf.h"
|
|
|
|
#include "string.h"
|
2018-10-17 03:50:18 +08:00
|
|
|
|
2019-03-09 15:59:13 +08:00
|
|
|
#define DELAY_READER_AIR2ARM_AS_SNIFFER (2 + 3 + 8)
|
2018-10-28 20:01:36 +08:00
|
|
|
#define DELAY_TAG_AIR2ARM_AS_SNIFFER (3 + 14 + 8)
|
2018-10-17 03:50:18 +08:00
|
|
|
|
2018-10-28 20:01:36 +08:00
|
|
|
// Maximum number of auth attempts per standalone session
|
2018-10-28 21:34:42 +08:00
|
|
|
#define MAX_PWDS_PER_SESSION 64
|
2018-10-17 03:50:18 +08:00
|
|
|
|
2019-07-24 05:26:26 +08:00
|
|
|
#define HF_BOG_LOGFILE "hf_bog.log"
|
|
|
|
|
2018-11-07 05:20:55 +08:00
|
|
|
// This is actually copied from SniffIso14443a
|
2020-05-11 00:04:50 +08:00
|
|
|
static void RAMFUNC SniffAndStore(uint8_t param) {
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
iso14443a_setup(FPGA_HF_ISO14443A_SNIFFER);
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
// Allocate memory from BigBuf for some buffers
|
|
|
|
// free all previous allocations first
|
2019-03-10 07:00:59 +08:00
|
|
|
BigBuf_free();
|
|
|
|
BigBuf_Clear_ext(false);
|
2019-03-09 18:19:45 +08:00
|
|
|
clear_trace();
|
|
|
|
set_tracing(true);
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
// Array to store the authpwds
|
|
|
|
uint8_t *capturedPwds = BigBuf_malloc(4 * MAX_PWDS_PER_SESSION);
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
// The command (reader -> tag) that we're receiving.
|
|
|
|
uint8_t *receivedCmd = BigBuf_malloc(MAX_FRAME_SIZE);
|
|
|
|
uint8_t *receivedCmdPar = BigBuf_malloc(MAX_PARITY_SIZE);
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
// The response (tag -> reader) that we're receiving.
|
|
|
|
uint8_t *receivedResp = BigBuf_malloc(MAX_FRAME_SIZE);
|
|
|
|
uint8_t *receivedRespPar = BigBuf_malloc(MAX_PARITY_SIZE);
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
// The DMA buffer, used to stream samples from the FPGA
|
|
|
|
uint8_t *dmaBuf = BigBuf_malloc(DMA_BUFFER_SIZE);
|
|
|
|
uint8_t *data = dmaBuf;
|
2018-10-17 03:50:18 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
uint8_t previous_data = 0;
|
2019-04-07 16:17:43 +08:00
|
|
|
int dataLen;
|
2019-03-09 18:19:45 +08:00
|
|
|
bool TagIsActive = false;
|
|
|
|
bool ReaderIsActive = false;
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
// Set up the demodulator for tag -> reader responses.
|
2019-08-08 22:57:33 +08:00
|
|
|
Demod14aInit(receivedResp, receivedRespPar);
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
// Set up the demodulator for the reader -> tag commands
|
2019-08-08 22:57:33 +08:00
|
|
|
Uart14aInit(receivedCmd, receivedCmdPar);
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
// Setup and start DMA.
|
2019-07-24 05:26:26 +08:00
|
|
|
if (!FpgaSetupSscDma((uint8_t *)dmaBuf, DMA_BUFFER_SIZE)) {
|
|
|
|
if (DBGLEVEL > 1)
|
|
|
|
Dbprintf("FpgaSetupSscDma failed. Exiting");
|
2019-03-09 18:19:45 +08:00
|
|
|
return;
|
|
|
|
}
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-08-08 22:57:33 +08:00
|
|
|
tUart14a *uart = GetUart14a();
|
|
|
|
tDemod14a *demod = GetDemod14a();
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
// We won't start recording the frames that we acquire until we trigger;
|
|
|
|
// a good trigger condition to get started is probably when we see a
|
|
|
|
// response from the tag.
|
|
|
|
// triggered == false -- to wait first for card
|
|
|
|
bool triggered = !(param & 0x03);
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-04-07 16:17:43 +08:00
|
|
|
uint32_t my_rsamples = 0;
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
// Current captured passwords counter
|
|
|
|
uint8_t auth_attempts = 0;
|
2018-10-17 03:50:18 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
SpinDelay(50);
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
// loop and listen
|
|
|
|
while (!BUTTON_PRESS()) {
|
2018-10-17 03:50:18 +08:00
|
|
|
WDT_HIT();
|
|
|
|
LED_A_ON();
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
int register readBufDataP = data - dmaBuf;
|
|
|
|
int register dmaBufDataP = DMA_BUFFER_SIZE - AT91C_BASE_PDC_SSC->PDC_RCR;
|
|
|
|
if (readBufDataP <= dmaBufDataP)
|
|
|
|
dataLen = dmaBufDataP - readBufDataP;
|
|
|
|
else
|
|
|
|
dataLen = DMA_BUFFER_SIZE - readBufDataP + dmaBufDataP;
|
|
|
|
|
|
|
|
// test for length of buffer
|
|
|
|
if (dataLen > DMA_BUFFER_SIZE) { // TODO: Check if this works properly
|
|
|
|
Dbprintf("[!] blew circular buffer! | datalen %u", dataLen);
|
|
|
|
break;
|
|
|
|
}
|
2019-07-24 05:26:26 +08:00
|
|
|
if (dataLen < 1)
|
|
|
|
continue;
|
2019-03-09 18:19:45 +08:00
|
|
|
|
|
|
|
// primary buffer was stopped( <-- we lost data!
|
|
|
|
if (!AT91C_BASE_PDC_SSC->PDC_RCR) {
|
2019-07-24 05:26:26 +08:00
|
|
|
AT91C_BASE_PDC_SSC->PDC_RPR = (uint32_t)dmaBuf;
|
2019-03-09 18:19:45 +08:00
|
|
|
AT91C_BASE_PDC_SSC->PDC_RCR = DMA_BUFFER_SIZE;
|
2019-07-24 05:26:26 +08:00
|
|
|
// Dbprintf("[-] RxEmpty ERROR | data length %d", dataLen); // temporary
|
2019-03-09 18:19:45 +08:00
|
|
|
}
|
|
|
|
// secondary buffer sets as primary, secondary buffer was stopped
|
|
|
|
if (!AT91C_BASE_PDC_SSC->PDC_RNCR) {
|
2019-07-24 05:26:26 +08:00
|
|
|
AT91C_BASE_PDC_SSC->PDC_RNPR = (uint32_t)dmaBuf;
|
2019-03-09 18:19:45 +08:00
|
|
|
AT91C_BASE_PDC_SSC->PDC_RNCR = DMA_BUFFER_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
LED_A_OFF();
|
|
|
|
|
|
|
|
// Need two samples to feed Miller and Manchester-Decoder
|
2019-04-07 16:17:43 +08:00
|
|
|
if (my_rsamples & 0x01) {
|
2019-03-09 18:19:45 +08:00
|
|
|
|
|
|
|
if (!TagIsActive) { // no need to try decoding reader data if the tag is sending
|
|
|
|
uint8_t readerdata = (previous_data & 0xF0) | (*data >> 4);
|
2019-04-07 16:17:43 +08:00
|
|
|
if (MillerDecoding(readerdata, (my_rsamples - 1) * 4)) {
|
2019-03-09 18:19:45 +08:00
|
|
|
LED_C_ON();
|
|
|
|
|
|
|
|
// check - if there is a short 7bit request from reader
|
2019-07-24 05:26:26 +08:00
|
|
|
if ((!triggered) && (param & 0x02) && (uart->len == 1) && (uart->bitCount == 7))
|
|
|
|
triggered = true;
|
2019-03-09 18:19:45 +08:00
|
|
|
|
|
|
|
if (triggered) {
|
2019-07-24 05:26:26 +08:00
|
|
|
if ((receivedCmd) &&
|
2019-07-24 06:20:00 +08:00
|
|
|
((receivedCmd[0] == MIFARE_ULEV1_AUTH) || (receivedCmd[0] == MIFARE_ULC_AUTH_1))) {
|
2019-07-24 05:26:26 +08:00
|
|
|
if (DBGLEVEL > 1)
|
|
|
|
Dbprintf("PWD-AUTH KEY: 0x%02x%02x%02x%02x", receivedCmd[1], receivedCmd[2],
|
|
|
|
receivedCmd[3], receivedCmd[4]);
|
2019-03-09 18:19:45 +08:00
|
|
|
|
|
|
|
// temporarily save the captured pwd in our array
|
2019-03-10 07:00:59 +08:00
|
|
|
memcpy(&capturedPwds[4 * auth_attempts], receivedCmd + 1, 4);
|
2019-03-09 18:19:45 +08:00
|
|
|
auth_attempts++;
|
|
|
|
}
|
|
|
|
|
2019-07-24 05:26:26 +08:00
|
|
|
if (!LogTrace(receivedCmd, uart->len, uart->startTime * 16 - DELAY_READER_AIR2ARM_AS_SNIFFER,
|
|
|
|
uart->endTime * 16 - DELAY_READER_AIR2ARM_AS_SNIFFER, uart->parity, true))
|
|
|
|
break;
|
2019-03-09 18:19:45 +08:00
|
|
|
}
|
|
|
|
/* ready to receive another command. */
|
2019-08-08 22:57:33 +08:00
|
|
|
Uart14aReset();
|
2019-03-09 18:19:45 +08:00
|
|
|
/* reset the demod code, which might have been */
|
|
|
|
/* false-triggered by the commands from the reader. */
|
2019-08-08 22:57:33 +08:00
|
|
|
Demod14aReset();
|
2019-03-09 18:19:45 +08:00
|
|
|
LED_B_OFF();
|
|
|
|
}
|
2019-08-08 22:57:33 +08:00
|
|
|
ReaderIsActive = (uart->state != STATE_14A_UNSYNCD);
|
2019-03-09 18:19:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// no need to try decoding tag data if the reader is sending - and we cannot afford the time
|
|
|
|
if (!ReaderIsActive) {
|
|
|
|
uint8_t tagdata = (previous_data << 4) | (*data & 0x0F);
|
2019-04-07 16:17:43 +08:00
|
|
|
if (ManchesterDecoding(tagdata, 0, (my_rsamples - 1) * 4)) {
|
2019-03-09 18:19:45 +08:00
|
|
|
LED_B_ON();
|
|
|
|
|
2019-07-24 05:26:26 +08:00
|
|
|
if (!LogTrace(receivedResp, demod->len, demod->startTime * 16 - DELAY_TAG_AIR2ARM_AS_SNIFFER,
|
|
|
|
demod->endTime * 16 - DELAY_TAG_AIR2ARM_AS_SNIFFER, demod->parity, false))
|
|
|
|
break;
|
2019-03-09 18:19:45 +08:00
|
|
|
|
2019-07-24 05:26:26 +08:00
|
|
|
if ((!triggered) && (param & 0x01))
|
|
|
|
triggered = true;
|
2019-03-09 18:19:45 +08:00
|
|
|
|
|
|
|
// ready to receive another response.
|
2019-08-08 22:57:33 +08:00
|
|
|
Demod14aReset();
|
2019-03-09 18:19:45 +08:00
|
|
|
// reset the Miller decoder including its (now outdated) input buffer
|
2019-08-08 22:57:33 +08:00
|
|
|
Uart14aReset();
|
2019-07-24 05:26:26 +08:00
|
|
|
// UartInit(receivedCmd, receivedCmdPar);
|
2019-03-09 18:19:45 +08:00
|
|
|
LED_C_OFF();
|
|
|
|
}
|
2019-08-08 22:57:33 +08:00
|
|
|
TagIsActive = (demod->state != DEMOD_14A_UNSYNCD);
|
2019-03-09 18:19:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
previous_data = *data;
|
2019-04-07 16:17:43 +08:00
|
|
|
my_rsamples++;
|
2019-03-09 18:19:45 +08:00
|
|
|
data++;
|
|
|
|
if (data == dmaBuf + DMA_BUFFER_SIZE) {
|
|
|
|
data = dmaBuf;
|
|
|
|
}
|
|
|
|
} // end main loop
|
|
|
|
|
|
|
|
FpgaDisableSscDma();
|
|
|
|
set_tracing(false);
|
|
|
|
|
|
|
|
Dbprintf("Stopped sniffing");
|
|
|
|
|
|
|
|
SpinDelay(200);
|
|
|
|
|
2019-07-24 05:26:26 +08:00
|
|
|
// Write stuff to spiffs logfile
|
2019-03-09 18:19:45 +08:00
|
|
|
if (auth_attempts > 0) {
|
2019-07-24 05:26:26 +08:00
|
|
|
if (DBGLEVEL > 1)
|
|
|
|
Dbprintf("[!] Authentication attempts = %u", auth_attempts);
|
2019-03-09 18:19:45 +08:00
|
|
|
|
2019-07-24 05:26:26 +08:00
|
|
|
if (!exists_in_spiffs((char *)HF_BOG_LOGFILE)) {
|
2019-11-03 02:06:02 +08:00
|
|
|
rdv40_spiffs_write((char *)HF_BOG_LOGFILE, capturedPwds, 4 * auth_attempts, RDV40_SPIFFS_SAFETY_SAFE);
|
2019-07-24 05:26:26 +08:00
|
|
|
} else {
|
2019-11-03 02:06:02 +08:00
|
|
|
rdv40_spiffs_append((char *)HF_BOG_LOGFILE, capturedPwds, 4 * auth_attempts, RDV40_SPIFFS_SAFETY_SAFE);
|
2019-03-09 18:19:45 +08:00
|
|
|
}
|
2019-07-24 05:26:26 +08:00
|
|
|
}
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-07-24 05:26:26 +08:00
|
|
|
if (DBGLEVEL > 1)
|
|
|
|
Dbprintf("[!] Wrote %u Authentification attempts into logfile", auth_attempts);
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-09-12 15:58:05 +08:00
|
|
|
SpinErr(LED_A, 200, 5);
|
2019-07-24 05:26:26 +08:00
|
|
|
SpinDelay(100);
|
2018-10-17 03:50:18 +08:00
|
|
|
}
|
|
|
|
|
2019-04-26 03:44:34 +08:00
|
|
|
void ModInfo(void) {
|
2019-07-10 04:49:57 +08:00
|
|
|
DbpString(" HF 14a sniff standalone with ULC/ULEV1/NTAG auth storing in flashmem - aka BogitoRun (Bogito)");
|
2019-04-26 03:44:34 +08:00
|
|
|
}
|
|
|
|
|
2020-05-10 22:59:38 +08:00
|
|
|
void RunMod(void) {
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
StandAloneMode();
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
Dbprintf(">> Bogiton 14a Sniff UL/UL-EV1/NTAG a.k.a BogitoRun Started <<");
|
|
|
|
Dbprintf("Starting to sniff");
|
2019-03-09 15:59:13 +08:00
|
|
|
|
2019-03-09 18:19:45 +08:00
|
|
|
// param:
|
|
|
|
// bit 0 - trigger from first card answer
|
|
|
|
// bit 1 - trigger from first reader 7-bit request
|
|
|
|
SniffAndStore(0);
|
|
|
|
LEDsoff();
|
|
|
|
SpinDelay(300);
|
|
|
|
Dbprintf("- [ End ] -> You can take shell back ...");
|
2019-11-09 01:42:17 +08:00
|
|
|
Dbprintf("- [ ! ] -> use 'script run read_pwd_mem_spiffs' to print passwords");
|
2018-10-17 03:50:18 +08:00
|
|
|
}
|