|
@@ -1,6 +1,11 @@
|
|
|
#include "subghz_cli.h"
|
|
#include "subghz_cli.h"
|
|
|
|
|
+
|
|
|
#include <furi.h>
|
|
#include <furi.h>
|
|
|
#include <api-hal.h>
|
|
#include <api-hal.h>
|
|
|
|
|
+#include <stream_buffer.h>
|
|
|
|
|
+
|
|
|
|
|
+#define CC1101_FREQUENCY_RANGE_STR \
|
|
|
|
|
+ "300000000...348000000 or 387000000...464000000 or 779000000...928000000"
|
|
|
|
|
|
|
|
static const uint8_t subghz_test_packet_data[] = {
|
|
static const uint8_t subghz_test_packet_data[] = {
|
|
|
0x30, // 48bytes to transmit
|
|
0x30, // 48bytes to transmit
|
|
@@ -9,6 +14,15 @@ static const uint8_t subghz_test_packet_data[] = {
|
|
|
0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77,
|
|
0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77, 0x77,
|
|
|
};
|
|
};
|
|
|
|
|
|
|
|
|
|
+bool subghz_check_frequency_range(uint32_t frequency) {
|
|
|
|
|
+ if(!(frequency >= 300000000 && frequency <= 348000000) &&
|
|
|
|
|
+ !(frequency >= 387000000 && frequency <= 464000000) &&
|
|
|
|
|
+ !(frequency >= 779000000 && frequency <= 928000000)) {
|
|
|
|
|
+ return false;
|
|
|
|
|
+ }
|
|
|
|
|
+ return true;
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
void subghz_cli_init() {
|
|
void subghz_cli_init() {
|
|
|
Cli* cli = furi_record_open("cli");
|
|
Cli* cli = furi_record_open("cli");
|
|
|
|
|
|
|
@@ -16,12 +30,14 @@ void subghz_cli_init() {
|
|
|
cli_add_command(cli, "subghz_rx_carrier", subghz_cli_command_rx_carrier, NULL);
|
|
cli_add_command(cli, "subghz_rx_carrier", subghz_cli_command_rx_carrier, NULL);
|
|
|
cli_add_command(cli, "subghz_tx_pt", subghz_cli_command_tx_pt, NULL);
|
|
cli_add_command(cli, "subghz_tx_pt", subghz_cli_command_tx_pt, NULL);
|
|
|
cli_add_command(cli, "subghz_rx_pt", subghz_cli_command_rx_pt, NULL);
|
|
cli_add_command(cli, "subghz_rx_pt", subghz_cli_command_rx_pt, NULL);
|
|
|
|
|
+ cli_add_command(cli, "subghz_tx", subghz_cli_command_tx, NULL);
|
|
|
|
|
+ cli_add_command(cli, "subghz_rx", subghz_cli_command_rx, NULL);
|
|
|
|
|
|
|
|
furi_record_close("cli");
|
|
furi_record_close("cli");
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void subghz_cli_command_tx_carrier(Cli* cli, string_t args, void* context) {
|
|
void subghz_cli_command_tx_carrier(Cli* cli, string_t args, void* context) {
|
|
|
- uint32_t frequency;
|
|
|
|
|
|
|
+ uint32_t frequency = 0;
|
|
|
int ret = sscanf(string_get_cstr(args), "%lu", &frequency);
|
|
int ret = sscanf(string_get_cstr(args), "%lu", &frequency);
|
|
|
if(ret != 1) {
|
|
if(ret != 1) {
|
|
|
printf("sscanf returned %d, frequency: %lu\r\n", ret, frequency);
|
|
printf("sscanf returned %d, frequency: %lu\r\n", ret, frequency);
|
|
@@ -29,23 +45,17 @@ void subghz_cli_command_tx_carrier(Cli* cli, string_t args, void* context) {
|
|
|
return;
|
|
return;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- if(frequency < 300000000 || frequency > 925000000) {
|
|
|
|
|
- printf("Frequency must be in 300000000...925000000 range, not %lu\r\n", frequency);
|
|
|
|
|
|
|
+ if(!subghz_check_frequency_range(frequency)) {
|
|
|
|
|
+ printf(
|
|
|
|
|
+ "Frequency must be in " CC1101_FREQUENCY_RANGE_STR " range, not %lu\r\n", frequency);
|
|
|
return;
|
|
return;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
api_hal_subghz_reset();
|
|
api_hal_subghz_reset();
|
|
|
api_hal_subghz_load_preset(ApiHalSubGhzPresetOokAsync);
|
|
api_hal_subghz_load_preset(ApiHalSubGhzPresetOokAsync);
|
|
|
- frequency = api_hal_subghz_set_frequency(frequency);
|
|
|
|
|
|
|
+ frequency = api_hal_subghz_set_frequency_and_path(frequency);
|
|
|
printf("Transmitting at frequency %lu Hz\r\n", frequency);
|
|
printf("Transmitting at frequency %lu Hz\r\n", frequency);
|
|
|
printf("Press CTRL+C to stop\r\n");
|
|
printf("Press CTRL+C to stop\r\n");
|
|
|
- if(frequency < 400000000) {
|
|
|
|
|
- api_hal_subghz_set_path(ApiHalSubGhzPath315);
|
|
|
|
|
- } else if(frequency < 500000000) {
|
|
|
|
|
- api_hal_subghz_set_path(ApiHalSubGhzPath433);
|
|
|
|
|
- } else {
|
|
|
|
|
- api_hal_subghz_set_path(ApiHalSubGhzPath868);
|
|
|
|
|
- }
|
|
|
|
|
|
|
|
|
|
hal_gpio_init(&gpio_cc1101_g0, GpioModeOutputPushPull, GpioPullNo, GpioSpeedLow);
|
|
hal_gpio_init(&gpio_cc1101_g0, GpioModeOutputPushPull, GpioPullNo, GpioSpeedLow);
|
|
|
hal_gpio_write(&gpio_cc1101_g0, false);
|
|
hal_gpio_write(&gpio_cc1101_g0, false);
|
|
@@ -62,7 +72,7 @@ void subghz_cli_command_tx_carrier(Cli* cli, string_t args, void* context) {
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void subghz_cli_command_rx_carrier(Cli* cli, string_t args, void* context) {
|
|
void subghz_cli_command_rx_carrier(Cli* cli, string_t args, void* context) {
|
|
|
- uint32_t frequency;
|
|
|
|
|
|
|
+ uint32_t frequency = 0;
|
|
|
int ret = sscanf(string_get_cstr(args), "%lu", &frequency);
|
|
int ret = sscanf(string_get_cstr(args), "%lu", &frequency);
|
|
|
if(ret != 1) {
|
|
if(ret != 1) {
|
|
|
printf("sscanf returned %d, frequency: %lu\r\n", ret, frequency);
|
|
printf("sscanf returned %d, frequency: %lu\r\n", ret, frequency);
|
|
@@ -70,23 +80,17 @@ void subghz_cli_command_rx_carrier(Cli* cli, string_t args, void* context) {
|
|
|
return;
|
|
return;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- if(frequency < 300000000 || frequency > 925000000) {
|
|
|
|
|
- printf("Frequency must be in 300000000...925000000 range, not %lu\r\n", frequency);
|
|
|
|
|
|
|
+ if(!subghz_check_frequency_range(frequency)) {
|
|
|
|
|
+ printf(
|
|
|
|
|
+ "Frequency must be in " CC1101_FREQUENCY_RANGE_STR " range, not %lu\r\n", frequency);
|
|
|
return;
|
|
return;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
api_hal_subghz_reset();
|
|
api_hal_subghz_reset();
|
|
|
api_hal_subghz_load_preset(ApiHalSubGhzPresetOokAsync);
|
|
api_hal_subghz_load_preset(ApiHalSubGhzPresetOokAsync);
|
|
|
- frequency = api_hal_subghz_set_frequency(frequency);
|
|
|
|
|
|
|
+ frequency = api_hal_subghz_set_frequency_and_path(frequency);
|
|
|
printf("Receiving at frequency %lu Hz\r\n", frequency);
|
|
printf("Receiving at frequency %lu Hz\r\n", frequency);
|
|
|
printf("Press CTRL+C to stop\r\n");
|
|
printf("Press CTRL+C to stop\r\n");
|
|
|
- if(frequency < 400000000) {
|
|
|
|
|
- api_hal_subghz_set_path(ApiHalSubGhzPath315);
|
|
|
|
|
- } else if(frequency < 500000000) {
|
|
|
|
|
- api_hal_subghz_set_path(ApiHalSubGhzPath433);
|
|
|
|
|
- } else {
|
|
|
|
|
- api_hal_subghz_set_path(ApiHalSubGhzPath868);
|
|
|
|
|
- }
|
|
|
|
|
|
|
|
|
|
hal_gpio_init(&gpio_cc1101_g0, GpioModeAnalog, GpioPullNo, GpioSpeedLow);
|
|
hal_gpio_init(&gpio_cc1101_g0, GpioModeAnalog, GpioPullNo, GpioSpeedLow);
|
|
|
|
|
|
|
@@ -103,7 +107,7 @@ void subghz_cli_command_rx_carrier(Cli* cli, string_t args, void* context) {
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void subghz_cli_command_tx_pt(Cli* cli, string_t args, void* context) {
|
|
void subghz_cli_command_tx_pt(Cli* cli, string_t args, void* context) {
|
|
|
- uint32_t frequency;
|
|
|
|
|
|
|
+ uint32_t frequency = 0;
|
|
|
uint32_t pattern;
|
|
uint32_t pattern;
|
|
|
uint32_t count;
|
|
uint32_t count;
|
|
|
|
|
|
|
@@ -120,8 +124,9 @@ void subghz_cli_command_tx_pt(Cli* cli, string_t args, void* context) {
|
|
|
return;
|
|
return;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- if(frequency < 300000000 || frequency > 925000000) {
|
|
|
|
|
- printf("Frequency must be in 300000000...925000000 range, not %lu\r\n", frequency);
|
|
|
|
|
|
|
+ if(!subghz_check_frequency_range(frequency)) {
|
|
|
|
|
+ printf(
|
|
|
|
|
+ "Frequency must be in " CC1101_FREQUENCY_RANGE_STR " range, not %lu\r\n", frequency);
|
|
|
return;
|
|
return;
|
|
|
}
|
|
}
|
|
|
if(pattern > 1) {
|
|
if(pattern > 1) {
|
|
@@ -133,14 +138,7 @@ void subghz_cli_command_tx_pt(Cli* cli, string_t args, void* context) {
|
|
|
|
|
|
|
|
api_hal_subghz_load_preset(ApiHalSubGhzPreset2FskPacket);
|
|
api_hal_subghz_load_preset(ApiHalSubGhzPreset2FskPacket);
|
|
|
|
|
|
|
|
- frequency = api_hal_subghz_set_frequency(frequency);
|
|
|
|
|
- if(frequency < 400000000) {
|
|
|
|
|
- api_hal_subghz_set_path(ApiHalSubGhzPath315);
|
|
|
|
|
- } else if(frequency < 500000000) {
|
|
|
|
|
- api_hal_subghz_set_path(ApiHalSubGhzPath433);
|
|
|
|
|
- } else {
|
|
|
|
|
- api_hal_subghz_set_path(ApiHalSubGhzPath868);
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ frequency = api_hal_subghz_set_frequency_and_path(frequency);
|
|
|
hal_gpio_init(&gpio_cc1101_g0, GpioModeInput, GpioPullNo, GpioSpeedLow);
|
|
hal_gpio_init(&gpio_cc1101_g0, GpioModeInput, GpioPullNo, GpioSpeedLow);
|
|
|
|
|
|
|
|
uint8_t status = api_hal_subghz_get_status();
|
|
uint8_t status = api_hal_subghz_get_status();
|
|
@@ -163,7 +161,7 @@ void subghz_cli_command_tx_pt(Cli* cli, string_t args, void* context) {
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void subghz_cli_command_rx_pt(Cli* cli, string_t args, void* context) {
|
|
void subghz_cli_command_rx_pt(Cli* cli, string_t args, void* context) {
|
|
|
- uint32_t frequency;
|
|
|
|
|
|
|
+ uint32_t frequency = 0;
|
|
|
|
|
|
|
|
int ret = sscanf(string_get_cstr(args), "%lu", &frequency);
|
|
int ret = sscanf(string_get_cstr(args), "%lu", &frequency);
|
|
|
if(ret != 1) {
|
|
if(ret != 1) {
|
|
@@ -172,8 +170,9 @@ void subghz_cli_command_rx_pt(Cli* cli, string_t args, void* context) {
|
|
|
return;
|
|
return;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- if(frequency < 300000000 || frequency > 925000000) {
|
|
|
|
|
- printf("Frequency must be in 300000000...925000000 range, not %lu\r\n", frequency);
|
|
|
|
|
|
|
+ if(!subghz_check_frequency_range(frequency)) {
|
|
|
|
|
+ printf(
|
|
|
|
|
+ "Frequency must be in " CC1101_FREQUENCY_RANGE_STR " range, not %lu\r\n", frequency);
|
|
|
return;
|
|
return;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
@@ -181,14 +180,7 @@ void subghz_cli_command_rx_pt(Cli* cli, string_t args, void* context) {
|
|
|
api_hal_subghz_idle();
|
|
api_hal_subghz_idle();
|
|
|
api_hal_subghz_load_preset(ApiHalSubGhzPreset2FskPacket);
|
|
api_hal_subghz_load_preset(ApiHalSubGhzPreset2FskPacket);
|
|
|
|
|
|
|
|
- frequency = api_hal_subghz_set_frequency(frequency);
|
|
|
|
|
- if(frequency < 400000000) {
|
|
|
|
|
- api_hal_subghz_set_path(ApiHalSubGhzPath315);
|
|
|
|
|
- } else if(frequency < 500000000) {
|
|
|
|
|
- api_hal_subghz_set_path(ApiHalSubGhzPath433);
|
|
|
|
|
- } else {
|
|
|
|
|
- api_hal_subghz_set_path(ApiHalSubGhzPath868);
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ frequency = api_hal_subghz_set_frequency_and_path(frequency);
|
|
|
hal_gpio_init(&gpio_cc1101_g0, GpioModeInput, GpioPullNo, GpioSpeedLow);
|
|
hal_gpio_init(&gpio_cc1101_g0, GpioModeInput, GpioPullNo, GpioSpeedLow);
|
|
|
|
|
|
|
|
uint8_t status = api_hal_subghz_get_status();
|
|
uint8_t status = api_hal_subghz_get_status();
|
|
@@ -216,3 +208,84 @@ void subghz_cli_command_rx_pt(Cli* cli, string_t args, void* context) {
|
|
|
api_hal_subghz_set_path(ApiHalSubGhzPathIsolate);
|
|
api_hal_subghz_set_path(ApiHalSubGhzPathIsolate);
|
|
|
hal_gpio_init(&gpio_cc1101_g0, GpioModeAnalog, GpioPullNo, GpioSpeedLow);
|
|
hal_gpio_init(&gpio_cc1101_g0, GpioModeAnalog, GpioPullNo, GpioSpeedLow);
|
|
|
}
|
|
}
|
|
|
|
|
+
|
|
|
|
|
+void subghz_cli_command_tx(Cli* cli, string_t args, void* context) {
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+#include <fl_subghz/protocols/subghz_protocol.h>
|
|
|
|
|
+
|
|
|
|
|
+volatile bool subghz_cli_overrun = false;
|
|
|
|
|
+
|
|
|
|
|
+void subghz_cli_command_rx_callback(
|
|
|
|
|
+ ApiHalSubGhzCaptureLevel level,
|
|
|
|
|
+ uint32_t duration,
|
|
|
|
|
+ void* context) {
|
|
|
|
|
+ BaseType_t xHigherPriorityTaskWoken = pdFALSE;
|
|
|
|
|
+ LevelPair pair = {.level = level, .duration = duration};
|
|
|
|
|
+ if(subghz_cli_overrun) {
|
|
|
|
|
+ subghz_cli_overrun = false;
|
|
|
|
|
+ pair.level = ApiHalSubGhzCaptureLevelOverrun;
|
|
|
|
|
+ }
|
|
|
|
|
+ size_t ret =
|
|
|
|
|
+ xStreamBufferSendFromISR(context, &pair, sizeof(LevelPair), &xHigherPriorityTaskWoken);
|
|
|
|
|
+ if(sizeof(LevelPair) != ret) subghz_cli_overrun = true;
|
|
|
|
|
+ portYIELD_FROM_ISR(xHigherPriorityTaskWoken);
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+void subghz_cli_command_rx(Cli* cli, string_t args, void* context) {
|
|
|
|
|
+ uint32_t frequency = 433920000;
|
|
|
|
|
+ if(string_size(args)) {
|
|
|
|
|
+ int ret = sscanf(string_get_cstr(args), "%lu", &frequency);
|
|
|
|
|
+ if(ret != 1) {
|
|
|
|
|
+ printf("sscanf returned %d, frequency: %lu\r\n", ret, frequency);
|
|
|
|
|
+ cli_print_usage("subghz_rx", "<Frequency in HZ>", string_get_cstr(args));
|
|
|
|
|
+ return;
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ if(!subghz_check_frequency_range(frequency)) {
|
|
|
|
|
+ printf(
|
|
|
|
|
+ "Frequency must be in " CC1101_FREQUENCY_RANGE_STR " range, not %lu\r\n",
|
|
|
|
|
+ frequency);
|
|
|
|
|
+ return;
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ api_hal_subghz_reset();
|
|
|
|
|
+ api_hal_subghz_idle();
|
|
|
|
|
+ api_hal_subghz_load_preset(ApiHalSubGhzPresetMP);
|
|
|
|
|
+
|
|
|
|
|
+ SubGhzProtocol* protocol = subghz_protocol_alloc();
|
|
|
|
|
+ subghz_protocol_load_keeloq_file(protocol, "/assets/subghz/keeloq_mfcodes");
|
|
|
|
|
+ subghz_protocol_enable_dump(protocol, NULL, NULL);
|
|
|
|
|
+
|
|
|
|
|
+ frequency = api_hal_subghz_set_frequency_and_path(frequency);
|
|
|
|
|
+ hal_gpio_init(&gpio_cc1101_g0, GpioModeInput, GpioPullNo, GpioSpeedLow);
|
|
|
|
|
+
|
|
|
|
|
+ StreamBufferHandle_t rx_stream =
|
|
|
|
|
+ xStreamBufferCreate(sizeof(LevelPair) * 1024, sizeof(LevelPair));
|
|
|
|
|
+
|
|
|
|
|
+ api_hal_subghz_set_capture_callback(subghz_cli_command_rx_callback, rx_stream);
|
|
|
|
|
+ api_hal_subghz_enable_capture();
|
|
|
|
|
+
|
|
|
|
|
+ api_hal_subghz_flush_rx();
|
|
|
|
|
+ api_hal_subghz_rx();
|
|
|
|
|
+
|
|
|
|
|
+ printf("Listening at %lu. Press CTRL+C to stop\r\n", frequency);
|
|
|
|
|
+ LevelPair pair;
|
|
|
|
|
+ while(!cli_cmd_interrupt_received(cli)) {
|
|
|
|
|
+ int ret = xStreamBufferReceive(rx_stream, &pair, sizeof(LevelPair), 10);
|
|
|
|
|
+ if(ret == sizeof(LevelPair)) {
|
|
|
|
|
+ if(pair.level == ApiHalSubGhzCaptureLevelOverrun) {
|
|
|
|
|
+ printf(".");
|
|
|
|
|
+ subghz_protocol_reset(protocol);
|
|
|
|
|
+ } else {
|
|
|
|
|
+ subghz_protocol_parse(protocol, pair);
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ subghz_protocol_free(protocol);
|
|
|
|
|
+ vStreamBufferDelete(rx_stream);
|
|
|
|
|
+ api_hal_subghz_disable_capture();
|
|
|
|
|
+ api_hal_subghz_init();
|
|
|
|
|
+}
|