diskop.c 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259
  1. #include "diskop.h"
  2. #define CFG_FILENAME "settings.cfg"
  3. void save_instrument_inner(Stream* stream, Instrument* inst) {
  4. size_t rwops = stream_write(stream, (uint8_t*)inst->name, sizeof(inst->name));
  5. rwops = stream_write(stream, (uint8_t*)&inst->waveform, sizeof(inst->waveform));
  6. rwops = stream_write(stream, (uint8_t*)&inst->flags, sizeof(inst->flags));
  7. rwops = stream_write(
  8. stream, (uint8_t*)&inst->sound_engine_flags, sizeof(inst->sound_engine_flags));
  9. rwops = stream_write(stream, (uint8_t*)&inst->base_note, sizeof(inst->base_note));
  10. rwops = stream_write(stream, (uint8_t*)&inst->finetune, sizeof(inst->finetune));
  11. rwops = stream_write(stream, (uint8_t*)&inst->slide_speed, sizeof(inst->slide_speed));
  12. rwops = stream_write(stream, (uint8_t*)&inst->adsr, sizeof(inst->adsr));
  13. rwops = stream_write(stream, (uint8_t*)&inst->pw, sizeof(inst->pw));
  14. if(inst->sound_engine_flags & SE_ENABLE_RING_MOD) {
  15. rwops = stream_write(stream, (uint8_t*)&inst->ring_mod, sizeof(inst->ring_mod));
  16. }
  17. if(inst->sound_engine_flags & SE_ENABLE_HARD_SYNC) {
  18. rwops = stream_write(stream, (uint8_t*)&inst->hard_sync, sizeof(inst->hard_sync));
  19. }
  20. uint8_t progsteps = 0;
  21. for(uint8_t i = 0; i < INST_PROG_LEN; i++) {
  22. if((inst->program[i] & 0x7fff) != TE_PROGRAM_NOP) {
  23. progsteps = i + 1;
  24. }
  25. }
  26. rwops = stream_write(stream, (uint8_t*)&progsteps, sizeof(progsteps));
  27. if(progsteps > 0) {
  28. rwops =
  29. stream_write(stream, (uint8_t*)inst->program, progsteps * sizeof(inst->program[0]));
  30. }
  31. rwops = stream_write(stream, (uint8_t*)&inst->program_period, sizeof(inst->program_period));
  32. if(inst->flags & TE_ENABLE_VIBRATO) {
  33. rwops = stream_write(stream, (uint8_t*)&inst->vibrato_speed, sizeof(inst->vibrato_speed));
  34. rwops = stream_write(stream, (uint8_t*)&inst->vibrato_depth, sizeof(inst->vibrato_depth));
  35. rwops = stream_write(stream, (uint8_t*)&inst->vibrato_delay, sizeof(inst->vibrato_delay));
  36. }
  37. if(inst->flags & TE_ENABLE_PWM) {
  38. rwops = stream_write(stream, (uint8_t*)&inst->pwm_speed, sizeof(inst->pwm_speed));
  39. rwops = stream_write(stream, (uint8_t*)&inst->pwm_depth, sizeof(inst->pwm_depth));
  40. rwops = stream_write(stream, (uint8_t*)&inst->pwm_delay, sizeof(inst->pwm_delay));
  41. }
  42. if(inst->sound_engine_flags & SE_ENABLE_FILTER) {
  43. rwops = stream_write(stream, (uint8_t*)&inst->filter_cutoff, sizeof(inst->filter_cutoff));
  44. rwops = stream_write(
  45. stream, (uint8_t*)&inst->filter_resonance, sizeof(inst->filter_resonance));
  46. rwops = stream_write(stream, (uint8_t*)&inst->filter_type, sizeof(inst->filter_type));
  47. }
  48. UNUSED(rwops);
  49. }
  50. bool save_instrument(FlizzerTrackerApp* tracker, FuriString* filepath)
  51. {
  52. bool file_removed =
  53. storage_simply_remove(tracker->storage, furi_string_get_cstr(filepath)); // just in case
  54. bool open_file = file_stream_open(
  55. tracker->stream, furi_string_get_cstr(filepath), FSAM_WRITE, FSOM_OPEN_ALWAYS);
  56. uint8_t version = TRACKER_ENGINE_VERSION;
  57. size_t rwops =
  58. stream_write(tracker->stream, (uint8_t*)INST_FILE_SIG, sizeof(INST_FILE_SIG) - 1);
  59. rwops = stream_write(tracker->stream, (uint8_t*)&version, sizeof(uint8_t));
  60. Instrument* inst = tracker->song.instrument[tracker->current_instrument];
  61. save_instrument_inner(tracker->stream, inst);
  62. file_stream_close(tracker->stream);
  63. tracker->is_saving_instrument = false;
  64. furi_string_free(filepath);
  65. UNUSED(file_removed);
  66. UNUSED(open_file);
  67. UNUSED(rwops);
  68. return false;
  69. }
  70. bool save_song(FlizzerTrackerApp* tracker, FuriString* filepath) {
  71. bool file_removed =
  72. storage_simply_remove(tracker->storage, furi_string_get_cstr(filepath)); // just in case
  73. bool open_file = file_stream_open(
  74. tracker->stream, furi_string_get_cstr(filepath), FSAM_WRITE, FSOM_OPEN_ALWAYS);
  75. uint8_t version = TRACKER_ENGINE_VERSION;
  76. size_t rwops =
  77. stream_write(tracker->stream, (uint8_t*)SONG_FILE_SIG, sizeof(SONG_FILE_SIG) - 1);
  78. rwops = stream_write(tracker->stream, (uint8_t*)&version, sizeof(uint8_t));
  79. TrackerSong* song = &tracker->song;
  80. /*for(uint32_t i = 0; i < 23444; i++)
  81. {
  82. rwops = stream_write(tracker->stream, (uint8_t*)&song->loop_end, sizeof(uint8_t));
  83. }*/
  84. rwops = stream_write(tracker->stream, (uint8_t*)song->song_name, sizeof(song->song_name));
  85. rwops = stream_write(tracker->stream, (uint8_t*)&song->loop_start, sizeof(song->loop_start));
  86. rwops = stream_write(tracker->stream, (uint8_t*)&song->loop_end, sizeof(song->loop_end));
  87. rwops = stream_write(
  88. tracker->stream, (uint8_t*)&song->pattern_length, sizeof(song->pattern_length));
  89. rwops = stream_write(tracker->stream, (uint8_t*)&song->speed, sizeof(song->speed));
  90. rwops = stream_write(tracker->stream, (uint8_t*)&song->rate, sizeof(song->rate));
  91. rwops = stream_write(
  92. tracker->stream, (uint8_t*)&song->num_sequence_steps, sizeof(song->num_sequence_steps));
  93. for(uint16_t i = 0; i < song->num_sequence_steps; i++) {
  94. rwops = stream_write(
  95. tracker->stream,
  96. (uint8_t*)&song->sequence.sequence_step[i],
  97. sizeof(song->sequence.sequence_step[0]));
  98. }
  99. rwops =
  100. stream_write(tracker->stream, (uint8_t*)&song->num_patterns, sizeof(song->num_patterns));
  101. for(uint16_t i = 0; i < song->num_patterns; i++) {
  102. rwops = stream_write(
  103. tracker->stream,
  104. (uint8_t*)song->pattern[i].step,
  105. sizeof(TrackerSongPatternStep) * (song->pattern_length));
  106. }
  107. rwops = stream_write(
  108. tracker->stream, (uint8_t*)&song->num_instruments, sizeof(song->num_instruments));
  109. for(uint16_t i = 0; i < song->num_instruments; i++) {
  110. save_instrument_inner(tracker->stream, song->instrument[i]);
  111. }
  112. file_stream_close(tracker->stream);
  113. tracker->is_saving = false;
  114. furi_string_free(filepath);
  115. UNUSED(file_removed);
  116. UNUSED(open_file);
  117. UNUSED(rwops);
  118. return false;
  119. }
  120. bool load_song_util(FlizzerTrackerApp* tracker, FuriString* filepath) {
  121. bool open_file = file_stream_open(
  122. tracker->stream, furi_string_get_cstr(filepath), FSAM_READ, FSOM_OPEN_ALWAYS);
  123. bool result = load_song(&tracker->song, tracker->stream);
  124. tracker->is_loading = false;
  125. file_stream_close(tracker->stream);
  126. furi_string_free(filepath);
  127. UNUSED(open_file);
  128. return result;
  129. }
  130. bool load_instrument_disk(TrackerSong* song, uint8_t inst, Stream* stream) {
  131. char header[sizeof(INST_FILE_SIG) + 2] = {0};
  132. size_t rwops = stream_read(stream, (uint8_t*)&header, sizeof(INST_FILE_SIG) - 1);
  133. header[sizeof(INST_FILE_SIG)] = '\0';
  134. uint8_t version = 0;
  135. if(strcmp(header, INST_FILE_SIG) == 0) {
  136. rwops = stream_read(stream, (uint8_t*)&version, sizeof(version));
  137. if(version <= TRACKER_ENGINE_VERSION)
  138. {
  139. load_instrument_inner(stream, song->instrument[inst], version);
  140. }
  141. }
  142. UNUSED(rwops);
  143. return false;
  144. }
  145. bool load_instrument_util(FlizzerTrackerApp* tracker, FuriString* filepath)
  146. {
  147. bool open_file = file_stream_open(
  148. tracker->stream, furi_string_get_cstr(filepath), FSAM_READ, FSOM_OPEN_ALWAYS);
  149. bool result = load_instrument_disk(&tracker->song, tracker->current_instrument, tracker->stream);
  150. tracker->is_loading_instrument = false;
  151. file_stream_close(tracker->stream);
  152. furi_string_free(filepath);
  153. UNUSED(open_file);
  154. return result;
  155. }
  156. void save_config(FlizzerTrackerApp* tracker) {
  157. // stream_read_line
  158. FuriString* filepath = furi_string_alloc();
  159. FuriString* config_line = furi_string_alloc();
  160. furi_string_cat_printf(filepath, "%s/%s", FLIZZER_TRACKER_FOLDER, CFG_FILENAME);
  161. bool open_file = file_stream_open(
  162. tracker->stream, furi_string_get_cstr(filepath), FSAM_WRITE, FSOM_OPEN_ALWAYS);
  163. UNUSED(open_file);
  164. furi_string_cat_printf(
  165. config_line, "%s = %s\n", "external_audio", tracker->external_audio ? "true" : "false");
  166. stream_write_string(tracker->stream, config_line);
  167. file_stream_close(tracker->stream);
  168. furi_string_free(filepath);
  169. furi_string_free(config_line);
  170. }
  171. void load_config(FlizzerTrackerApp* tracker) {
  172. FuriString* filepath = furi_string_alloc();
  173. FuriString* config_line = furi_string_alloc();
  174. furi_string_cat_printf(filepath, "%s/%s", FLIZZER_TRACKER_FOLDER, CFG_FILENAME);
  175. bool open_file = file_stream_open(
  176. tracker->stream, furi_string_get_cstr(filepath), FSAM_READ, FSOM_OPEN_ALWAYS);
  177. UNUSED(open_file);
  178. stream_read_line(tracker->stream, config_line);
  179. sscanf(
  180. furi_string_get_cstr(config_line), "%s%s%s", tracker->param, tracker->eq, tracker->value);
  181. if(strcmp(tracker->param, "external_audio") == 0) {
  182. if(strcmp(tracker->value, "false") == 0) {
  183. tracker->external_audio = false;
  184. // strcpy(tracker->value, "false_");
  185. }
  186. if(strcmp(tracker->value, "true") == 0) {
  187. tracker->external_audio = true;
  188. // strcpy(tracker->value, "true_");
  189. }
  190. sound_engine_init(
  191. &tracker->sound_engine,
  192. tracker->sound_engine.sample_rate,
  193. tracker->external_audio,
  194. tracker->sound_engine.audio_buffer_size);
  195. // sound_engine_set_audio_output(tracker->external_audio);
  196. }
  197. file_stream_close(tracker->stream);
  198. furi_string_free(filepath);
  199. furi_string_free(config_line);
  200. }