Ingen beskrivning
Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

sequence.c 3.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. /*
  2. * sequence.c
  3. *
  4. * Copyright (c) 2024 Thomas Buck (thomas@xythobuz.de)
  5. *
  6. * This program is free software: you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License as published by
  8. * the Free Software Foundation, either version 3 of the License, or
  9. * (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * See <http://www.gnu.org/licenses/>.
  17. */
  18. #include <stdio.h>
  19. #include "pico/stdlib.h"
  20. #include "led.h"
  21. #include "sequence.h"
  22. #define MAX_BEATS 32
  23. static uint32_t ms_per_beat = 500;
  24. static uint32_t beats = 16;
  25. static uint32_t last_t = 0;
  26. static uint32_t last_i = 0;
  27. static enum channels sequence[MAX_BEATS] = {0};
  28. void sequence_init(void) {
  29. last_t = to_ms_since_boot(get_absolute_time());
  30. last_i = 0;
  31. }
  32. void sequence_set_bpm(uint32_t new_bpm) {
  33. ms_per_beat = 60000 / new_bpm;
  34. }
  35. void sequence_set_beats(uint32_t new_beats) {
  36. beats = (new_beats <= MAX_BEATS) ? new_beats : MAX_BEATS;
  37. }
  38. static void sequence_set(uint32_t beat, enum channels ch, bool value) {
  39. if (beat < MAX_BEATS) {
  40. if (value) {
  41. sequence[beat] |= ch;
  42. } else {
  43. sequence[beat] &= ~ch;
  44. }
  45. }
  46. }
  47. static bool sequence_get(uint32_t beat, enum channels ch) {
  48. if (beat < MAX_BEATS) {
  49. return (sequence[beat] & ch) != 0;
  50. }
  51. return false;
  52. }
  53. void sequence_handle_button_loopstation(enum buttons btn, bool rec) {
  54. switch (btn) {
  55. case BTN_A:
  56. // TODO trigger gpio impulse
  57. break;
  58. case BTN_B:
  59. // TODO trigger gpio impulse
  60. break;
  61. case BTN_C:
  62. // TODO trigger gpio impulse
  63. break;
  64. default:
  65. break;
  66. }
  67. if (rec) {
  68. uint32_t beat = 42; // TODO!!
  69. switch (btn) {
  70. case BTN_A:
  71. sequence_set(beat, CH_KICK, true);
  72. break;
  73. case BTN_B:
  74. sequence_set(beat, CH_SNARE, true);
  75. break;
  76. case BTN_C:
  77. sequence_set(beat, CH_HIHAT, true);
  78. break;
  79. default:
  80. break;
  81. }
  82. }
  83. }
  84. void sequence_handle_button_drummachine(enum buttons btn) {
  85. uint32_t beat = 42; // TODO!!
  86. switch (btn) {
  87. case BTN_A:
  88. sequence_set(beat, CH_KICK, !sequence_get(beat, CH_KICK));
  89. break;
  90. case BTN_B:
  91. sequence_set(beat, CH_SNARE, !sequence_get(beat, CH_SNARE));
  92. break;
  93. case BTN_C:
  94. sequence_set(beat, CH_HIHAT, !sequence_get(beat, CH_HIHAT));
  95. break;
  96. default:
  97. break;
  98. }
  99. }
  100. void sequence_run(void) {
  101. uint32_t now = to_ms_since_boot(get_absolute_time());
  102. if ((last_t + ms_per_beat) >= now) {
  103. uint32_t i = last_i + 1;
  104. if (i >= beats) i = 0;
  105. led_set(last_i, false);
  106. led_set(i, true);
  107. if (sequence[i]) {
  108. // TODO trigger GPIO impulse
  109. }
  110. last_t = now;
  111. last_i = i;
  112. }
  113. }