 b9a766d909
			
		
	
	
		b9a766d909
		
	
	
	
	
		
			
			* Added support for running applications from SD card (FAPs - Flipper Application Packages) * Added plugin_dist target for fbt to build FAPs * All apps of type FlipperAppType.EXTERNAL and FlipperAppType.PLUGIN are built as FAPs by default * Updated VSCode configuration for new fbt features - re-deploy stock configuration to use them * Added debugging support for FAPs with fbt debug & VSCode * Added public firmware API with automated versioning Co-authored-by: hedger <hedger@users.noreply.github.com> Co-authored-by: SG <who.just.the.doctor@gmail.com> Co-authored-by: あく <alleteam@gmail.com>
		
			
				
	
	
		
			222 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			222 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include <furi.h>
 | |
| #include "../minunit.h"
 | |
| #include <toolbox/protocols/protocol_dict.h>
 | |
| 
 | |
| typedef enum {
 | |
|     TestDictProtocol0,
 | |
|     TestDictProtocol1,
 | |
| 
 | |
|     TestDictProtocolMax,
 | |
| } TestDictProtocols;
 | |
| 
 | |
| /*********************** PROTOCOL 0 START ***********************/
 | |
| 
 | |
| typedef struct {
 | |
|     uint32_t data;
 | |
|     size_t encoder_counter;
 | |
| } Protocol0Data;
 | |
| 
 | |
| static const uint32_t protocol_0_decoder_result = 0xDEADBEEF;
 | |
| 
 | |
| static void* protocol_0_alloc() {
 | |
|     void* data = malloc(sizeof(Protocol0Data));
 | |
|     return data;
 | |
| }
 | |
| 
 | |
| static void protocol_0_free(Protocol0Data* data) {
 | |
|     free(data);
 | |
| }
 | |
| 
 | |
| static uint8_t* protocol_0_get_data(Protocol0Data* data) {
 | |
|     return (uint8_t*)&data->data;
 | |
| }
 | |
| 
 | |
| static void protocol_0_decoder_start(Protocol0Data* data) {
 | |
|     data->data = 0;
 | |
| }
 | |
| 
 | |
| static bool protocol_0_decoder_feed(Protocol0Data* data, bool level, uint32_t duration) {
 | |
|     if(level && duration == 666) {
 | |
|         data->data = protocol_0_decoder_result;
 | |
|         return true;
 | |
|     } else {
 | |
|         return false;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static bool protocol_0_encoder_start(Protocol0Data* data) {
 | |
|     data->encoder_counter = 0;
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static LevelDuration protocol_0_encoder_yield(Protocol0Data* data) {
 | |
|     data->encoder_counter++;
 | |
|     return level_duration_make(data->encoder_counter % 2, data->data);
 | |
| }
 | |
| 
 | |
| /*********************** PROTOCOL 1 START ***********************/
 | |
| 
 | |
| typedef struct {
 | |
|     uint64_t data;
 | |
|     size_t encoder_counter;
 | |
| } Protocol1Data;
 | |
| 
 | |
| static const uint64_t protocol_1_decoder_result = 0x1234567890ABCDEF;
 | |
| 
 | |
| static void* protocol_1_alloc() {
 | |
|     void* data = malloc(sizeof(Protocol1Data));
 | |
|     return data;
 | |
| }
 | |
| 
 | |
| static void protocol_1_free(Protocol1Data* data) {
 | |
|     free(data);
 | |
| }
 | |
| 
 | |
| static uint8_t* protocol_1_get_data(Protocol1Data* data) {
 | |
|     return (uint8_t*)&data->data;
 | |
| }
 | |
| 
 | |
| static void protocol_1_decoder_start(Protocol1Data* data) {
 | |
|     data->data = 0;
 | |
| }
 | |
| 
 | |
| static bool protocol_1_decoder_feed(Protocol1Data* data, bool level, uint32_t duration) {
 | |
|     if(level && duration == 543) {
 | |
|         data->data = 0x1234567890ABCDEF;
 | |
|         return true;
 | |
|     } else {
 | |
|         return false;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static bool protocol_1_encoder_start(Protocol1Data* data) {
 | |
|     data->encoder_counter = 0;
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static LevelDuration protocol_1_encoder_yield(Protocol1Data* data) {
 | |
|     data->encoder_counter++;
 | |
|     return level_duration_make(!(data->encoder_counter % 2), 100);
 | |
| }
 | |
| 
 | |
| /*********************** PROTOCOLS DESCRIPTION ***********************/
 | |
| static const ProtocolBase protocol_0 = {
 | |
|     .name = "Protocol 0",
 | |
|     .manufacturer = "Manufacturer 0",
 | |
|     .data_size = 4,
 | |
|     .alloc = (ProtocolAlloc)protocol_0_alloc,
 | |
|     .free = (ProtocolFree)protocol_0_free,
 | |
|     .get_data = (ProtocolGetData)protocol_0_get_data,
 | |
|     .decoder =
 | |
|         {
 | |
|             .start = (ProtocolDecoderStart)protocol_0_decoder_start,
 | |
|             .feed = (ProtocolDecoderFeed)protocol_0_decoder_feed,
 | |
|         },
 | |
|     .encoder =
 | |
|         {
 | |
|             .start = (ProtocolEncoderStart)protocol_0_encoder_start,
 | |
|             .yield = (ProtocolEncoderYield)protocol_0_encoder_yield,
 | |
|         },
 | |
| };
 | |
| 
 | |
| static const ProtocolBase protocol_1 = {
 | |
|     .name = "Protocol 1",
 | |
|     .manufacturer = "Manufacturer 1",
 | |
|     .data_size = 8,
 | |
|     .alloc = (ProtocolAlloc)protocol_1_alloc,
 | |
|     .free = (ProtocolFree)protocol_1_free,
 | |
|     .get_data = (ProtocolGetData)protocol_1_get_data,
 | |
|     .decoder =
 | |
|         {
 | |
|             .start = (ProtocolDecoderStart)protocol_1_decoder_start,
 | |
|             .feed = (ProtocolDecoderFeed)protocol_1_decoder_feed,
 | |
|         },
 | |
|     .encoder =
 | |
|         {
 | |
|             .start = (ProtocolEncoderStart)protocol_1_encoder_start,
 | |
|             .yield = (ProtocolEncoderYield)protocol_1_encoder_yield,
 | |
|         },
 | |
| };
 | |
| 
 | |
| static const ProtocolBase* test_protocols_base[] = {
 | |
|     [TestDictProtocol0] = &protocol_0,
 | |
|     [TestDictProtocol1] = &protocol_1,
 | |
| };
 | |
| 
 | |
| MU_TEST(test_protocol_dict) {
 | |
|     ProtocolDict* dict = protocol_dict_alloc(test_protocols_base, TestDictProtocolMax);
 | |
|     size_t max_data_size = protocol_dict_get_max_data_size(dict);
 | |
|     mu_assert_int_eq(8, max_data_size);
 | |
|     uint8_t* data = malloc(max_data_size);
 | |
| 
 | |
|     protocol_dict_decoders_start(dict);
 | |
|     ProtocolId protocol_id = PROTOCOL_NO;
 | |
| 
 | |
|     for(size_t i = 0; i < 100; i++) {
 | |
|         protocol_id = protocol_dict_decoders_feed(dict, i % 2, 100);
 | |
|         mu_assert_int_eq(PROTOCOL_NO, protocol_id);
 | |
|     }
 | |
| 
 | |
|     // trigger protocol 1
 | |
|     protocol_id = protocol_dict_decoders_feed(dict, true, 543);
 | |
|     mu_assert_int_eq(TestDictProtocol1, protocol_id);
 | |
| 
 | |
|     mu_assert_string_eq("Protocol 1", protocol_dict_get_name(dict, protocol_id));
 | |
|     mu_assert_string_eq("Manufacturer 1", protocol_dict_get_manufacturer(dict, protocol_id));
 | |
| 
 | |
|     size_t data_size = protocol_dict_get_data_size(dict, protocol_id);
 | |
|     mu_assert_int_eq(8, data_size);
 | |
| 
 | |
|     protocol_dict_get_data(dict, protocol_id, data, data_size);
 | |
|     mu_assert_mem_eq(&protocol_1_decoder_result, data, data_size);
 | |
| 
 | |
|     // trigger protocol 0
 | |
|     protocol_id = protocol_dict_decoders_feed(dict, true, 666);
 | |
|     mu_assert_int_eq(TestDictProtocol0, protocol_id);
 | |
| 
 | |
|     mu_assert_string_eq("Protocol 0", protocol_dict_get_name(dict, protocol_id));
 | |
|     mu_assert_string_eq("Manufacturer 0", protocol_dict_get_manufacturer(dict, protocol_id));
 | |
| 
 | |
|     data_size = protocol_dict_get_data_size(dict, protocol_id);
 | |
|     mu_assert_int_eq(4, data_size);
 | |
| 
 | |
|     protocol_dict_get_data(dict, protocol_id, data, data_size);
 | |
|     mu_assert_mem_eq(&protocol_0_decoder_result, data, data_size);
 | |
| 
 | |
|     protocol_dict_decoders_start(dict);
 | |
| 
 | |
|     protocol_id = TestDictProtocol0;
 | |
| 
 | |
|     const uint8_t protocol_0_test_data[4] = {100, 0, 0, 0};
 | |
|     protocol_dict_set_data(dict, protocol_id, protocol_0_test_data, 4);
 | |
| 
 | |
|     mu_check(protocol_dict_encoder_start(dict, protocol_id));
 | |
| 
 | |
|     LevelDuration level;
 | |
|     level = protocol_dict_encoder_yield(dict, protocol_id);
 | |
|     mu_assert_int_eq(true, level_duration_get_level(level));
 | |
|     mu_assert_int_eq(100, level_duration_get_duration(level));
 | |
|     level = protocol_dict_encoder_yield(dict, protocol_id);
 | |
|     mu_assert_int_eq(false, level_duration_get_level(level));
 | |
|     mu_assert_int_eq(100, level_duration_get_duration(level));
 | |
|     level = protocol_dict_encoder_yield(dict, protocol_id);
 | |
|     mu_assert_int_eq(true, level_duration_get_level(level));
 | |
|     mu_assert_int_eq(100, level_duration_get_duration(level));
 | |
| 
 | |
|     mu_check(protocol_dict_encoder_start(dict, protocol_id));
 | |
|     level = protocol_dict_encoder_yield(dict, protocol_id);
 | |
|     mu_assert_int_eq(true, level_duration_get_level(level));
 | |
|     mu_assert_int_eq(100, level_duration_get_duration(level));
 | |
| 
 | |
|     protocol_dict_free(dict);
 | |
|     free(data);
 | |
| }
 | |
| 
 | |
| MU_TEST_SUITE(test_protocol_dict_suite) {
 | |
|     MU_RUN_TEST(test_protocol_dict);
 | |
| }
 | |
| 
 | |
| int run_minunit_test_protocol_dict() {
 | |
|     MU_RUN_SUITE(test_protocol_dict_suite);
 | |
|     return MU_EXIT_CODE;
 | |
| } |