util.c 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. #include "util.h"
  2. #include "macros.h"
  3. void reset_buffer(SoundEngine* sound_engine) {
  4. for(uint16_t i = 0; i < sound_engine->audio_buffer_size; i++) {
  5. sound_engine->audio_buffer[i] = 512;
  6. }
  7. }
  8. void stop_song(FlizzerTrackerApp* tracker) {
  9. tracker->tracker_engine.playing = false;
  10. tracker->editing = tracker->was_editing;
  11. for(int i = 0; i < SONG_MAX_CHANNELS; i++) {
  12. tracker->sound_engine.channel[i].adsr.volume = 0;
  13. tracker->tracker_engine.channel[i].channel_flags &= ~(TEC_PROGRAM_RUNNING);
  14. }
  15. stop();
  16. reset_buffer(&tracker->sound_engine);
  17. }
  18. void play_song(FlizzerTrackerApp* tracker, bool from_cursor) {
  19. uint16_t temppos = tracker->tracker_engine.pattern_position;
  20. stop_song(tracker);
  21. sound_engine_dma_init(
  22. (uint32_t)tracker->sound_engine.audio_buffer, tracker->sound_engine.audio_buffer_size);
  23. tracker->tracker_engine.playing = true;
  24. tracker->was_editing = tracker->editing;
  25. tracker->editing = false;
  26. if(!(from_cursor)) {
  27. tracker->tracker_engine.pattern_position = 0;
  28. temppos = 0;
  29. }
  30. tracker_engine_timer_init(tracker->song.rate);
  31. /*sound_engine_init_hardware(tracker->sound_engine.sample_rate,
  32. tracker->sound_engine.external_audio_output,
  33. tracker->sound_engine.audio_buffer,
  34. tracker->sound_engine.audio_buffer_size);
  35. tracker_engine_init_hardware(tracker->song.rate);*/
  36. tracker->tracker_engine.current_tick = 0;
  37. tracker_engine_set_song(&tracker->tracker_engine, &tracker->song);
  38. for(uint8_t i = 0; i < SONG_MAX_CHANNELS; i++)
  39. {
  40. bool was_disabled = tracker->tracker_engine.channel[i].channel_flags & TEC_DISABLED;
  41. memset(&tracker->sound_engine.channel[i], 0, sizeof(SoundEngineChannel));
  42. memset(&tracker->tracker_engine.channel[i], 0, sizeof(TrackerEngineChannel));
  43. if(was_disabled)
  44. {
  45. tracker->tracker_engine.channel[i].channel_flags |= TEC_DISABLED;
  46. }
  47. }
  48. tracker->tracker_engine.pattern_position = temppos;
  49. play();
  50. }
  51. bool is_pattern_empty(TrackerSong* song, uint8_t pattern) {
  52. TrackerSongPattern song_pattern = song->pattern[pattern];
  53. for(int i = 0; i < song->pattern_length; i++) {
  54. TrackerSongPatternStep* step = &song_pattern.step[i];
  55. if(tracker_engine_get_note(step) != MUS_NOTE_NONE ||
  56. tracker_engine_get_instrument(step) != MUS_NOTE_INSTRUMENT_NONE ||
  57. tracker_engine_get_volume(step) != MUS_NOTE_VOLUME_NONE ||
  58. tracker_engine_get_command(step) != 0) {
  59. return false;
  60. }
  61. }
  62. return true;
  63. }
  64. bool check_and_allocate_pattern(TrackerSong* song, uint8_t pattern) {
  65. if(pattern < song->num_patterns) // we can set this pattern since it already exists
  66. {
  67. return true;
  68. }
  69. else {
  70. if(song->pattern[pattern - 1].step == NULL)
  71. return false; // if we hop through several patterns (e.g. editing upper digit)
  72. if(!(is_pattern_empty(
  73. song, pattern - 1))) // don't let the user flood the song with empty patterns
  74. {
  75. song->pattern[pattern].step =
  76. malloc(sizeof(TrackerSongPatternStep) * song->pattern_length);
  77. set_empty_pattern(&song->pattern[pattern], song->pattern_length);
  78. song->num_patterns++;
  79. return true;
  80. }
  81. else {
  82. return false;
  83. }
  84. }
  85. }
  86. void resize_pattern(TrackerSongPattern* pattern, uint16_t old_length, uint16_t new_length) {
  87. TrackerSongPattern temp;
  88. temp.step = malloc((new_length) * sizeof(TrackerSongPatternStep));
  89. set_empty_pattern(&temp, new_length);
  90. memcpy(
  91. temp.step, pattern->step, my_min(old_length, new_length) * sizeof(TrackerSongPatternStep));
  92. free(pattern->step);
  93. pattern->step = temp.step;
  94. }
  95. void change_pattern_length(TrackerSong* song, uint16_t new_length) {
  96. for(int i = 0; i < MAX_PATTERNS; i++) {
  97. if(song->pattern[i].step) {
  98. resize_pattern(&song->pattern[i], song->pattern_length, new_length);
  99. }
  100. }
  101. song->pattern_length = new_length;
  102. }
  103. bool is_default_instrument(Instrument* inst) {
  104. Instrument* ref = malloc(sizeof(Instrument));
  105. set_default_instrument(ref);
  106. bool is_default = memcmp(ref, inst, sizeof(Instrument)) != 0 ? false : true;
  107. free(ref);
  108. return is_default;
  109. }
  110. bool check_and_allocate_instrument(TrackerSong* song, uint8_t inst) {
  111. if(inst < song->num_instruments) // we can go to this instrument since it already exists
  112. {
  113. return true;
  114. }
  115. else {
  116. if(inst >= MAX_INSTRUMENTS) return false;
  117. if(!(is_default_instrument(
  118. song->instrument
  119. [inst - 1]))) // don't let the user flood the song with default instrument
  120. {
  121. song->instrument[inst] = malloc(sizeof(Instrument));
  122. set_default_instrument(song->instrument[inst]);
  123. song->num_instruments++;
  124. return true;
  125. }
  126. else {
  127. return false;
  128. }
  129. }
  130. }
  131. void set_default_song(FlizzerTrackerApp* tracker) {
  132. tracker->tracker_engine.master_volume = 0x80;
  133. tracker->song.speed = 6;
  134. tracker->song.rate = tracker->tracker_engine.rate;
  135. tracker->song.num_instruments = 1;
  136. tracker->song.num_patterns = 5;
  137. tracker->song.num_sequence_steps = 1;
  138. tracker->song.pattern_length = 64;
  139. tracker->song.sequence.sequence_step[0].pattern_indices[0] = 1;
  140. tracker->song.sequence.sequence_step[0].pattern_indices[1] = 2;
  141. tracker->song.sequence.sequence_step[0].pattern_indices[2] = 3;
  142. tracker->song.sequence.sequence_step[0].pattern_indices[3] = 4;
  143. for(int i = 0; i < 5; i++) {
  144. tracker->song.pattern[i].step = malloc(64 * sizeof(TrackerSongPatternStep));
  145. memset(tracker->song.pattern[i].step, 0, 64 * sizeof(TrackerSongPatternStep));
  146. }
  147. for(int i = 0; i < 64; ++i) {
  148. for(int j = 0; j < 5; j++) {
  149. set_note(&tracker->song.pattern[j].step[i], MUS_NOTE_NONE);
  150. set_instrument(&tracker->song.pattern[j].step[i], MUS_NOTE_INSTRUMENT_NONE);
  151. set_volume(&tracker->song.pattern[j].step[i], MUS_NOTE_VOLUME_NONE);
  152. }
  153. }
  154. tracker->song.instrument[0] = malloc(sizeof(Instrument));
  155. set_default_instrument(tracker->song.instrument[0]);
  156. tracker->tracker_engine.playing = false;
  157. }