|
@@ -190,12 +190,87 @@ ReturnCode picopass_read_card(PicopassBlock* AA1) {
|
|
|
return ERR_NONE;
|
|
return ERR_NONE;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
+ReturnCode picopass_write_card(PicopassBlock* AA1) {
|
|
|
|
|
+ rfalPicoPassIdentifyRes idRes;
|
|
|
|
|
+ rfalPicoPassSelectRes selRes;
|
|
|
|
|
+ rfalPicoPassReadCheckRes rcRes;
|
|
|
|
|
+ rfalPicoPassCheckRes chkRes;
|
|
|
|
|
+
|
|
|
|
|
+ ReturnCode err;
|
|
|
|
|
+
|
|
|
|
|
+ uint8_t div_key[8] = {0};
|
|
|
|
|
+ uint8_t mac[4] = {0};
|
|
|
|
|
+ uint8_t ccnr[12] = {0};
|
|
|
|
|
+
|
|
|
|
|
+ err = rfalPicoPassPollerIdentify(&idRes);
|
|
|
|
|
+ if(err != ERR_NONE) {
|
|
|
|
|
+ FURI_LOG_E(TAG, "rfalPicoPassPollerIdentify error %d", err);
|
|
|
|
|
+ return err;
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ err = rfalPicoPassPollerSelect(idRes.CSN, &selRes);
|
|
|
|
|
+ if(err != ERR_NONE) {
|
|
|
|
|
+ FURI_LOG_E(TAG, "rfalPicoPassPollerSelect error %d", err);
|
|
|
|
|
+ return err;
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ err = rfalPicoPassPollerReadCheck(&rcRes);
|
|
|
|
|
+ if(err != ERR_NONE) {
|
|
|
|
|
+ FURI_LOG_E(TAG, "rfalPicoPassPollerReadCheck error %d", err);
|
|
|
|
|
+ return err;
|
|
|
|
|
+ }
|
|
|
|
|
+ memcpy(ccnr, rcRes.CCNR, sizeof(rcRes.CCNR)); // last 4 bytes left 0
|
|
|
|
|
+
|
|
|
|
|
+ loclass_diversifyKey(selRes.CSN, picopass_iclass_key, div_key);
|
|
|
|
|
+ loclass_opt_doReaderMAC(ccnr, div_key, mac);
|
|
|
|
|
+
|
|
|
|
|
+ err = rfalPicoPassPollerCheck(mac, &chkRes);
|
|
|
|
|
+ if(err != ERR_NONE) {
|
|
|
|
|
+ FURI_LOG_E(TAG, "rfalPicoPassPollerCheck error %d", err);
|
|
|
|
|
+ return err;
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ for(size_t i = 6; i < 10; i++) {
|
|
|
|
|
+ FURI_LOG_D(TAG, "rfalPicoPassPollerWriteBlock %d", i);
|
|
|
|
|
+ uint8_t data[9] = {0};
|
|
|
|
|
+ data[0] = i;
|
|
|
|
|
+ memcpy(data + 1, AA1[i].data, RFAL_PICOPASS_MAX_BLOCK_LEN);
|
|
|
|
|
+ loclass_doMAC_N(data, sizeof(data), div_key, mac);
|
|
|
|
|
+ FURI_LOG_D(
|
|
|
|
|
+ TAG,
|
|
|
|
|
+ "loclass_doMAC_N %d %02x%02x%02x%02x%02x%02x%02x%02x %02x%02x%02x%02x",
|
|
|
|
|
+ i,
|
|
|
|
|
+ data[1],
|
|
|
|
|
+ data[2],
|
|
|
|
|
+ data[3],
|
|
|
|
|
+ data[4],
|
|
|
|
|
+ data[5],
|
|
|
|
|
+ data[6],
|
|
|
|
|
+ data[7],
|
|
|
|
|
+ data[8],
|
|
|
|
|
+ mac[0],
|
|
|
|
|
+ mac[1],
|
|
|
|
|
+ mac[2],
|
|
|
|
|
+ mac[3]);
|
|
|
|
|
+
|
|
|
|
|
+ err = rfalPicoPassPollerWriteBlock(i, AA1[i].data, mac);
|
|
|
|
|
+ if(err != ERR_NONE) {
|
|
|
|
|
+ FURI_LOG_E(TAG, "rfalPicoPassPollerWriteBlock error %d", err);
|
|
|
|
|
+ return err;
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ return ERR_NONE;
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
int32_t picopass_worker_task(void* context) {
|
|
int32_t picopass_worker_task(void* context) {
|
|
|
PicopassWorker* picopass_worker = context;
|
|
PicopassWorker* picopass_worker = context;
|
|
|
|
|
|
|
|
picopass_worker_enable_field();
|
|
picopass_worker_enable_field();
|
|
|
if(picopass_worker->state == PicopassWorkerStateDetect) {
|
|
if(picopass_worker->state == PicopassWorkerStateDetect) {
|
|
|
picopass_worker_detect(picopass_worker);
|
|
picopass_worker_detect(picopass_worker);
|
|
|
|
|
+ } else if(picopass_worker->state == PicopassWorkerStateWrite) {
|
|
|
|
|
+ picopass_worker_write(picopass_worker);
|
|
|
}
|
|
}
|
|
|
picopass_worker_disable_field(ERR_NONE);
|
|
picopass_worker_disable_field(ERR_NONE);
|
|
|
|
|
|
|
@@ -212,27 +287,60 @@ void picopass_worker_detect(PicopassWorker* picopass_worker) {
|
|
|
PicopassPacs* pacs = &dev_data->pacs;
|
|
PicopassPacs* pacs = &dev_data->pacs;
|
|
|
ReturnCode err;
|
|
ReturnCode err;
|
|
|
|
|
|
|
|
|
|
+ PicopassWorkerEvent nextState = PicopassWorkerEventSuccess;
|
|
|
|
|
+
|
|
|
while(picopass_worker->state == PicopassWorkerStateDetect) {
|
|
while(picopass_worker->state == PicopassWorkerStateDetect) {
|
|
|
if(picopass_detect_card(1000) == ERR_NONE) {
|
|
if(picopass_detect_card(1000) == ERR_NONE) {
|
|
|
// Process first found device
|
|
// Process first found device
|
|
|
err = picopass_read_card(AA1);
|
|
err = picopass_read_card(AA1);
|
|
|
if(err != ERR_NONE) {
|
|
if(err != ERR_NONE) {
|
|
|
FURI_LOG_E(TAG, "picopass_read_card error %d", err);
|
|
FURI_LOG_E(TAG, "picopass_read_card error %d", err);
|
|
|
|
|
+ nextState = PicopassWorkerEventFail;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- err = picopass_device_parse_credential(AA1, pacs);
|
|
|
|
|
|
|
+ if(nextState == PicopassWorkerEventSuccess) {
|
|
|
|
|
+ err = picopass_device_parse_credential(AA1, pacs);
|
|
|
|
|
+ }
|
|
|
if(err != ERR_NONE) {
|
|
if(err != ERR_NONE) {
|
|
|
FURI_LOG_E(TAG, "picopass_device_parse_credential error %d", err);
|
|
FURI_LOG_E(TAG, "picopass_device_parse_credential error %d", err);
|
|
|
|
|
+ nextState = PicopassWorkerEventFail;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- err = picopass_device_parse_wiegand(pacs->credential, &pacs->record);
|
|
|
|
|
|
|
+ if(nextState == PicopassWorkerEventSuccess) {
|
|
|
|
|
+ err = picopass_device_parse_wiegand(pacs->credential, &pacs->record);
|
|
|
|
|
+ }
|
|
|
if(err != ERR_NONE) {
|
|
if(err != ERR_NONE) {
|
|
|
FURI_LOG_E(TAG, "picopass_device_parse_wiegand error %d", err);
|
|
FURI_LOG_E(TAG, "picopass_device_parse_wiegand error %d", err);
|
|
|
|
|
+ nextState = PicopassWorkerEventFail;
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ // Notify caller and exit
|
|
|
|
|
+ if(picopass_worker->callback) {
|
|
|
|
|
+ picopass_worker->callback(nextState, picopass_worker->context);
|
|
|
|
|
+ }
|
|
|
|
|
+ break;
|
|
|
|
|
+ }
|
|
|
|
|
+ furi_delay_ms(100);
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+void picopass_worker_write(PicopassWorker* picopass_worker) {
|
|
|
|
|
+ PicopassDeviceData* dev_data = picopass_worker->dev_data;
|
|
|
|
|
+ PicopassBlock* AA1 = dev_data->AA1;
|
|
|
|
|
+ ReturnCode err;
|
|
|
|
|
+ PicopassWorkerEvent nextState = PicopassWorkerEventSuccess;
|
|
|
|
|
+
|
|
|
|
|
+ while(picopass_worker->state == PicopassWorkerStateWrite) {
|
|
|
|
|
+ if(picopass_detect_card(1000) == ERR_NONE) {
|
|
|
|
|
+ err = picopass_write_card(AA1);
|
|
|
|
|
+ if(err != ERR_NONE) {
|
|
|
|
|
+ FURI_LOG_E(TAG, "picopass_write_card error %d", err);
|
|
|
|
|
+ nextState = PicopassWorkerEventFail;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// Notify caller and exit
|
|
// Notify caller and exit
|
|
|
if(picopass_worker->callback) {
|
|
if(picopass_worker->callback) {
|
|
|
- picopass_worker->callback(PicopassWorkerEventSuccess, picopass_worker->context);
|
|
|
|
|
|
|
+ picopass_worker->callback(nextState, picopass_worker->context);
|
|
|
}
|
|
}
|
|
|
break;
|
|
break;
|
|
|
}
|
|
}
|