Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

6 роки тому
6 роки тому
6 роки тому
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. /* Teensyduino Core Library
  2. * http://www.pjrc.com/teensy/
  3. * Copyright (c) 2017 PJRC.COM, LLC.
  4. *
  5. * Permission is hereby granted, free of charge, to any person obtaining
  6. * a copy of this software and associated documentation files (the
  7. * "Software"), to deal in the Software without restriction, including
  8. * without limitation the rights to use, copy, modify, merge, publish,
  9. * distribute, sublicense, and/or sell copies of the Software, and to
  10. * permit persons to whom the Software is furnished to do so, subject to
  11. * the following conditions:
  12. *
  13. * 1. The above copyright notice and this permission notice shall be
  14. * included in all copies or substantial portions of the Software.
  15. *
  16. * 2. If the Software is incorporated into a build system that allows
  17. * selection among a list of target devices, then similar target
  18. * devices manufactured by PJRC.COM must be included in the list of
  19. * target devices and selectable in the same manner.
  20. *
  21. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  22. * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  23. * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  24. * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
  25. * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
  26. * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
  27. * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  28. * SOFTWARE.
  29. */
  30. #ifndef AudioStream_h
  31. #define AudioStream_h
  32. #ifndef __ASSEMBLER__
  33. #include <stdio.h> // for NULL
  34. #include <string.h> // for memcpy
  35. #endif
  36. // AUDIO_BLOCK_SAMPLES determines how many samples the audio library processes
  37. // per update. It may be reduced to achieve lower latency response to events,
  38. // at the expense of higher interrupt and DMA setup overhead.
  39. //
  40. // Less than 32 may not work with some input & output objects. Multiples of 16
  41. // should be used, since some synthesis objects generate 16 samples per loop.
  42. //
  43. // Some parts of the audio library may have hard-coded dependency on 128 samples.
  44. // Please report these on the forum with reproducible test cases. The following
  45. // audio classes are known to have problems with smaller block sizes:
  46. // AudioInputUSB, AudioOutputUSB, AudioPlaySdWav, AudioAnalyzeFFT256,
  47. // AudioAnalyzeFFT1024
  48. #ifndef AUDIO_BLOCK_SAMPLES
  49. #define AUDIO_BLOCK_SAMPLES 128
  50. #endif
  51. #ifndef AUDIO_SAMPLE_RATE_EXACT
  52. #define AUDIO_SAMPLE_RATE_EXACT 44100.0f
  53. #endif
  54. #define AUDIO_SAMPLE_RATE AUDIO_SAMPLE_RATE_EXACT
  55. #ifndef __ASSEMBLER__
  56. class AudioStream;
  57. class AudioConnection;
  58. typedef struct audio_block_struct {
  59. uint8_t ref_count;
  60. uint8_t reserved1;
  61. uint16_t memory_pool_index;
  62. int16_t data[AUDIO_BLOCK_SAMPLES];
  63. } audio_block_t;
  64. class AudioConnection
  65. {
  66. public:
  67. AudioConnection(AudioStream &source, AudioStream &destination) :
  68. src(source), dst(destination), src_index(0), dest_index(0),
  69. next_dest(NULL)
  70. { isConnected = false;
  71. connect(); }
  72. AudioConnection(AudioStream &source, unsigned char sourceOutput,
  73. AudioStream &destination, unsigned char destinationInput) :
  74. src(source), dst(destination),
  75. src_index(sourceOutput), dest_index(destinationInput),
  76. next_dest(NULL)
  77. { isConnected = false;
  78. connect(); }
  79. friend class AudioStream;
  80. ~AudioConnection() {
  81. disconnect();
  82. }
  83. void disconnect(void);
  84. void connect(void);
  85. protected:
  86. AudioStream &src;
  87. AudioStream &dst;
  88. unsigned char src_index;
  89. unsigned char dest_index;
  90. AudioConnection *next_dest;
  91. bool isConnected;
  92. };
  93. #define AudioMemory(num) ({ \
  94. static DMAMEM audio_block_t data[num]; \
  95. AudioStream::initialize_memory(data, num); \
  96. })
  97. #define CYCLE_COUNTER_APPROX_PERCENT(n) (((n) + (F_CPU_ACTUAL / 128 / AUDIO_SAMPLE_RATE * AUDIO_BLOCK_SAMPLES / 100)) / (F_CPU_ACTUAL / 64 / AUDIO_SAMPLE_RATE * AUDIO_BLOCK_SAMPLES / 100))
  98. #define AudioProcessorUsage() (CYCLE_COUNTER_APPROX_PERCENT(AudioStream::cpu_cycles_total))
  99. #define AudioProcessorUsageMax() (CYCLE_COUNTER_APPROX_PERCENT(AudioStream::cpu_cycles_total_max))
  100. #define AudioProcessorUsageMaxReset() (AudioStream::cpu_cycles_total_max = AudioStream::cpu_cycles_total)
  101. #define AudioMemoryUsage() (AudioStream::memory_used)
  102. #define AudioMemoryUsageMax() (AudioStream::memory_used_max)
  103. #define AudioMemoryUsageMaxReset() (AudioStream::memory_used_max = AudioStream::memory_used)
  104. class AudioStream
  105. {
  106. public:
  107. AudioStream(unsigned char ninput, audio_block_t **iqueue) :
  108. num_inputs(ninput), inputQueue(iqueue) {
  109. active = false;
  110. destination_list = NULL;
  111. for (int i=0; i < num_inputs; i++) {
  112. inputQueue[i] = NULL;
  113. }
  114. // add to a simple list, for update_all
  115. // TODO: replace with a proper data flow analysis in update_all
  116. if (first_update == NULL) {
  117. first_update = this;
  118. } else {
  119. AudioStream *p;
  120. for (p=first_update; p->next_update; p = p->next_update) ;
  121. p->next_update = this;
  122. }
  123. next_update = NULL;
  124. cpu_cycles = 0;
  125. cpu_cycles_max = 0;
  126. numConnections = 0;
  127. }
  128. static void initialize_memory(audio_block_t *data, unsigned int num);
  129. int processorUsage(void) { return CYCLE_COUNTER_APPROX_PERCENT(cpu_cycles); }
  130. int processorUsageMax(void) { return CYCLE_COUNTER_APPROX_PERCENT(cpu_cycles_max); }
  131. void processorUsageMaxReset(void) { cpu_cycles_max = cpu_cycles; }
  132. bool isActive(void) { return active; }
  133. uint16_t cpu_cycles;
  134. uint16_t cpu_cycles_max;
  135. static uint16_t cpu_cycles_total;
  136. static uint16_t cpu_cycles_total_max;
  137. static uint16_t memory_used;
  138. static uint16_t memory_used_max;
  139. protected:
  140. bool active;
  141. unsigned char num_inputs;
  142. static audio_block_t * allocate(void);
  143. static void release(audio_block_t * block);
  144. void transmit(audio_block_t *block, unsigned char index = 0);
  145. audio_block_t * receiveReadOnly(unsigned int index = 0);
  146. audio_block_t * receiveWritable(unsigned int index = 0);
  147. static bool update_setup(void);
  148. static void update_stop(void);
  149. static void update_all(void) { NVIC_SET_PENDING(IRQ_SOFTWARE); }
  150. friend void software_isr(void);
  151. friend class AudioConnection;
  152. uint8_t numConnections;
  153. private:
  154. AudioConnection *destination_list;
  155. audio_block_t **inputQueue;
  156. static bool update_scheduled;
  157. virtual void update(void) = 0;
  158. static AudioStream *first_update; // for update_all
  159. AudioStream *next_update; // for update_all
  160. static audio_block_t *memory_pool;
  161. static uint32_t memory_pool_available_mask[];
  162. static uint16_t memory_pool_first_mask;
  163. };
  164. #endif
  165. #endif