util.c 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. #include "util.h"
  2. void reset_buffer(SoundEngine *sound_engine)
  3. {
  4. for (uint16_t i = 0; i < sound_engine->audio_buffer_size; i++)
  5. {
  6. sound_engine->audio_buffer[i] = 512;
  7. }
  8. }
  9. void play_song(FlizzerTrackerApp *tracker, bool from_cursor)
  10. {
  11. reset_buffer(&tracker->sound_engine);
  12. sound_engine_dma_init((uint32_t)tracker->sound_engine.audio_buffer, tracker->sound_engine.audio_buffer_size);
  13. tracker->tracker_engine.playing = true;
  14. tracker->was_editing = tracker->editing;
  15. tracker->editing = false;
  16. if (!(from_cursor))
  17. {
  18. tracker->tracker_engine.pattern_position = 0;
  19. }
  20. tracker_engine_set_rate(tracker->song.rate);
  21. tracker->tracker_engine.current_tick = 0;
  22. tracker_engine_set_song(&tracker->tracker_engine, &tracker->song);
  23. play();
  24. }
  25. void stop_song(FlizzerTrackerApp *tracker)
  26. {
  27. tracker->tracker_engine.playing = false;
  28. tracker->editing = tracker->was_editing;
  29. stop();
  30. }
  31. bool is_pattern_empty(TrackerSong *song, uint8_t pattern)
  32. {
  33. TrackerSongPattern song_pattern = song->pattern[pattern];
  34. for (int i = 0; i < song->pattern_length; i++)
  35. {
  36. TrackerSongPatternStep *step = &song_pattern.step[i];
  37. if (tracker_engine_get_note(step) != MUS_NOTE_NONE ||
  38. tracker_engine_get_instrument(step) != MUS_NOTE_INSTRUMENT_NONE ||
  39. tracker_engine_get_volume(step) != MUS_NOTE_VOLUME_NONE ||
  40. tracker_engine_get_command(step) != 0)
  41. {
  42. return false;
  43. }
  44. }
  45. return true;
  46. }
  47. bool check_and_allocate_pattern(TrackerSong *song, uint8_t pattern)
  48. {
  49. if (pattern < song->num_patterns) // we can set this pattern since it already exists
  50. {
  51. return true;
  52. }
  53. else
  54. {
  55. if (song->pattern[pattern - 1].step == NULL)
  56. return false; // if we hop through several patterns (e.g. editing upper digit)
  57. if (!(is_pattern_empty(song, pattern - 1))) // don't let the user flood the song with empty patterns
  58. {
  59. song->pattern[pattern].step = malloc(sizeof(TrackerSongPatternStep) * song->pattern_length);
  60. set_empty_pattern(&song->pattern[pattern], song->pattern_length);
  61. song->num_patterns++;
  62. return true;
  63. }
  64. else
  65. {
  66. return false;
  67. }
  68. }
  69. }
  70. void resize_pattern(TrackerSongPattern *pattern, uint16_t old_length, uint16_t new_length)
  71. {
  72. TrackerSongPattern temp;
  73. temp.step = malloc((new_length) * sizeof(TrackerSongPatternStep));
  74. set_empty_pattern(&temp, new_length);
  75. memcpy(temp.step, pattern->step, fmin(old_length, new_length) * sizeof(TrackerSongPatternStep));
  76. free(pattern->step);
  77. pattern->step = temp.step;
  78. }
  79. void change_pattern_length(TrackerSong *song, uint16_t new_length)
  80. {
  81. for (int i = 0; i < MAX_PATTERNS; i++)
  82. {
  83. if (song->pattern[i].step)
  84. {
  85. resize_pattern(&song->pattern[i], song->pattern_length, new_length);
  86. }
  87. }
  88. song->pattern_length = new_length;
  89. }
  90. bool is_default_instrument(Instrument *inst)
  91. {
  92. Instrument *ref = malloc(sizeof(Instrument));
  93. set_default_instrument(ref);
  94. bool is_default = memcmp(ref, inst, sizeof(Instrument)) != 0 ? false : true;
  95. free(ref);
  96. return is_default;
  97. }
  98. bool check_and_allocate_instrument(TrackerSong *song, uint8_t inst)
  99. {
  100. if (inst < song->num_instruments) // we can go to this instrument since it already exists
  101. {
  102. return true;
  103. }
  104. else
  105. {
  106. if (inst >= MAX_INSTRUMENTS)
  107. return false;
  108. if (!(is_default_instrument(song->instrument[inst - 1]))) // don't let the user flood the song with default instrument
  109. {
  110. song->instrument[inst] = malloc(sizeof(Instrument));
  111. set_default_instrument(song->instrument[inst]);
  112. song->num_instruments++;
  113. return true;
  114. }
  115. else
  116. {
  117. return false;
  118. }
  119. }
  120. }