#include "subghz_file_encoder_worker.h" #include #include "file-worker.h" #define SUBGHZ_FILE_ENCODER_LOAD 512 struct SubGhzFileEncoderWorker { FuriThread* thread; StreamBufferHandle_t stream; FileWorker* file_worker; volatile bool worker_running; bool level; int16_t duration; string_t str_data; string_t file_path; }; void subghz_file_encoder_worker_add_livel_duration( SubGhzFileEncoderWorker* instance, int16_t duration) { bool res = true; if(duration < 0 && !instance->level) { instance->duration += duration; res = false; } else if(duration > 0 && instance->level) { instance->duration += duration; res = false; } else if(duration == 0) { instance->duration = 0; } if(res) { instance->level = !instance->level; instance->duration += duration; xStreamBufferSend(instance->stream, &instance->duration, sizeof(int16_t), 10); instance->duration = 0; } } bool subghz_file_encoder_worker_data_parse( SubGhzFileEncoderWorker* instance, const char* strStart, size_t len) { char* str1; size_t ind_start = (size_t)strStart; //store the start address of the beginning of the line bool res = false; str1 = strstr( strStart, "RAW_Data: "); //looking for the beginning of the desired title in the line if(str1 != NULL) { str1 = strchr( str1, ' '); //if found, shift the pointer by 1 element per line "RAW_Data: -1, 2, -2..." subghz_file_encoder_worker_add_livel_duration(instance, atoi(str1)); while( strchr(str1, ',') != NULL && ((size_t)str1 < (len + ind_start))) { //check that there is still an element in the line and that it has not gone beyond the line str1 = strchr(str1, ','); str1 += 2; //if found, shift the pointer by next element per line subghz_file_encoder_worker_add_livel_duration(instance, atoi(str1)); } res = true; } return res; } LevelDuration subghz_file_encoder_worker_get_level_duration(void* context) { furi_assert(context); SubGhzFileEncoderWorker* instance = context; int16_t duration; BaseType_t xHigherPriorityTaskWoken = pdFALSE; int ret = xStreamBufferReceiveFromISR( instance->stream, &duration, sizeof(int16_t), &xHigherPriorityTaskWoken); portYIELD_FROM_ISR(xHigherPriorityTaskWoken); if(ret == sizeof(int16_t)) { LevelDuration level_duration = {.level = LEVEL_DURATION_RESET}; if(duration < 0) { level_duration = level_duration_make(false, duration * -1); } else if(duration > 0) { level_duration = level_duration_make(true, duration); } else if(duration == 0) { level_duration = level_duration_reset(); FURI_LOG_I("SubGhzFileEncoderWorker", "Stop transmission"); } return level_duration; } else { FURI_LOG_E("SubGhzFileEncoderWorker", "Slow flash read"); return level_duration_wait(); } } /** Worker thread * * @param context * @return exit code */ static int32_t subghz_file_encoder_worker_thread(void* context) { SubGhzFileEncoderWorker* instance = context; FURI_LOG_I("SubGhzFileEncoderWorker", "Worker start"); bool res = false; do { if(!file_worker_open( instance->file_worker, string_get_cstr(instance->file_path), FSAM_READ, FSOM_OPEN_EXISTING)) { break; } //todo skips 3 lines file header if(!file_worker_read_until(instance->file_worker, instance->str_data, '\n')) { break; } if(!file_worker_read_until(instance->file_worker, instance->str_data, '\n')) { break; } if(!file_worker_read_until(instance->file_worker, instance->str_data, '\n')) { break; } res = true; FURI_LOG_I("SubGhzFileEncoderWorker", "Start transmission"); } while(0); while(res && instance->worker_running) { size_t stream_free_byte = xStreamBufferSpacesAvailable(instance->stream); if((stream_free_byte / sizeof(int16_t)) >= SUBGHZ_FILE_ENCODER_LOAD) { if(file_worker_read_until(instance->file_worker, instance->str_data, '\n')) { if(!subghz_file_encoder_worker_data_parse( instance, string_get_cstr(instance->str_data), strlen(string_get_cstr(instance->str_data)))) { //to stop DMA correctly subghz_file_encoder_worker_add_livel_duration(instance, LEVEL_DURATION_RESET); subghz_file_encoder_worker_add_livel_duration(instance, LEVEL_DURATION_RESET); break; } } else { subghz_file_encoder_worker_add_livel_duration(instance, LEVEL_DURATION_RESET); subghz_file_encoder_worker_add_livel_duration(instance, LEVEL_DURATION_RESET); break; } } } //waiting for the end of the transfer while(instance->worker_running) { osDelay(50); } file_worker_close(instance->file_worker); FURI_LOG_I("SubGhzFileEncoderWorker", "Worker stop"); return 0; } SubGhzFileEncoderWorker* subghz_file_encoder_worker_alloc() { SubGhzFileEncoderWorker* instance = furi_alloc(sizeof(SubGhzFileEncoderWorker)); instance->thread = furi_thread_alloc(); furi_thread_set_name(instance->thread, "subghz_file_encoder_worker"); furi_thread_set_stack_size(instance->thread, 2048); furi_thread_set_context(instance->thread, instance); furi_thread_set_callback(instance->thread, subghz_file_encoder_worker_thread); instance->stream = xStreamBufferCreate(sizeof(int16_t) * 4096, sizeof(int16_t)); instance->file_worker = file_worker_alloc(false); string_init(instance->str_data); string_init(instance->file_path); instance->level = false; return instance; } void subghz_file_encoder_worker_free(SubGhzFileEncoderWorker* instance) { furi_assert(instance); vStreamBufferDelete(instance->stream); furi_thread_free(instance->thread); string_clear(instance->str_data); string_clear(instance->file_path); file_worker_free(instance->file_worker); free(instance); } bool subghz_file_encoder_worker_start(SubGhzFileEncoderWorker* instance, const char* file_path) { furi_assert(instance); furi_assert(!instance->worker_running); xStreamBufferReset(instance->stream); string_set(instance->file_path, file_path); instance->worker_running = true; bool res = furi_thread_start(instance->thread); return res; } void subghz_file_encoder_worker_stop(SubGhzFileEncoderWorker* instance) { furi_assert(instance); furi_assert(instance->worker_running); instance->worker_running = false; furi_thread_join(instance->thread); } bool subghz_file_encoder_worker_is_running(SubGhzFileEncoderWorker* instance) { furi_assert(instance); return instance->worker_running; }