furi_hal_subghz.c 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769
  1. #include <furi_hal_subghz.h>
  2. #include <furi_hal_subghz_configs.h>
  3. #include <furi_hal_region.h>
  4. #include <furi_hal_version.h>
  5. #include <furi_hal_rtc.h>
  6. #include <furi_hal_spi.h>
  7. #include <furi_hal_interrupt.h>
  8. #include <furi_hal_resources.h>
  9. #include <stm32wbxx_ll_dma.h>
  10. #include <furi.h>
  11. #include <cc1101.h>
  12. #include <stdio.h>
  13. #define TAG "FuriHalSubGhz"
  14. static uint32_t furi_hal_subghz_debug_gpio_buff[2];
  15. /* DMA Channels definition */
  16. #define SUBGHZ_DMA DMA2
  17. #define SUBGHZ_DMA_CH1_CHANNEL LL_DMA_CHANNEL_1
  18. #define SUBGHZ_DMA_CH2_CHANNEL LL_DMA_CHANNEL_2
  19. #define SUBGHZ_DMA_CH1_IRQ FuriHalInterruptIdDma2Ch1
  20. #define SUBGHZ_DMA_CH1_DEF SUBGHZ_DMA, SUBGHZ_DMA_CH1_CHANNEL
  21. #define SUBGHZ_DMA_CH2_DEF SUBGHZ_DMA, SUBGHZ_DMA_CH2_CHANNEL
  22. typedef struct {
  23. volatile SubGhzState state;
  24. volatile SubGhzRegulation regulation;
  25. volatile FuriHalSubGhzPreset preset;
  26. const GpioPin* async_mirror_pin;
  27. } FuriHalSubGhz;
  28. volatile FuriHalSubGhz furi_hal_subghz = {
  29. .state = SubGhzStateInit,
  30. .regulation = SubGhzRegulationTxRx,
  31. .preset = FuriHalSubGhzPresetIDLE,
  32. .async_mirror_pin = NULL,
  33. };
  34. void furi_hal_subghz_set_async_mirror_pin(const GpioPin* pin) {
  35. furi_hal_subghz.async_mirror_pin = pin;
  36. }
  37. void furi_hal_subghz_init() {
  38. furi_assert(furi_hal_subghz.state == SubGhzStateInit);
  39. furi_hal_subghz.state = SubGhzStateIdle;
  40. furi_hal_subghz.preset = FuriHalSubGhzPresetIDLE;
  41. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  42. #ifdef FURI_HAL_SUBGHZ_TX_GPIO
  43. furi_hal_gpio_init(&FURI_HAL_SUBGHZ_TX_GPIO, GpioModeOutputPushPull, GpioPullNo, GpioSpeedLow);
  44. #endif
  45. // Reset
  46. furi_hal_gpio_init(&gpio_cc1101_g0, GpioModeAnalog, GpioPullNo, GpioSpeedLow);
  47. cc1101_reset(&furi_hal_spi_bus_handle_subghz);
  48. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, CC1101_IOCFG0, CC1101IocfgHighImpedance);
  49. // Prepare GD0 for power on self test
  50. furi_hal_gpio_init(&gpio_cc1101_g0, GpioModeInput, GpioPullNo, GpioSpeedLow);
  51. // GD0 low
  52. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, CC1101_IOCFG0, CC1101IocfgHW);
  53. while(furi_hal_gpio_read(&gpio_cc1101_g0) != false)
  54. ;
  55. // GD0 high
  56. cc1101_write_reg(
  57. &furi_hal_spi_bus_handle_subghz, CC1101_IOCFG0, CC1101IocfgHW | CC1101_IOCFG_INV);
  58. while(furi_hal_gpio_read(&gpio_cc1101_g0) != true)
  59. ;
  60. // Reset GD0 to floating state
  61. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, CC1101_IOCFG0, CC1101IocfgHighImpedance);
  62. furi_hal_gpio_init(&gpio_cc1101_g0, GpioModeAnalog, GpioPullNo, GpioSpeedLow);
  63. // RF switches
  64. furi_hal_gpio_init(&gpio_rf_sw_0, GpioModeOutputPushPull, GpioPullNo, GpioSpeedLow);
  65. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, CC1101_IOCFG2, CC1101IocfgHW);
  66. // Go to sleep
  67. cc1101_shutdown(&furi_hal_spi_bus_handle_subghz);
  68. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  69. FURI_LOG_I(TAG, "Init OK");
  70. }
  71. void furi_hal_subghz_sleep() {
  72. furi_assert(furi_hal_subghz.state == SubGhzStateIdle);
  73. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  74. cc1101_switch_to_idle(&furi_hal_spi_bus_handle_subghz);
  75. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, CC1101_IOCFG0, CC1101IocfgHighImpedance);
  76. furi_hal_gpio_init(&gpio_cc1101_g0, GpioModeAnalog, GpioPullNo, GpioSpeedLow);
  77. cc1101_shutdown(&furi_hal_spi_bus_handle_subghz);
  78. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  79. furi_hal_subghz.preset = FuriHalSubGhzPresetIDLE;
  80. }
  81. void furi_hal_subghz_dump_state() {
  82. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  83. printf(
  84. "[furi_hal_subghz] cc1101 chip %d, version %d\r\n",
  85. cc1101_get_partnumber(&furi_hal_spi_bus_handle_subghz),
  86. cc1101_get_version(&furi_hal_spi_bus_handle_subghz));
  87. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  88. }
  89. void furi_hal_subghz_load_preset(FuriHalSubGhzPreset preset) {
  90. if(preset == FuriHalSubGhzPresetOok650Async) {
  91. furi_hal_subghz_load_registers((uint8_t*)furi_hal_subghz_preset_ook_650khz_async_regs);
  92. furi_hal_subghz_load_patable(furi_hal_subghz_preset_ook_async_patable);
  93. } else if(preset == FuriHalSubGhzPresetOok270Async) {
  94. furi_hal_subghz_load_registers((uint8_t*)furi_hal_subghz_preset_ook_270khz_async_regs);
  95. furi_hal_subghz_load_patable(furi_hal_subghz_preset_ook_async_patable);
  96. } else if(preset == FuriHalSubGhzPreset2FSKDev238Async) {
  97. furi_hal_subghz_load_registers(
  98. (uint8_t*)furi_hal_subghz_preset_2fsk_dev2_38khz_async_regs);
  99. furi_hal_subghz_load_patable(furi_hal_subghz_preset_2fsk_async_patable);
  100. } else if(preset == FuriHalSubGhzPreset2FSKDev476Async) {
  101. furi_hal_subghz_load_registers(
  102. (uint8_t*)furi_hal_subghz_preset_2fsk_dev47_6khz_async_regs);
  103. furi_hal_subghz_load_patable(furi_hal_subghz_preset_2fsk_async_patable);
  104. } else if(preset == FuriHalSubGhzPresetMSK99_97KbAsync) {
  105. furi_hal_subghz_load_registers((uint8_t*)furi_hal_subghz_preset_msk_99_97kb_async_regs);
  106. furi_hal_subghz_load_patable(furi_hal_subghz_preset_msk_async_patable);
  107. } else if(preset == FuriHalSubGhzPresetGFSK9_99KbAsync) {
  108. furi_hal_subghz_load_registers((uint8_t*)furi_hal_subghz_preset_gfsk_9_99kb_async_regs);
  109. furi_hal_subghz_load_patable(furi_hal_subghz_preset_gfsk_async_patable);
  110. } else {
  111. furi_crash("SubGhz: Missing config.");
  112. }
  113. furi_hal_subghz.preset = preset;
  114. }
  115. void furi_hal_subghz_load_custom_preset(uint8_t* preset_data) {
  116. //load config
  117. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  118. cc1101_reset(&furi_hal_spi_bus_handle_subghz);
  119. uint32_t i = 0;
  120. uint8_t pa[8] = {0};
  121. while(preset_data[i]) {
  122. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, preset_data[i], preset_data[i + 1]);
  123. i += 2;
  124. }
  125. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  126. //load pa table
  127. memcpy(&pa[0], &preset_data[i + 2], 8);
  128. furi_hal_subghz_load_patable(pa);
  129. furi_hal_subghz.preset = FuriHalSubGhzPresetCustom;
  130. //show debug
  131. if(furi_hal_rtc_is_flag_set(FuriHalRtcFlagDebug)) {
  132. i = 0;
  133. FURI_LOG_D(TAG, "Loading custom preset");
  134. while(preset_data[i]) {
  135. FURI_LOG_D(TAG, "Reg[%lu]: %02X=%02X", i, preset_data[i], preset_data[i + 1]);
  136. i += 2;
  137. }
  138. for(uint8_t y = i; y < i + 10; y++) {
  139. FURI_LOG_D(TAG, "PA[%u]: %02X", y, preset_data[y]);
  140. }
  141. }
  142. }
  143. void furi_hal_subghz_load_registers(uint8_t* data) {
  144. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  145. cc1101_reset(&furi_hal_spi_bus_handle_subghz);
  146. uint32_t i = 0;
  147. while(data[i]) {
  148. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, data[i], data[i + 1]);
  149. i += 2;
  150. }
  151. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  152. }
  153. void furi_hal_subghz_load_patable(const uint8_t data[8]) {
  154. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  155. cc1101_set_pa_table(&furi_hal_spi_bus_handle_subghz, data);
  156. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  157. }
  158. void furi_hal_subghz_write_packet(const uint8_t* data, uint8_t size) {
  159. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  160. cc1101_flush_tx(&furi_hal_spi_bus_handle_subghz);
  161. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, CC1101_FIFO, size);
  162. cc1101_write_fifo(&furi_hal_spi_bus_handle_subghz, data, size);
  163. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  164. }
  165. void furi_hal_subghz_flush_rx() {
  166. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  167. cc1101_flush_rx(&furi_hal_spi_bus_handle_subghz);
  168. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  169. }
  170. void furi_hal_subghz_flush_tx() {
  171. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  172. cc1101_flush_tx(&furi_hal_spi_bus_handle_subghz);
  173. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  174. }
  175. bool furi_hal_subghz_rx_pipe_not_empty() {
  176. CC1101RxBytes status[1];
  177. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  178. cc1101_read_reg(
  179. &furi_hal_spi_bus_handle_subghz, (CC1101_STATUS_RXBYTES) | CC1101_BURST, (uint8_t*)status);
  180. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  181. // TODO: you can add a buffer overflow flag if needed
  182. if(status->NUM_RXBYTES > 0) {
  183. return true;
  184. } else {
  185. return false;
  186. }
  187. }
  188. bool furi_hal_subghz_is_rx_data_crc_valid() {
  189. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  190. uint8_t data[1];
  191. cc1101_read_reg(&furi_hal_spi_bus_handle_subghz, CC1101_STATUS_LQI | CC1101_BURST, data);
  192. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  193. if(((data[0] >> 7) & 0x01)) {
  194. return true;
  195. } else {
  196. return false;
  197. }
  198. }
  199. void furi_hal_subghz_read_packet(uint8_t* data, uint8_t* size) {
  200. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  201. cc1101_read_fifo(&furi_hal_spi_bus_handle_subghz, data, size);
  202. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  203. }
  204. void furi_hal_subghz_shutdown() {
  205. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  206. // Reset and shutdown
  207. cc1101_shutdown(&furi_hal_spi_bus_handle_subghz);
  208. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  209. }
  210. void furi_hal_subghz_reset() {
  211. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  212. furi_hal_gpio_init(&gpio_cc1101_g0, GpioModeAnalog, GpioPullNo, GpioSpeedLow);
  213. cc1101_switch_to_idle(&furi_hal_spi_bus_handle_subghz);
  214. cc1101_reset(&furi_hal_spi_bus_handle_subghz);
  215. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, CC1101_IOCFG0, CC1101IocfgHighImpedance);
  216. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  217. }
  218. void furi_hal_subghz_idle() {
  219. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  220. cc1101_switch_to_idle(&furi_hal_spi_bus_handle_subghz);
  221. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  222. }
  223. void furi_hal_subghz_rx() {
  224. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  225. cc1101_switch_to_rx(&furi_hal_spi_bus_handle_subghz);
  226. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  227. }
  228. bool furi_hal_subghz_tx() {
  229. if(furi_hal_subghz.regulation != SubGhzRegulationTxRx) return false;
  230. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  231. cc1101_switch_to_tx(&furi_hal_spi_bus_handle_subghz);
  232. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  233. return true;
  234. }
  235. float furi_hal_subghz_get_rssi() {
  236. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  237. int32_t rssi_dec = cc1101_get_rssi(&furi_hal_spi_bus_handle_subghz);
  238. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  239. float rssi = rssi_dec;
  240. if(rssi_dec >= 128) {
  241. rssi = ((rssi - 256.0f) / 2.0f) - 74.0f;
  242. } else {
  243. rssi = (rssi / 2.0f) - 74.0f;
  244. }
  245. return rssi;
  246. }
  247. uint8_t furi_hal_subghz_get_lqi() {
  248. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  249. uint8_t data[1];
  250. cc1101_read_reg(&furi_hal_spi_bus_handle_subghz, CC1101_STATUS_LQI | CC1101_BURST, data);
  251. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  252. return data[0] & 0x7F;
  253. }
  254. bool furi_hal_subghz_is_frequency_valid(uint32_t value) {
  255. if(!(value >= 299999755 && value <= 348000335) &&
  256. !(value >= 386999938 && value <= 464000000) &&
  257. !(value >= 778999847 && value <= 928000000)) {
  258. return false;
  259. }
  260. return true;
  261. }
  262. uint32_t furi_hal_subghz_set_frequency_and_path(uint32_t value) {
  263. value = furi_hal_subghz_set_frequency(value);
  264. if(value >= 299999755 && value <= 348000335) {
  265. furi_hal_subghz_set_path(FuriHalSubGhzPath315);
  266. } else if(value >= 386999938 && value <= 464000000) {
  267. furi_hal_subghz_set_path(FuriHalSubGhzPath433);
  268. } else if(value >= 778999847 && value <= 928000000) {
  269. furi_hal_subghz_set_path(FuriHalSubGhzPath868);
  270. } else {
  271. furi_crash("SubGhz: Incorrect frequency during set.");
  272. }
  273. return value;
  274. }
  275. uint32_t furi_hal_subghz_set_frequency(uint32_t value) {
  276. if(furi_hal_region_is_frequency_allowed(value)) {
  277. furi_hal_subghz.regulation = SubGhzRegulationTxRx;
  278. } else {
  279. furi_hal_subghz.regulation = SubGhzRegulationOnlyRx;
  280. }
  281. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  282. uint32_t real_frequency = cc1101_set_frequency(&furi_hal_spi_bus_handle_subghz, value);
  283. cc1101_calibrate(&furi_hal_spi_bus_handle_subghz);
  284. while(true) {
  285. CC1101Status status = cc1101_get_status(&furi_hal_spi_bus_handle_subghz);
  286. if(status.STATE == CC1101StateIDLE) break;
  287. }
  288. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  289. return real_frequency;
  290. }
  291. void furi_hal_subghz_set_path(FuriHalSubGhzPath path) {
  292. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  293. if(path == FuriHalSubGhzPath433) {
  294. furi_hal_gpio_write(&gpio_rf_sw_0, 0);
  295. cc1101_write_reg(
  296. &furi_hal_spi_bus_handle_subghz, CC1101_IOCFG2, CC1101IocfgHW | CC1101_IOCFG_INV);
  297. } else if(path == FuriHalSubGhzPath315) {
  298. furi_hal_gpio_write(&gpio_rf_sw_0, 1);
  299. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, CC1101_IOCFG2, CC1101IocfgHW);
  300. } else if(path == FuriHalSubGhzPath868) {
  301. furi_hal_gpio_write(&gpio_rf_sw_0, 1);
  302. cc1101_write_reg(
  303. &furi_hal_spi_bus_handle_subghz, CC1101_IOCFG2, CC1101IocfgHW | CC1101_IOCFG_INV);
  304. } else if(path == FuriHalSubGhzPathIsolate) {
  305. furi_hal_gpio_write(&gpio_rf_sw_0, 0);
  306. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, CC1101_IOCFG2, CC1101IocfgHW);
  307. } else {
  308. furi_crash("SubGhz: Incorrect path during set.");
  309. }
  310. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  311. }
  312. static bool furi_hal_subghz_start_debug() {
  313. bool ret = false;
  314. if(furi_hal_subghz.async_mirror_pin != NULL) {
  315. furi_hal_gpio_init(
  316. furi_hal_subghz.async_mirror_pin,
  317. GpioModeOutputPushPull,
  318. GpioPullNo,
  319. GpioSpeedVeryHigh);
  320. ret = true;
  321. }
  322. return ret;
  323. }
  324. static bool furi_hal_subghz_stop_debug() {
  325. bool ret = false;
  326. if(furi_hal_subghz.async_mirror_pin != NULL) {
  327. furi_hal_gpio_init(
  328. furi_hal_subghz.async_mirror_pin, GpioModeAnalog, GpioPullNo, GpioSpeedLow);
  329. ret = true;
  330. }
  331. return ret;
  332. }
  333. volatile uint32_t furi_hal_subghz_capture_delta_duration = 0;
  334. volatile FuriHalSubGhzCaptureCallback furi_hal_subghz_capture_callback = NULL;
  335. volatile void* furi_hal_subghz_capture_callback_context = NULL;
  336. static void furi_hal_subghz_capture_ISR() {
  337. // Channel 1
  338. if(LL_TIM_IsActiveFlag_CC1(TIM2)) {
  339. LL_TIM_ClearFlag_CC1(TIM2);
  340. furi_hal_subghz_capture_delta_duration = LL_TIM_IC_GetCaptureCH1(TIM2);
  341. if(furi_hal_subghz_capture_callback) {
  342. if(furi_hal_subghz.async_mirror_pin != NULL)
  343. furi_hal_gpio_write(furi_hal_subghz.async_mirror_pin, false);
  344. furi_hal_subghz_capture_callback(
  345. true,
  346. furi_hal_subghz_capture_delta_duration,
  347. (void*)furi_hal_subghz_capture_callback_context);
  348. }
  349. }
  350. // Channel 2
  351. if(LL_TIM_IsActiveFlag_CC2(TIM2)) {
  352. LL_TIM_ClearFlag_CC2(TIM2);
  353. if(furi_hal_subghz_capture_callback) {
  354. if(furi_hal_subghz.async_mirror_pin != NULL)
  355. furi_hal_gpio_write(furi_hal_subghz.async_mirror_pin, true);
  356. furi_hal_subghz_capture_callback(
  357. false,
  358. LL_TIM_IC_GetCaptureCH2(TIM2) - furi_hal_subghz_capture_delta_duration,
  359. (void*)furi_hal_subghz_capture_callback_context);
  360. }
  361. }
  362. }
  363. void furi_hal_subghz_start_async_rx(FuriHalSubGhzCaptureCallback callback, void* context) {
  364. furi_assert(furi_hal_subghz.state == SubGhzStateIdle);
  365. furi_hal_subghz.state = SubGhzStateAsyncRx;
  366. furi_hal_subghz_capture_callback = callback;
  367. furi_hal_subghz_capture_callback_context = context;
  368. furi_hal_gpio_init_ex(
  369. &gpio_cc1101_g0, GpioModeAltFunctionPushPull, GpioPullNo, GpioSpeedLow, GpioAltFn1TIM2);
  370. // Timer: base
  371. LL_TIM_InitTypeDef TIM_InitStruct = {0};
  372. TIM_InitStruct.Prescaler = 64 - 1;
  373. TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP;
  374. TIM_InitStruct.Autoreload = 0x7FFFFFFE;
  375. TIM_InitStruct.ClockDivision = LL_TIM_CLOCKDIVISION_DIV4;
  376. LL_TIM_Init(TIM2, &TIM_InitStruct);
  377. // Timer: advanced
  378. LL_TIM_SetClockSource(TIM2, LL_TIM_CLOCKSOURCE_INTERNAL);
  379. LL_TIM_DisableARRPreload(TIM2);
  380. LL_TIM_SetTriggerInput(TIM2, LL_TIM_TS_TI2FP2);
  381. LL_TIM_SetSlaveMode(TIM2, LL_TIM_SLAVEMODE_RESET);
  382. LL_TIM_SetTriggerOutput(TIM2, LL_TIM_TRGO_RESET);
  383. LL_TIM_EnableMasterSlaveMode(TIM2);
  384. LL_TIM_DisableDMAReq_TRIG(TIM2);
  385. LL_TIM_DisableIT_TRIG(TIM2);
  386. // Timer: channel 1 indirect
  387. LL_TIM_IC_SetActiveInput(TIM2, LL_TIM_CHANNEL_CH1, LL_TIM_ACTIVEINPUT_INDIRECTTI);
  388. LL_TIM_IC_SetPrescaler(TIM2, LL_TIM_CHANNEL_CH1, LL_TIM_ICPSC_DIV1);
  389. LL_TIM_IC_SetPolarity(TIM2, LL_TIM_CHANNEL_CH1, LL_TIM_IC_POLARITY_FALLING);
  390. LL_TIM_IC_SetFilter(TIM2, LL_TIM_CHANNEL_CH1, LL_TIM_IC_FILTER_FDIV1);
  391. // Timer: channel 2 direct
  392. LL_TIM_IC_SetActiveInput(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ACTIVEINPUT_DIRECTTI);
  393. LL_TIM_IC_SetPrescaler(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ICPSC_DIV1);
  394. LL_TIM_IC_SetPolarity(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_POLARITY_RISING);
  395. LL_TIM_IC_SetFilter(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_FILTER_FDIV32_N8);
  396. // ISR setup
  397. furi_hal_interrupt_set_isr(FuriHalInterruptIdTIM2, furi_hal_subghz_capture_ISR, NULL);
  398. // Interrupts and channels
  399. LL_TIM_EnableIT_CC1(TIM2);
  400. LL_TIM_EnableIT_CC2(TIM2);
  401. LL_TIM_CC_EnableChannel(TIM2, LL_TIM_CHANNEL_CH1);
  402. LL_TIM_CC_EnableChannel(TIM2, LL_TIM_CHANNEL_CH2);
  403. // Start timer
  404. LL_TIM_SetCounter(TIM2, 0);
  405. LL_TIM_EnableCounter(TIM2);
  406. // Start debug
  407. furi_hal_subghz_start_debug();
  408. // Switch to RX
  409. furi_hal_subghz_rx();
  410. }
  411. void furi_hal_subghz_stop_async_rx() {
  412. furi_assert(furi_hal_subghz.state == SubGhzStateAsyncRx);
  413. furi_hal_subghz.state = SubGhzStateIdle;
  414. // Shutdown radio
  415. furi_hal_subghz_idle();
  416. FURI_CRITICAL_ENTER();
  417. LL_TIM_DeInit(TIM2);
  418. // Stop debug
  419. furi_hal_subghz_stop_debug();
  420. FURI_CRITICAL_EXIT();
  421. furi_hal_interrupt_set_isr(FuriHalInterruptIdTIM2, NULL, NULL);
  422. furi_hal_gpio_init(&gpio_cc1101_g0, GpioModeAnalog, GpioPullNo, GpioSpeedLow);
  423. }
  424. typedef struct {
  425. uint32_t* buffer;
  426. LevelDuration carry_ld;
  427. FuriHalSubGhzAsyncTxCallback callback;
  428. void* callback_context;
  429. uint64_t duty_high;
  430. uint64_t duty_low;
  431. } FuriHalSubGhzAsyncTx;
  432. static FuriHalSubGhzAsyncTx furi_hal_subghz_async_tx = {0};
  433. static void furi_hal_subghz_async_tx_refill(uint32_t* buffer, size_t samples) {
  434. furi_assert(furi_hal_subghz.state == SubGhzStateAsyncTx);
  435. while(samples > 0) {
  436. bool is_odd = samples % 2;
  437. LevelDuration ld;
  438. if(level_duration_is_reset(furi_hal_subghz_async_tx.carry_ld)) {
  439. ld = furi_hal_subghz_async_tx.callback(furi_hal_subghz_async_tx.callback_context);
  440. } else {
  441. ld = furi_hal_subghz_async_tx.carry_ld;
  442. furi_hal_subghz_async_tx.carry_ld = level_duration_reset();
  443. }
  444. if(level_duration_is_wait(ld)) {
  445. *buffer = API_HAL_SUBGHZ_ASYNC_TX_GUARD_TIME;
  446. buffer++;
  447. samples--;
  448. } else if(level_duration_is_reset(ld)) {
  449. *buffer = 0;
  450. buffer++;
  451. samples--;
  452. LL_DMA_DisableIT_HT(SUBGHZ_DMA_CH1_DEF);
  453. LL_DMA_DisableIT_TC(SUBGHZ_DMA_CH1_DEF);
  454. LL_TIM_EnableIT_UPDATE(TIM2);
  455. break;
  456. } else {
  457. bool level = level_duration_get_level(ld);
  458. // Inject guard time if level is incorrect
  459. if(is_odd != level) {
  460. *buffer = API_HAL_SUBGHZ_ASYNC_TX_GUARD_TIME;
  461. buffer++;
  462. samples--;
  463. if(is_odd) {
  464. furi_hal_subghz_async_tx.duty_high += API_HAL_SUBGHZ_ASYNC_TX_GUARD_TIME;
  465. } else {
  466. furi_hal_subghz_async_tx.duty_low += API_HAL_SUBGHZ_ASYNC_TX_GUARD_TIME;
  467. }
  468. // Special case: prevent buffer overflow if sample is last
  469. if(samples == 0) {
  470. furi_hal_subghz_async_tx.carry_ld = ld;
  471. break;
  472. }
  473. }
  474. uint32_t duration = level_duration_get_duration(ld);
  475. furi_assert(duration > 0);
  476. *buffer = duration;
  477. buffer++;
  478. samples--;
  479. if(is_odd) {
  480. furi_hal_subghz_async_tx.duty_high += duration;
  481. } else {
  482. furi_hal_subghz_async_tx.duty_low += duration;
  483. }
  484. }
  485. }
  486. }
  487. static void furi_hal_subghz_async_tx_dma_isr() {
  488. furi_assert(furi_hal_subghz.state == SubGhzStateAsyncTx);
  489. #if SUBGHZ_DMA_CH1_CHANNEL == LL_DMA_CHANNEL_1
  490. if(LL_DMA_IsActiveFlag_HT1(SUBGHZ_DMA)) {
  491. LL_DMA_ClearFlag_HT1(SUBGHZ_DMA);
  492. furi_hal_subghz_async_tx_refill(
  493. furi_hal_subghz_async_tx.buffer, API_HAL_SUBGHZ_ASYNC_TX_BUFFER_HALF);
  494. }
  495. if(LL_DMA_IsActiveFlag_TC1(SUBGHZ_DMA)) {
  496. LL_DMA_ClearFlag_TC1(SUBGHZ_DMA);
  497. furi_hal_subghz_async_tx_refill(
  498. furi_hal_subghz_async_tx.buffer + API_HAL_SUBGHZ_ASYNC_TX_BUFFER_HALF,
  499. API_HAL_SUBGHZ_ASYNC_TX_BUFFER_HALF);
  500. }
  501. #else
  502. #error Update this code. Would you kindly?
  503. #endif
  504. }
  505. static void furi_hal_subghz_async_tx_timer_isr() {
  506. if(LL_TIM_IsActiveFlag_UPDATE(TIM2)) {
  507. LL_TIM_ClearFlag_UPDATE(TIM2);
  508. if(LL_TIM_GetAutoReload(TIM2) == 0) {
  509. if(furi_hal_subghz.state == SubGhzStateAsyncTx) {
  510. furi_hal_subghz.state = SubGhzStateAsyncTxLast;
  511. LL_DMA_DisableChannel(SUBGHZ_DMA_CH1_DEF);
  512. } else if(furi_hal_subghz.state == SubGhzStateAsyncTxLast) {
  513. furi_hal_subghz.state = SubGhzStateAsyncTxEnd;
  514. //forcibly pulls the pin to the ground so that there is no carrier
  515. furi_hal_gpio_init(&gpio_cc1101_g0, GpioModeInput, GpioPullDown, GpioSpeedLow);
  516. LL_TIM_DisableCounter(TIM2);
  517. } else {
  518. furi_crash(NULL);
  519. }
  520. }
  521. }
  522. }
  523. bool furi_hal_subghz_start_async_tx(FuriHalSubGhzAsyncTxCallback callback, void* context) {
  524. furi_assert(furi_hal_subghz.state == SubGhzStateIdle);
  525. furi_assert(callback);
  526. //If transmission is prohibited by regional settings
  527. if(furi_hal_subghz.regulation != SubGhzRegulationTxRx) return false;
  528. furi_hal_subghz_async_tx.callback = callback;
  529. furi_hal_subghz_async_tx.callback_context = context;
  530. furi_hal_subghz.state = SubGhzStateAsyncTx;
  531. furi_hal_subghz_async_tx.duty_low = 0;
  532. furi_hal_subghz_async_tx.duty_high = 0;
  533. furi_hal_subghz_async_tx.buffer =
  534. malloc(API_HAL_SUBGHZ_ASYNC_TX_BUFFER_FULL * sizeof(uint32_t));
  535. // Connect CC1101_GD0 to TIM2 as output
  536. furi_hal_gpio_init_ex(
  537. &gpio_cc1101_g0, GpioModeAltFunctionPushPull, GpioPullDown, GpioSpeedLow, GpioAltFn1TIM2);
  538. // Configure DMA
  539. LL_DMA_InitTypeDef dma_config = {0};
  540. dma_config.PeriphOrM2MSrcAddress = (uint32_t) & (TIM2->ARR);
  541. dma_config.MemoryOrM2MDstAddress = (uint32_t)furi_hal_subghz_async_tx.buffer;
  542. dma_config.Direction = LL_DMA_DIRECTION_MEMORY_TO_PERIPH;
  543. dma_config.Mode = LL_DMA_MODE_CIRCULAR;
  544. dma_config.PeriphOrM2MSrcIncMode = LL_DMA_PERIPH_NOINCREMENT;
  545. dma_config.MemoryOrM2MDstIncMode = LL_DMA_MEMORY_INCREMENT;
  546. dma_config.PeriphOrM2MSrcDataSize = LL_DMA_PDATAALIGN_WORD;
  547. dma_config.MemoryOrM2MDstDataSize = LL_DMA_MDATAALIGN_WORD;
  548. dma_config.NbData = API_HAL_SUBGHZ_ASYNC_TX_BUFFER_FULL;
  549. dma_config.PeriphRequest = LL_DMAMUX_REQ_TIM2_UP;
  550. dma_config.Priority = LL_DMA_MODE_NORMAL;
  551. LL_DMA_Init(SUBGHZ_DMA_CH1_DEF, &dma_config);
  552. furi_hal_interrupt_set_isr(SUBGHZ_DMA_CH1_IRQ, furi_hal_subghz_async_tx_dma_isr, NULL);
  553. LL_DMA_EnableIT_TC(SUBGHZ_DMA_CH1_DEF);
  554. LL_DMA_EnableIT_HT(SUBGHZ_DMA_CH1_DEF);
  555. LL_DMA_EnableChannel(SUBGHZ_DMA_CH1_DEF);
  556. // Configure TIM2
  557. LL_TIM_InitTypeDef TIM_InitStruct = {0};
  558. TIM_InitStruct.Prescaler = 64 - 1;
  559. TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP;
  560. TIM_InitStruct.Autoreload = 1000;
  561. TIM_InitStruct.ClockDivision = LL_TIM_CLOCKDIVISION_DIV1;
  562. LL_TIM_Init(TIM2, &TIM_InitStruct);
  563. LL_TIM_SetClockSource(TIM2, LL_TIM_CLOCKSOURCE_INTERNAL);
  564. LL_TIM_EnableARRPreload(TIM2);
  565. // Configure TIM2 CH2
  566. LL_TIM_OC_InitTypeDef TIM_OC_InitStruct = {0};
  567. TIM_OC_InitStruct.OCMode = LL_TIM_OCMODE_TOGGLE;
  568. TIM_OC_InitStruct.OCState = LL_TIM_OCSTATE_DISABLE;
  569. TIM_OC_InitStruct.OCNState = LL_TIM_OCSTATE_DISABLE;
  570. TIM_OC_InitStruct.CompareValue = 0;
  571. TIM_OC_InitStruct.OCPolarity = LL_TIM_OCPOLARITY_LOW;
  572. LL_TIM_OC_Init(TIM2, LL_TIM_CHANNEL_CH2, &TIM_OC_InitStruct);
  573. LL_TIM_OC_DisableFast(TIM2, LL_TIM_CHANNEL_CH2);
  574. LL_TIM_DisableMasterSlaveMode(TIM2);
  575. furi_hal_interrupt_set_isr(FuriHalInterruptIdTIM2, furi_hal_subghz_async_tx_timer_isr, NULL);
  576. furi_hal_subghz_async_tx_refill(
  577. furi_hal_subghz_async_tx.buffer, API_HAL_SUBGHZ_ASYNC_TX_BUFFER_FULL);
  578. LL_TIM_EnableDMAReq_UPDATE(TIM2);
  579. LL_TIM_CC_EnableChannel(TIM2, LL_TIM_CHANNEL_CH2);
  580. // Start counter
  581. LL_TIM_GenerateEvent_UPDATE(TIM2);
  582. #ifdef FURI_HAL_SUBGHZ_TX_GPIO
  583. furi_hal_gpio_write(&FURI_HAL_SUBGHZ_TX_GPIO, true);
  584. #endif
  585. furi_hal_subghz_tx();
  586. LL_TIM_SetCounter(TIM2, 0);
  587. LL_TIM_EnableCounter(TIM2);
  588. // Start debug
  589. if(furi_hal_subghz_start_debug()) {
  590. const GpioPin* gpio = furi_hal_subghz.async_mirror_pin;
  591. furi_hal_subghz_debug_gpio_buff[0] = (uint32_t)gpio->pin << GPIO_NUMBER;
  592. furi_hal_subghz_debug_gpio_buff[1] = gpio->pin;
  593. dma_config.MemoryOrM2MDstAddress = (uint32_t)furi_hal_subghz_debug_gpio_buff;
  594. dma_config.PeriphOrM2MSrcAddress = (uint32_t) & (gpio->port->BSRR);
  595. dma_config.Direction = LL_DMA_DIRECTION_MEMORY_TO_PERIPH;
  596. dma_config.Mode = LL_DMA_MODE_CIRCULAR;
  597. dma_config.PeriphOrM2MSrcIncMode = LL_DMA_PERIPH_NOINCREMENT;
  598. dma_config.MemoryOrM2MDstIncMode = LL_DMA_MEMORY_INCREMENT;
  599. dma_config.PeriphOrM2MSrcDataSize = LL_DMA_PDATAALIGN_WORD;
  600. dma_config.MemoryOrM2MDstDataSize = LL_DMA_MDATAALIGN_WORD;
  601. dma_config.NbData = 2;
  602. dma_config.PeriphRequest = LL_DMAMUX_REQ_TIM2_UP;
  603. dma_config.Priority = LL_DMA_PRIORITY_VERYHIGH;
  604. LL_DMA_Init(SUBGHZ_DMA_CH2_DEF, &dma_config);
  605. LL_DMA_SetDataLength(SUBGHZ_DMA_CH2_DEF, 2);
  606. LL_DMA_EnableChannel(SUBGHZ_DMA_CH2_DEF);
  607. }
  608. return true;
  609. }
  610. bool furi_hal_subghz_is_async_tx_complete() {
  611. return furi_hal_subghz.state == SubGhzStateAsyncTxEnd;
  612. }
  613. void furi_hal_subghz_stop_async_tx() {
  614. furi_assert(
  615. furi_hal_subghz.state == SubGhzStateAsyncTx ||
  616. furi_hal_subghz.state == SubGhzStateAsyncTxLast ||
  617. furi_hal_subghz.state == SubGhzStateAsyncTxEnd);
  618. // Shutdown radio
  619. furi_hal_subghz_idle();
  620. #ifdef FURI_HAL_SUBGHZ_TX_GPIO
  621. furi_hal_gpio_write(&FURI_HAL_SUBGHZ_TX_GPIO, false);
  622. #endif
  623. // Deinitialize Timer
  624. FURI_CRITICAL_ENTER();
  625. LL_TIM_DeInit(TIM2);
  626. furi_hal_interrupt_set_isr(FuriHalInterruptIdTIM2, NULL, NULL);
  627. // Deinitialize DMA
  628. LL_DMA_DeInit(SUBGHZ_DMA_CH1_DEF);
  629. furi_hal_interrupt_set_isr(SUBGHZ_DMA_CH1_IRQ, NULL, NULL);
  630. // Deinitialize GPIO
  631. furi_hal_gpio_init(&gpio_cc1101_g0, GpioModeAnalog, GpioPullNo, GpioSpeedLow);
  632. // Stop debug
  633. if(furi_hal_subghz_stop_debug()) {
  634. LL_DMA_DisableChannel(SUBGHZ_DMA_CH2_DEF);
  635. }
  636. FURI_CRITICAL_EXIT();
  637. free(furi_hal_subghz_async_tx.buffer);
  638. float duty_cycle =
  639. 100.0f * (float)furi_hal_subghz_async_tx.duty_high /
  640. ((float)furi_hal_subghz_async_tx.duty_low + (float)furi_hal_subghz_async_tx.duty_high);
  641. FURI_LOG_D(
  642. TAG,
  643. "Async TX Radio stats: on %0.0fus, off %0.0fus, DutyCycle: %0.0f%%",
  644. (double)furi_hal_subghz_async_tx.duty_high,
  645. (double)furi_hal_subghz_async_tx.duty_low,
  646. (double)duty_cycle);
  647. furi_hal_subghz.state = SubGhzStateIdle;
  648. }