 9d38f28de7
			
		
	
	
		9d38f28de7
		
			
		
	
	
	
	
		
			
			* Storage-cli: the file write function is splitted into a function for working with text and function for raw data * Storage-cli: read_chunks, renamed write_raw to write_chunk * Storage-cli: fix typo * SD Hal: fixed wrong read/write block address * HAL-console: printf * Storage benchmark: more popular sizes * Toolbox: md5 * Storage-cli: better read_chunks function, md5 hash function * Notification: fixed incorrect error message when loading settings * Storage-cli: stat command * Storage-cli: stat, "/" is also storage * Scripts: add storage managment script * Scripts, storage lib: send command with known response syntax instead of SOH * Scripts: python3 from env * Storage-cli: fixed mess with error texts * Storage-cli: write, show welcome message only if we didn't have any errors * Storage: poorly fixed folders copying * Storage: close an old file if an error occurred while opening a new file * Storage-cli: fixed storage info spacing * Scripts: use positional arguments in storage. * Scripts: explicit string encoding and decoding, documentation in comments. Co-authored-by: あく <alleteam@gmail.com>
		
			
				
	
	
		
			162 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			162 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "../storage-settings.h"
 | |
| 
 | |
| #define BENCH_DATA_SIZE 4096
 | |
| #define BENCH_COUNT 6
 | |
| #define BENCH_REPEATS 4
 | |
| #define BENCH_FILE "/ext/rwfiletest.bin"
 | |
| 
 | |
| static void
 | |
|     storage_settings_scene_benchmark_dialog_callback(DialogExResult result, void* context) {
 | |
|     StorageSettings* app = context;
 | |
| 
 | |
|     view_dispatcher_send_custom_event(app->view_dispatcher, result);
 | |
| }
 | |
| 
 | |
| static bool storage_settings_bench_write(
 | |
|     Storage* api,
 | |
|     uint16_t size,
 | |
|     const uint8_t* data,
 | |
|     uint32_t* speed) {
 | |
|     File* file = storage_file_alloc(api);
 | |
|     bool result = true;
 | |
|     if(storage_file_open(file, BENCH_FILE, FSAM_WRITE, FSOM_CREATE_ALWAYS)) {
 | |
|         uint32_t ticks;
 | |
|         ticks = osKernelGetTickCount();
 | |
| 
 | |
|         for(size_t repeat = 0; repeat < BENCH_REPEATS; repeat++) {
 | |
|             for(size_t i = 0; i < BENCH_DATA_SIZE / size; i++) {
 | |
|                 if(storage_file_write(file, (data + i * size), size) != size) {
 | |
|                     result = false;
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         ticks = osKernelGetTickCount() - ticks;
 | |
|         *speed = BENCH_DATA_SIZE * osKernelGetTickFreq() * BENCH_REPEATS;
 | |
|         *speed /= ticks;
 | |
|         *speed /= 1024;
 | |
|     }
 | |
|     storage_file_close(file);
 | |
|     storage_file_free(file);
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| static bool
 | |
|     storage_settings_bench_read(Storage* api, uint16_t size, uint8_t* data, uint32_t* speed) {
 | |
|     File* file = storage_file_alloc(api);
 | |
|     bool result = true;
 | |
|     *speed = -1;
 | |
| 
 | |
|     if(storage_file_open(file, BENCH_FILE, FSAM_READ, FSOM_OPEN_EXISTING)) {
 | |
|         uint32_t ticks;
 | |
|         ticks = osKernelGetTickCount();
 | |
| 
 | |
|         for(size_t repeat = 0; repeat < BENCH_REPEATS; repeat++) {
 | |
|             for(size_t i = 0; i < BENCH_DATA_SIZE / size; i++) {
 | |
|                 if(storage_file_read(file, (data + i * size), size) != size) {
 | |
|                     result = false;
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         ticks = osKernelGetTickCount() - ticks;
 | |
|         *speed = BENCH_DATA_SIZE * osKernelGetTickFreq() * BENCH_REPEATS;
 | |
|         *speed /= ticks;
 | |
|         *speed /= 1024;
 | |
|     }
 | |
|     storage_file_close(file);
 | |
|     storage_file_free(file);
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| static void storage_settings_benchmark(StorageSettings* app) {
 | |
|     DialogEx* dialog_ex = app->dialog_ex;
 | |
|     uint8_t* bench_data;
 | |
|     dialog_ex_set_header(dialog_ex, "Preparing data...", 64, 32, AlignCenter, AlignCenter);
 | |
| 
 | |
|     bench_data = malloc(BENCH_DATA_SIZE);
 | |
|     for(size_t i = 0; i < BENCH_DATA_SIZE; i++) {
 | |
|         bench_data[i] = (uint8_t)i;
 | |
|     }
 | |
| 
 | |
|     uint16_t bench_size[BENCH_COUNT] = {1, 8, 32, 256, 512, 1024};
 | |
|     uint32_t bench_w_speed[BENCH_COUNT] = {0, 0, 0, 0, 0, 0};
 | |
|     uint32_t bench_r_speed[BENCH_COUNT] = {0, 0, 0, 0, 0, 0};
 | |
| 
 | |
|     dialog_ex_set_header(dialog_ex, "Benchmarking...", 64, 32, AlignCenter, AlignCenter);
 | |
|     for(size_t i = 0; i < BENCH_COUNT; i++) {
 | |
|         if(!storage_settings_bench_write(app->fs_api, bench_size[i], bench_data, &bench_w_speed[i]))
 | |
|             break;
 | |
| 
 | |
|         if(i > 0) string_cat_printf(app->text_string, "\n");
 | |
|         string_cat_printf(app->text_string, "%ub : W %luK ", bench_size[i], bench_w_speed[i]);
 | |
|         dialog_ex_set_header(dialog_ex, NULL, 0, 0, AlignCenter, AlignCenter);
 | |
|         dialog_ex_set_text(
 | |
|             dialog_ex, string_get_cstr(app->text_string), 0, 32, AlignLeft, AlignCenter);
 | |
| 
 | |
|         if(!storage_settings_bench_read(app->fs_api, bench_size[i], bench_data, &bench_r_speed[i]))
 | |
|             break;
 | |
| 
 | |
|         string_cat_printf(app->text_string, "R %luK", bench_r_speed[i]);
 | |
|         dialog_ex_set_text(
 | |
|             dialog_ex, string_get_cstr(app->text_string), 0, 32, AlignLeft, AlignCenter);
 | |
|     }
 | |
| 
 | |
|     free(bench_data);
 | |
| }
 | |
| 
 | |
| void storage_settings_scene_benchmark_on_enter(void* context) {
 | |
|     StorageSettings* app = context;
 | |
|     DialogEx* dialog_ex = app->dialog_ex;
 | |
| 
 | |
|     dialog_ex_set_context(dialog_ex, app);
 | |
|     dialog_ex_set_result_callback(dialog_ex, storage_settings_scene_benchmark_dialog_callback);
 | |
|     view_dispatcher_switch_to_view(app->view_dispatcher, StorageSettingsViewDialogEx);
 | |
| 
 | |
|     if(storage_sd_status(app->fs_api) != FSE_OK) {
 | |
|         dialog_ex_set_header(dialog_ex, "SD card not mounted", 64, 10, AlignCenter, AlignCenter);
 | |
|         dialog_ex_set_text(
 | |
|             dialog_ex,
 | |
|             "If an SD card is inserted,\r\npull it out and reinsert it",
 | |
|             64,
 | |
|             32,
 | |
|             AlignCenter,
 | |
|             AlignCenter);
 | |
|         dialog_ex_set_left_button_text(dialog_ex, "Back");
 | |
|     } else {
 | |
|         storage_settings_benchmark(app);
 | |
|         notification_message(app->notification, &sequence_blink_green_100);
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool storage_settings_scene_benchmark_on_event(void* context, SceneManagerEvent event) {
 | |
|     StorageSettings* app = context;
 | |
|     bool consumed = false;
 | |
| 
 | |
|     if(event.type == SceneManagerEventTypeCustom) {
 | |
|         switch(event.event) {
 | |
|         case DialogExResultLeft:
 | |
|             consumed = scene_manager_previous_scene(app->scene_manager);
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
|     return consumed;
 | |
| }
 | |
| 
 | |
| void storage_settings_scene_benchmark_on_exit(void* context) {
 | |
|     StorageSettings* app = context;
 | |
|     DialogEx* dialog_ex = app->dialog_ex;
 | |
| 
 | |
|     dialog_ex_set_header(dialog_ex, NULL, 0, 0, AlignCenter, AlignCenter);
 | |
|     dialog_ex_set_text(dialog_ex, NULL, 0, 0, AlignCenter, AlignTop);
 | |
|     dialog_ex_set_icon(dialog_ex, 0, 0, NULL);
 | |
|     dialog_ex_set_left_button_text(dialog_ex, NULL);
 | |
|     dialog_ex_set_right_button_text(dialog_ex, NULL);
 | |
|     dialog_ex_set_result_callback(dialog_ex, NULL);
 | |
|     dialog_ex_set_context(dialog_ex, NULL);
 | |
| 
 | |
|     string_clean(app->text_string);
 | |
| }
 |