mirror of
https://github.com/DarkFlippers/unleashed-firmware
synced 2024-12-24 03:33:08 +00:00
422 lines
No EOL
13 KiB
C
422 lines
No EOL
13 KiB
C
#include "subbrute_protocols.h"
|
|
#include <string.h>
|
|
|
|
#define TAG "SubBruteProtocols"
|
|
|
|
/**
|
|
* CAME 12bit 303MHz
|
|
*/
|
|
const SubBruteProtocol subbrute_protocol_came_12bit_303 = {
|
|
.frequency = 303875000,
|
|
.bits = 12,
|
|
.te = 0,
|
|
.repeat = 3,
|
|
.preset = FuriHalSubGhzPresetOok650Async,
|
|
.file = CAMEFileProtocol};
|
|
|
|
/**
|
|
* CAME 12bit 307MHz
|
|
*/
|
|
const SubBruteProtocol subbrute_protocol_came_12bit_307 = {
|
|
.frequency = 307800000,
|
|
.bits = 12,
|
|
.te = 0,
|
|
.repeat = 3,
|
|
.preset = FuriHalSubGhzPresetOok650Async,
|
|
.file = CAMEFileProtocol};
|
|
|
|
/**
|
|
* CAME 12bit 433MHz
|
|
*/
|
|
const SubBruteProtocol subbrute_protocol_came_12bit_433 = {
|
|
.frequency = 433920000,
|
|
.bits = 12,
|
|
.te = 0,
|
|
.repeat = 3,
|
|
.preset = FuriHalSubGhzPresetOok650Async,
|
|
.file = CAMEFileProtocol};
|
|
|
|
/**
|
|
* CAME 12bit 868MHz
|
|
*/
|
|
const SubBruteProtocol subbrute_protocol_came_12bit_868 = {
|
|
.frequency = 868350000,
|
|
.bits = 12,
|
|
.te = 0,
|
|
.repeat = 3,
|
|
.preset = FuriHalSubGhzPresetOok650Async,
|
|
.file = CAMEFileProtocol};
|
|
|
|
/**
|
|
* NICE 12bit 433MHz
|
|
*/
|
|
const SubBruteProtocol subbrute_protocol_nice_12bit_433 = {
|
|
.frequency = 433920000,
|
|
.bits = 12,
|
|
.te = 0,
|
|
.repeat = 3,
|
|
.preset = FuriHalSubGhzPresetOok650Async,
|
|
.file = NICEFileProtocol};
|
|
|
|
/**
|
|
* NICE 12bit 868MHz
|
|
*/
|
|
const SubBruteProtocol subbrute_protocol_nice_12bit_868 = {
|
|
.frequency = 868350000,
|
|
.bits = 12,
|
|
.te = 0,
|
|
.repeat = 3,
|
|
.preset = FuriHalSubGhzPresetOok650Async,
|
|
.file = NICEFileProtocol};
|
|
|
|
/**
|
|
* Chamberlain 9bit 300MHz
|
|
*/
|
|
const SubBruteProtocol subbrute_protocol_chamberlain_9bit_300 = {
|
|
.frequency = 300000000,
|
|
.bits = 9,
|
|
.te = 0,
|
|
.repeat = 3,
|
|
.preset = FuriHalSubGhzPresetOok650Async,
|
|
.file = ChamberlainFileProtocol};
|
|
|
|
/**
|
|
* Chamberlain 9bit 315MHz
|
|
*/
|
|
const SubBruteProtocol subbrute_protocol_chamberlain_9bit_315 = {
|
|
.frequency = 315000000,
|
|
.bits = 9,
|
|
.te = 0,
|
|
.repeat = 3,
|
|
.preset = FuriHalSubGhzPresetOok650Async,
|
|
.file = ChamberlainFileProtocol};
|
|
|
|
/**
|
|
* Chamberlain 9bit 390MHz
|
|
*/
|
|
const SubBruteProtocol subbrute_protocol_chamberlain_9bit_390 = {
|
|
.frequency = 390000000,
|
|
.bits = 9,
|
|
.te = 0,
|
|
.repeat = 3,
|
|
.preset = FuriHalSubGhzPresetOok650Async,
|
|
.file = ChamberlainFileProtocol};
|
|
|
|
/**
|
|
* Linear 10bit 300MHz
|
|
*/
|
|
const SubBruteProtocol subbrute_protocol_linear_10bit_300 = {
|
|
.frequency = 300000000,
|
|
.bits = 10,
|
|
.te = 0,
|
|
.repeat = 5,
|
|
.preset = FuriHalSubGhzPresetOok650Async,
|
|
.file = LinearFileProtocol};
|
|
|
|
/**
|
|
* Linear 10bit 310MHz
|
|
*/
|
|
const SubBruteProtocol subbrute_protocol_linear_10bit_310 = {
|
|
.frequency = 310000000,
|
|
.bits = 10,
|
|
.te = 0,
|
|
.repeat = 5,
|
|
.preset = FuriHalSubGhzPresetOok650Async,
|
|
.file = LinearFileProtocol};
|
|
|
|
/**
|
|
* BF existing dump
|
|
*/
|
|
const SubBruteProtocol subbrute_protocol_load_file =
|
|
{0, 0, 0, 3, FuriHalSubGhzPresetOok650Async, RAWFileProtocol};
|
|
|
|
static const char* subbrute_protocol_names[] = {
|
|
[SubBruteAttackCAME12bit303] = "CAME 12bit 303MHz",
|
|
[SubBruteAttackCAME12bit307] = "CAME 12bit 307MHz",
|
|
[SubBruteAttackCAME12bit433] = "CAME 12bit 433MHz",
|
|
[SubBruteAttackCAME12bit868] = "CAME 12bit 868MHz",
|
|
[SubBruteAttackNICE12bit433] = "NICE 12bit 433MHz",
|
|
[SubBruteAttackNICE12bit868] = "NICE 12bit 868MHz",
|
|
[SubBruteAttackChamberlain9bit300] = "Chamberlain 9bit 300MHz",
|
|
[SubBruteAttackChamberlain9bit315] = "Chamberlain 9bit 315MHz",
|
|
[SubBruteAttackChamberlain9bit390] = "Chamberlain 9bit 390MHz",
|
|
[SubBruteAttackLinear10bit300] = "Linear 10bit 300MHz",
|
|
[SubBruteAttackLinear10bit310] = "Linear 10bit 310MHz",
|
|
[SubBruteAttackLoadFile] = "BF existing dump",
|
|
[SubBruteAttackTotalCount] = "Total Count",
|
|
};
|
|
|
|
static const char* subbrute_protocol_presets[] = {
|
|
[FuriHalSubGhzPresetIDLE] = "FuriHalSubGhzPresetIDLE",
|
|
[FuriHalSubGhzPresetOok270Async] = "FuriHalSubGhzPresetOok270Async",
|
|
[FuriHalSubGhzPresetOok650Async] = "FuriHalSubGhzPresetOok650Async",
|
|
[FuriHalSubGhzPreset2FSKDev238Async] = "FuriHalSubGhzPreset2FSKDev238Async",
|
|
[FuriHalSubGhzPreset2FSKDev476Async] = "FuriHalSubGhzPreset2FSKDev476Async",
|
|
[FuriHalSubGhzPresetMSK99_97KbAsync] = "FuriHalSubGhzPresetMSK99_97KbAsync",
|
|
[FuriHalSubGhzPresetGFSK9_99KbAsync] = "FuriHalSubGhzPresetGFSK9_99KbAsync",
|
|
};
|
|
|
|
const SubBruteProtocol* subbrute_protocol_registry[] = {
|
|
[SubBruteAttackCAME12bit303] = &subbrute_protocol_came_12bit_303,
|
|
[SubBruteAttackCAME12bit307] = &subbrute_protocol_came_12bit_307,
|
|
[SubBruteAttackCAME12bit433] = &subbrute_protocol_came_12bit_433,
|
|
[SubBruteAttackCAME12bit868] = &subbrute_protocol_came_12bit_868,
|
|
[SubBruteAttackNICE12bit433] = &subbrute_protocol_nice_12bit_433,
|
|
[SubBruteAttackNICE12bit868] = &subbrute_protocol_nice_12bit_868,
|
|
[SubBruteAttackChamberlain9bit300] = &subbrute_protocol_chamberlain_9bit_300,
|
|
[SubBruteAttackChamberlain9bit315] = &subbrute_protocol_chamberlain_9bit_315,
|
|
[SubBruteAttackChamberlain9bit390] = &subbrute_protocol_chamberlain_9bit_390,
|
|
[SubBruteAttackLinear10bit300] = &subbrute_protocol_linear_10bit_300,
|
|
[SubBruteAttackLinear10bit310] = &subbrute_protocol_linear_10bit_310,
|
|
[SubBruteAttackLoadFile] = &subbrute_protocol_load_file};
|
|
|
|
static const char* subbrute_protocol_file_types[] = {
|
|
[CAMEFileProtocol] = "CAME",
|
|
[NICEFileProtocol] = "Nice FLO",
|
|
[ChamberlainFileProtocol] = "Cham_Code",
|
|
[LinearFileProtocol] = "Linear",
|
|
[PrincetonFileProtocol] = "Princeton",
|
|
[RAWFileProtocol] = "RAW"};
|
|
|
|
/**
|
|
* Values to not use less memory for packet parse operations
|
|
*/
|
|
static const char* subbrute_key_file_start_no_tail =
|
|
"Filetype: Flipper SubGhz Key File\nVersion: 1\nFrequency: %u\nPreset: %s\nProtocol: %s\nBit: %d\nKey: %s\nRepeat: %d\n";
|
|
static const char* subbrute_key_file_start_with_tail =
|
|
"Filetype: Flipper SubGhz Key File\nVersion: 1\nFrequency: %u\nPreset: %s\nProtocol: %s\nBit: %d\nKey: %s\nTE: %d\nRepeat: %d\n";
|
|
static const char* subbrute_key_small_no_tail = "Bit: %d\nKey: %s\nRepeat: %d\n";
|
|
static const char* subbrute_key_small_with_tail = "Bit: %d\nKey: %s\nTE: %d\nRepeat: %d\n";
|
|
|
|
const char* subbrute_protocol_name(SubBruteAttacks index) {
|
|
return subbrute_protocol_names[index];
|
|
}
|
|
|
|
const SubBruteProtocol* subbrute_protocol(SubBruteAttacks index) {
|
|
return subbrute_protocol_registry[index];
|
|
}
|
|
|
|
const char* subbrute_protocol_preset(FuriHalSubGhzPreset preset) {
|
|
return subbrute_protocol_presets[preset];
|
|
}
|
|
|
|
const char* subbrute_protocol_file(SubBruteFileProtocol protocol) {
|
|
return subbrute_protocol_file_types[protocol];
|
|
}
|
|
|
|
FuriHalSubGhzPreset subbrute_protocol_convert_preset(FuriString* preset_name) {
|
|
for(size_t i = FuriHalSubGhzPresetIDLE; i < FuriHalSubGhzPresetCustom; i++) {
|
|
if(furi_string_cmp_str(preset_name, subbrute_protocol_presets[i]) == 0) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return FuriHalSubGhzPresetIDLE;
|
|
}
|
|
|
|
SubBruteFileProtocol subbrute_protocol_file_protocol_name(FuriString* name) {
|
|
for(size_t i = CAMEFileProtocol; i < TotalFileProtocol - 1; i++) {
|
|
if(furi_string_cmp_str(name, subbrute_protocol_file_types[i]) == 0) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return RAWFileProtocol;
|
|
}
|
|
|
|
void subbrute_protocol_default_payload(
|
|
Stream* stream,
|
|
uint64_t step,
|
|
uint8_t bits,
|
|
uint8_t te,
|
|
uint8_t repeat) {
|
|
FuriString* candidate = furi_string_alloc_set_str(" ");
|
|
|
|
FuriString* buffer = furi_string_alloc_printf("%16llX", step);
|
|
int j = 0;
|
|
for(uint8_t i = 0; i < 16; i++) {
|
|
if(furi_string_get_char(buffer, i) != ' ') {
|
|
furi_string_set_char(candidate, i + j, furi_string_get_char(buffer, i));
|
|
} else {
|
|
furi_string_set_char(candidate, i + j, '0');
|
|
}
|
|
if(i % 2 != 0) {
|
|
j++;
|
|
}
|
|
}
|
|
furi_string_free(buffer);
|
|
|
|
#ifdef FURI_DEBUG
|
|
//FURI_LOG_D(TAG, "candidate: %s, step: %lld", furi_string_get_cstr(candidate), step);
|
|
#endif
|
|
stream_clean(stream);
|
|
if(te) {
|
|
stream_write_format(
|
|
stream,
|
|
subbrute_key_small_with_tail,
|
|
bits,
|
|
furi_string_get_cstr(candidate),
|
|
te,
|
|
repeat);
|
|
} else {
|
|
stream_write_format(
|
|
stream, subbrute_key_small_no_tail, bits, furi_string_get_cstr(candidate), repeat);
|
|
}
|
|
|
|
furi_string_free(candidate);
|
|
}
|
|
|
|
void subbrute_protocol_file_payload(
|
|
Stream* stream,
|
|
uint64_t step,
|
|
uint8_t bits,
|
|
uint8_t te,
|
|
uint8_t repeat,
|
|
uint8_t load_index,
|
|
const char* file_key) {
|
|
FuriString* candidate = furi_string_alloc();
|
|
char subbrute_payload_byte[4];
|
|
furi_string_set_str(candidate, file_key);
|
|
snprintf(subbrute_payload_byte, 4, "%02X ", (uint8_t)step);
|
|
furi_string_replace_at(candidate, load_index * 3, 3, subbrute_payload_byte);
|
|
|
|
#ifdef FURI_DEBUG
|
|
//FURI_LOG_D(TAG, "candidate: %s, step: %lld", furi_string_get_cstr(candidate), step);
|
|
#endif
|
|
stream_clean(stream);
|
|
|
|
if(te) {
|
|
stream_write_format(
|
|
stream,
|
|
subbrute_key_small_with_tail,
|
|
bits,
|
|
furi_string_get_cstr(candidate),
|
|
te,
|
|
repeat);
|
|
} else {
|
|
stream_write_format(
|
|
stream, subbrute_key_small_no_tail, bits, furi_string_get_cstr(candidate), repeat);
|
|
}
|
|
|
|
furi_string_free(candidate);
|
|
}
|
|
|
|
void subbrute_protocol_default_generate_file(
|
|
Stream* stream,
|
|
uint32_t frequency,
|
|
FuriHalSubGhzPreset preset,
|
|
SubBruteFileProtocol file,
|
|
uint64_t step,
|
|
uint8_t bits,
|
|
uint8_t te,
|
|
uint8_t repeat) {
|
|
FuriString* candidate = furi_string_alloc_set_str(" ");
|
|
|
|
FuriString* buffer = furi_string_alloc_printf("%16llX", step);
|
|
int j = 0;
|
|
for(uint8_t i = 0; i < 16; i++) {
|
|
if(furi_string_get_char(buffer, i) != ' ') {
|
|
furi_string_set_char(candidate, i + j, furi_string_get_char(buffer, i));
|
|
} else {
|
|
furi_string_set_char(candidate, i + j, '0');
|
|
}
|
|
if(i % 2 != 0) {
|
|
j++;
|
|
}
|
|
}
|
|
furi_string_free(buffer);
|
|
|
|
#ifdef FURI_DEBUG
|
|
FURI_LOG_D(TAG, "candidate: %s, step: %lld", furi_string_get_cstr(candidate), step);
|
|
#endif
|
|
stream_clean(stream);
|
|
|
|
if(te) {
|
|
stream_write_format(
|
|
stream,
|
|
subbrute_key_file_start_with_tail,
|
|
frequency,
|
|
subbrute_protocol_preset(preset),
|
|
subbrute_protocol_file(file),
|
|
bits,
|
|
furi_string_get_cstr(candidate),
|
|
te,
|
|
repeat);
|
|
} else {
|
|
stream_write_format(
|
|
stream,
|
|
subbrute_key_file_start_no_tail,
|
|
frequency,
|
|
subbrute_protocol_preset(preset),
|
|
subbrute_protocol_file(file),
|
|
bits,
|
|
furi_string_get_cstr(candidate),
|
|
repeat);
|
|
}
|
|
|
|
furi_string_free(candidate);
|
|
}
|
|
|
|
void subbrute_protocol_file_generate_file(
|
|
Stream* stream,
|
|
uint32_t frequency,
|
|
FuriHalSubGhzPreset preset,
|
|
SubBruteFileProtocol file,
|
|
uint64_t step,
|
|
uint8_t bits,
|
|
uint8_t te,
|
|
uint8_t repeat,
|
|
uint8_t load_index,
|
|
const char* file_key) {
|
|
FuriString* candidate = furi_string_alloc();
|
|
char subbrute_payload_byte[4];
|
|
furi_string_set_str(candidate, file_key);
|
|
snprintf(subbrute_payload_byte, 4, "%02X ", (uint8_t)step);
|
|
furi_string_replace_at(candidate, load_index * 3, 3, subbrute_payload_byte);
|
|
|
|
#ifdef FURI_DEBUG
|
|
FURI_LOG_D(TAG, "candidate: %s, step: %lld", furi_string_get_cstr(candidate), step);
|
|
#endif
|
|
stream_clean(stream);
|
|
if(te) {
|
|
stream_write_format(
|
|
stream,
|
|
subbrute_key_file_start_with_tail,
|
|
frequency,
|
|
subbrute_protocol_preset(preset),
|
|
subbrute_protocol_file(file),
|
|
bits,
|
|
furi_string_get_cstr(candidate),
|
|
te,
|
|
repeat);
|
|
} else {
|
|
stream_write_format(
|
|
stream,
|
|
subbrute_key_file_start_no_tail,
|
|
frequency,
|
|
subbrute_protocol_preset(preset),
|
|
subbrute_protocol_file(file),
|
|
bits,
|
|
furi_string_get_cstr(candidate),
|
|
repeat);
|
|
}
|
|
|
|
furi_string_free(candidate);
|
|
}
|
|
|
|
uint64_t subbrute_protocol_calc_max_value(SubBruteAttacks attack_type, uint8_t bits) {
|
|
uint64_t max_value;
|
|
if(attack_type == SubBruteAttackLoadFile) {
|
|
max_value = 0x3F;
|
|
} else {
|
|
FuriString* max_value_s;
|
|
max_value_s = furi_string_alloc();
|
|
for(uint8_t i = 0; i < bits; i++) {
|
|
furi_string_cat_printf(max_value_s, "1");
|
|
}
|
|
max_value = (uint64_t)strtol(furi_string_get_cstr(max_value_s), NULL, 2);
|
|
furi_string_free(max_value_s);
|
|
}
|
|
|
|
return max_value;
|
|
} |