2018-03-12 16:09:13 -07:00
|
|
|
/*
|
|
|
|
* This file is part of the MicroPython project, http://micropython.org/
|
|
|
|
*
|
|
|
|
* The MIT License (MIT)
|
|
|
|
*
|
|
|
|
* Copyright (c) 2018 Scott Shawcroft for Adafruit Industries
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "audio_dma.h"
|
2018-06-15 16:16:21 -07:00
|
|
|
#include "samd/clocks.h"
|
|
|
|
#include "samd/events.h"
|
|
|
|
#include "samd/dma.h"
|
2018-03-12 16:09:13 -07:00
|
|
|
|
2019-07-24 20:57:32 -05:00
|
|
|
#include "shared-bindings/audiocore/RawSample.h"
|
|
|
|
#include "shared-bindings/audiocore/WaveFile.h"
|
2020-07-07 10:11:29 -05:00
|
|
|
#include "supervisor/background_callback.h"
|
2018-03-12 16:09:13 -07:00
|
|
|
|
|
|
|
#include "py/mpstate.h"
|
2018-08-15 18:32:37 -07:00
|
|
|
#include "py/runtime.h"
|
2018-03-12 16:09:13 -07:00
|
|
|
|
2019-08-07 21:29:24 -05:00
|
|
|
#if CIRCUITPY_AUDIOIO || CIRCUITPY_AUDIOBUSIO
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
static audio_dma_t *audio_dma_state[AUDIO_DMA_CHANNEL_COUNT];
|
2018-06-14 18:47:40 -04:00
|
|
|
|
|
|
|
// This cannot be in audio_dma_state because it's volatile.
|
|
|
|
static volatile bool audio_dma_pending[AUDIO_DMA_CHANNEL_COUNT];
|
|
|
|
|
samd: audio_dma: Track channel allocation
Previously, we depended on allocated channels to always be
"dma_channel_enabled". However, (A) sometimes, many operations
would take place between find_free_audio_dma_channel and
audio_dma_enable_channel, and (B) some debugging I did led me to believe
that "dma_channel_enabled" would become false when the hardware ended
a scheduled DMA transaction, but while a CP object would still think it
owned the DMA channel.
((B) is not documented in the datasheet and I am not 100% convinced that
my debugging session was not simply missing where we were disabling the
channel, but in either case, it shows a need to directly track allocated
separately from enabled)
Therefore,
* Add audio_dma_{allocate,free}_channel.
* audio_dma_free_channel implies audio_dma_disable_channel
* track via a new array audio_dma_allocated[]
* clear all allocated flags on soft-reboot
* Convert find_free_audio_dma_channel to audio_dma_allocate_channel
* use audio_dma_allocated[] instead of dma_channel_enabled() to check
availability
* remove find_free_audio_dma_channel
* For each one, find a matching audio_dma_disable_channel to convert
to audio_dma_free_channel
Closes: #2058
2019-08-28 16:55:17 -05:00
|
|
|
static bool audio_dma_allocated[AUDIO_DMA_CHANNEL_COUNT];
|
|
|
|
|
2021-04-20 13:33:11 -05:00
|
|
|
uint8_t find_sync_event_channel_raise() {
|
|
|
|
uint8_t event_channel = find_sync_event_channel();
|
|
|
|
if (event_channel >= EVSYS_SYNCH_NUM) {
|
|
|
|
mp_raise_RuntimeError(translate("All sync event channels in use"));
|
|
|
|
}
|
|
|
|
return event_channel;
|
|
|
|
}
|
|
|
|
|
2021-04-23 09:46:33 -05:00
|
|
|
uint8_t dma_allocate_channel(void) {
|
2018-03-12 16:09:13 -07:00
|
|
|
uint8_t channel;
|
|
|
|
for (channel = 0; channel < AUDIO_DMA_CHANNEL_COUNT; channel++) {
|
samd: audio_dma: Track channel allocation
Previously, we depended on allocated channels to always be
"dma_channel_enabled". However, (A) sometimes, many operations
would take place between find_free_audio_dma_channel and
audio_dma_enable_channel, and (B) some debugging I did led me to believe
that "dma_channel_enabled" would become false when the hardware ended
a scheduled DMA transaction, but while a CP object would still think it
owned the DMA channel.
((B) is not documented in the datasheet and I am not 100% convinced that
my debugging session was not simply missing where we were disabling the
channel, but in either case, it shows a need to directly track allocated
separately from enabled)
Therefore,
* Add audio_dma_{allocate,free}_channel.
* audio_dma_free_channel implies audio_dma_disable_channel
* track via a new array audio_dma_allocated[]
* clear all allocated flags on soft-reboot
* Convert find_free_audio_dma_channel to audio_dma_allocate_channel
* use audio_dma_allocated[] instead of dma_channel_enabled() to check
availability
* remove find_free_audio_dma_channel
* For each one, find a matching audio_dma_disable_channel to convert
to audio_dma_free_channel
Closes: #2058
2019-08-28 16:55:17 -05:00
|
|
|
if (!audio_dma_allocated[channel]) {
|
|
|
|
audio_dma_allocated[channel] = true;
|
2018-03-12 16:09:13 -07:00
|
|
|
return channel;
|
|
|
|
}
|
|
|
|
}
|
samd: audio_dma: Track channel allocation
Previously, we depended on allocated channels to always be
"dma_channel_enabled". However, (A) sometimes, many operations
would take place between find_free_audio_dma_channel and
audio_dma_enable_channel, and (B) some debugging I did led me to believe
that "dma_channel_enabled" would become false when the hardware ended
a scheduled DMA transaction, but while a CP object would still think it
owned the DMA channel.
((B) is not documented in the datasheet and I am not 100% convinced that
my debugging session was not simply missing where we were disabling the
channel, but in either case, it shows a need to directly track allocated
separately from enabled)
Therefore,
* Add audio_dma_{allocate,free}_channel.
* audio_dma_free_channel implies audio_dma_disable_channel
* track via a new array audio_dma_allocated[]
* clear all allocated flags on soft-reboot
* Convert find_free_audio_dma_channel to audio_dma_allocate_channel
* use audio_dma_allocated[] instead of dma_channel_enabled() to check
availability
* remove find_free_audio_dma_channel
* For each one, find a matching audio_dma_disable_channel to convert
to audio_dma_free_channel
Closes: #2058
2019-08-28 16:55:17 -05:00
|
|
|
return channel; // i.e., return failure
|
|
|
|
}
|
|
|
|
|
2021-04-23 09:46:33 -05:00
|
|
|
void dma_free_channel(uint8_t channel) {
|
samd: audio_dma: Track channel allocation
Previously, we depended on allocated channels to always be
"dma_channel_enabled". However, (A) sometimes, many operations
would take place between find_free_audio_dma_channel and
audio_dma_enable_channel, and (B) some debugging I did led me to believe
that "dma_channel_enabled" would become false when the hardware ended
a scheduled DMA transaction, but while a CP object would still think it
owned the DMA channel.
((B) is not documented in the datasheet and I am not 100% convinced that
my debugging session was not simply missing where we were disabling the
channel, but in either case, it shows a need to directly track allocated
separately from enabled)
Therefore,
* Add audio_dma_{allocate,free}_channel.
* audio_dma_free_channel implies audio_dma_disable_channel
* track via a new array audio_dma_allocated[]
* clear all allocated flags on soft-reboot
* Convert find_free_audio_dma_channel to audio_dma_allocate_channel
* use audio_dma_allocated[] instead of dma_channel_enabled() to check
availability
* remove find_free_audio_dma_channel
* For each one, find a matching audio_dma_disable_channel to convert
to audio_dma_free_channel
Closes: #2058
2019-08-28 16:55:17 -05:00
|
|
|
assert(channel < AUDIO_DMA_CHANNEL_COUNT);
|
|
|
|
assert(audio_dma_allocated[channel]);
|
|
|
|
audio_dma_disable_channel(channel);
|
|
|
|
audio_dma_allocated[channel] = false;
|
2018-03-12 16:09:13 -07:00
|
|
|
}
|
|
|
|
|
2019-08-06 21:42:01 -05:00
|
|
|
void audio_dma_disable_channel(uint8_t channel) {
|
2021-03-15 19:27:36 +05:30
|
|
|
if (channel >= AUDIO_DMA_CHANNEL_COUNT) {
|
2019-08-07 20:20:36 -05:00
|
|
|
return;
|
2021-03-15 19:27:36 +05:30
|
|
|
}
|
2019-08-06 21:42:01 -05:00
|
|
|
dma_disable_channel(channel);
|
|
|
|
}
|
|
|
|
|
|
|
|
void audio_dma_enable_channel(uint8_t channel) {
|
2021-03-15 19:27:36 +05:30
|
|
|
if (channel >= AUDIO_DMA_CHANNEL_COUNT) {
|
2019-08-07 20:20:36 -05:00
|
|
|
return;
|
2021-03-15 19:27:36 +05:30
|
|
|
}
|
2019-08-06 21:42:01 -05:00
|
|
|
dma_enable_channel(channel);
|
|
|
|
}
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
void audio_dma_convert_signed(audio_dma_t *dma, uint8_t *buffer, uint32_t buffer_length,
|
|
|
|
uint8_t **output_buffer, uint32_t *output_buffer_length,
|
|
|
|
uint8_t *output_spacing) {
|
2018-03-12 16:09:13 -07:00
|
|
|
if (dma->first_buffer_free) {
|
|
|
|
*output_buffer = dma->first_buffer;
|
|
|
|
} else {
|
|
|
|
*output_buffer = dma->second_buffer;
|
|
|
|
}
|
|
|
|
#pragma GCC diagnostic push
|
|
|
|
#pragma GCC diagnostic ignored "-Wcast-align"
|
|
|
|
if (dma->signed_to_unsigned || dma->unsigned_to_signed) {
|
|
|
|
*output_buffer_length = buffer_length / dma->spacing;
|
|
|
|
*output_spacing = 1;
|
|
|
|
uint32_t out_i = 0;
|
|
|
|
if (dma->bytes_per_sample == 1) {
|
|
|
|
for (uint32_t i = 0; i < buffer_length; i += dma->spacing) {
|
|
|
|
if (dma->signed_to_unsigned) {
|
2021-03-15 19:27:36 +05:30
|
|
|
((uint8_t *)*output_buffer)[out_i] = ((int8_t *)buffer)[i] + 0x80;
|
2018-03-12 16:09:13 -07:00
|
|
|
} else {
|
2021-03-15 19:27:36 +05:30
|
|
|
((int8_t *)*output_buffer)[out_i] = ((uint8_t *)buffer)[i] - 0x80;
|
2018-03-12 16:09:13 -07:00
|
|
|
}
|
|
|
|
out_i += 1;
|
|
|
|
}
|
|
|
|
} else if (dma->bytes_per_sample == 2) {
|
|
|
|
for (uint32_t i = 0; i < buffer_length / 2; i += dma->spacing) {
|
|
|
|
if (dma->signed_to_unsigned) {
|
2021-03-15 19:27:36 +05:30
|
|
|
((uint16_t *)*output_buffer)[out_i] = ((int16_t *)buffer)[i] + 0x8000;
|
2018-03-12 16:09:13 -07:00
|
|
|
} else {
|
2021-03-15 19:27:36 +05:30
|
|
|
((int16_t *)*output_buffer)[out_i] = ((uint16_t *)buffer)[i] - 0x8000;
|
2018-03-12 16:09:13 -07:00
|
|
|
}
|
|
|
|
out_i += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
*output_buffer = buffer;
|
|
|
|
*output_buffer_length = buffer_length;
|
|
|
|
*output_spacing = dma->spacing;
|
|
|
|
}
|
|
|
|
#pragma GCC diagnostic pop
|
|
|
|
dma->first_buffer_free = !dma->first_buffer_free;
|
|
|
|
}
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
void audio_dma_load_next_block(audio_dma_t *dma) {
|
|
|
|
uint8_t *buffer;
|
2018-03-12 16:09:13 -07:00
|
|
|
uint32_t buffer_length;
|
2018-06-14 18:47:40 -04:00
|
|
|
audioio_get_buffer_result_t get_buffer_result =
|
|
|
|
audiosample_get_buffer(dma->sample, dma->single_channel, dma->audio_channel,
|
2021-03-15 19:27:36 +05:30
|
|
|
&buffer, &buffer_length);
|
2018-03-12 16:09:13 -07:00
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
DmacDescriptor *descriptor = dma->second_descriptor;
|
2018-03-12 16:09:13 -07:00
|
|
|
if (dma->first_descriptor_free) {
|
|
|
|
descriptor = dma_descriptor(dma->dma_channel);
|
|
|
|
}
|
|
|
|
dma->first_descriptor_free = !dma->first_descriptor_free;
|
|
|
|
|
2018-06-14 18:47:40 -04:00
|
|
|
if (get_buffer_result == GET_BUFFER_ERROR) {
|
|
|
|
audio_dma_stop(dma);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
uint8_t *output_buffer;
|
2018-03-12 16:09:13 -07:00
|
|
|
uint32_t output_buffer_length;
|
|
|
|
uint8_t output_spacing;
|
|
|
|
audio_dma_convert_signed(dma, buffer, buffer_length, &output_buffer, &output_buffer_length,
|
|
|
|
&output_spacing);
|
|
|
|
|
|
|
|
descriptor->BTCNT.reg = output_buffer_length / dma->beat_size / output_spacing;
|
2021-03-15 19:27:36 +05:30
|
|
|
descriptor->SRCADDR.reg = ((uint32_t)output_buffer) + output_buffer_length;
|
2018-06-14 18:47:40 -04:00
|
|
|
if (get_buffer_result == GET_BUFFER_DONE) {
|
2018-03-12 16:09:13 -07:00
|
|
|
if (dma->loop) {
|
|
|
|
audiosample_reset_buffer(dma->sample, dma->single_channel, dma->audio_channel);
|
|
|
|
} else {
|
|
|
|
descriptor->DESCADDR.reg = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
descriptor->BTCTRL.bit.VALID = true;
|
|
|
|
}
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
static void setup_audio_descriptor(DmacDescriptor *descriptor, uint8_t beat_size,
|
|
|
|
uint8_t spacing, uint32_t output_register_address) {
|
2018-03-12 16:09:13 -07:00
|
|
|
uint32_t beat_size_reg = DMAC_BTCTRL_BEATSIZE_BYTE;
|
|
|
|
if (beat_size == 2) {
|
|
|
|
beat_size_reg = DMAC_BTCTRL_BEATSIZE_HWORD;
|
|
|
|
} else if (beat_size == 4) {
|
|
|
|
beat_size_reg = DMAC_BTCTRL_BEATSIZE_WORD;
|
|
|
|
}
|
|
|
|
descriptor->BTCTRL.reg = beat_size_reg |
|
2021-03-15 19:27:36 +05:30
|
|
|
DMAC_BTCTRL_SRCINC |
|
|
|
|
DMAC_BTCTRL_EVOSEL_BLOCK |
|
|
|
|
DMAC_BTCTRL_STEPSIZE(spacing - 1) |
|
|
|
|
DMAC_BTCTRL_STEPSEL_SRC;
|
2018-03-12 16:09:13 -07:00
|
|
|
descriptor->DSTADDR.reg = output_register_address;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Playback should be shutdown before calling this.
|
2021-03-15 19:27:36 +05:30
|
|
|
audio_dma_result audio_dma_setup_playback(audio_dma_t *dma,
|
|
|
|
mp_obj_t sample,
|
|
|
|
bool loop,
|
|
|
|
bool single_channel,
|
|
|
|
uint8_t audio_channel,
|
|
|
|
bool output_signed,
|
|
|
|
uint32_t output_register_address,
|
|
|
|
uint8_t dma_trigger_source) {
|
2021-04-23 09:46:33 -05:00
|
|
|
uint8_t dma_channel = dma_allocate_channel();
|
2018-03-12 16:09:13 -07:00
|
|
|
if (dma_channel >= AUDIO_DMA_CHANNEL_COUNT) {
|
|
|
|
return AUDIO_DMA_DMA_BUSY;
|
|
|
|
}
|
|
|
|
|
|
|
|
dma->sample = sample;
|
|
|
|
dma->loop = loop;
|
|
|
|
dma->single_channel = single_channel;
|
|
|
|
dma->audio_channel = audio_channel;
|
|
|
|
dma->dma_channel = dma_channel;
|
|
|
|
dma->signed_to_unsigned = false;
|
|
|
|
dma->unsigned_to_signed = false;
|
|
|
|
dma->second_descriptor = NULL;
|
|
|
|
dma->spacing = 1;
|
|
|
|
dma->first_descriptor_free = true;
|
|
|
|
audiosample_reset_buffer(sample, single_channel, audio_channel);
|
|
|
|
|
|
|
|
bool single_buffer;
|
|
|
|
bool samples_signed;
|
|
|
|
uint32_t max_buffer_length;
|
|
|
|
audiosample_get_buffer_structure(sample, single_channel, &single_buffer, &samples_signed,
|
2021-03-15 19:27:36 +05:30
|
|
|
&max_buffer_length, &dma->spacing);
|
2018-03-12 16:09:13 -07:00
|
|
|
uint8_t output_spacing = dma->spacing;
|
|
|
|
if (output_signed != samples_signed) {
|
|
|
|
output_spacing = 1;
|
|
|
|
max_buffer_length /= dma->spacing;
|
2021-03-15 19:27:36 +05:30
|
|
|
dma->first_buffer = (uint8_t *)m_realloc(dma->first_buffer, max_buffer_length);
|
2018-03-12 16:09:13 -07:00
|
|
|
if (dma->first_buffer == NULL) {
|
|
|
|
return AUDIO_DMA_MEMORY_ERROR;
|
|
|
|
}
|
|
|
|
dma->first_buffer_free = true;
|
|
|
|
if (!single_buffer) {
|
2021-03-15 19:27:36 +05:30
|
|
|
dma->second_buffer = (uint8_t *)m_realloc(dma->second_buffer, max_buffer_length);
|
2018-03-12 16:09:13 -07:00
|
|
|
if (dma->second_buffer == NULL) {
|
|
|
|
return AUDIO_DMA_MEMORY_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dma->signed_to_unsigned = !output_signed && samples_signed;
|
|
|
|
dma->unsigned_to_signed = output_signed && !samples_signed;
|
|
|
|
}
|
|
|
|
|
|
|
|
dma->event_channel = 0xff;
|
|
|
|
if (!single_buffer) {
|
2021-03-15 19:27:36 +05:30
|
|
|
dma->second_descriptor = (DmacDescriptor *)m_malloc(sizeof(DmacDescriptor), false);
|
2018-03-12 16:09:13 -07:00
|
|
|
if (dma->second_descriptor == NULL) {
|
|
|
|
return AUDIO_DMA_MEMORY_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We're likely double buffering so set up the block interrupts.
|
|
|
|
turn_on_event_system();
|
2021-04-20 13:33:11 -05:00
|
|
|
dma->event_channel = find_sync_event_channel_raise();
|
2018-03-12 16:09:13 -07:00
|
|
|
init_event_channel_interrupt(dma->event_channel, CORE_GCLK, EVSYS_ID_GEN_DMAC_CH_0 + dma_channel);
|
|
|
|
|
|
|
|
// We keep the audio_dma_t for internal use and the sample as a root pointer because it
|
|
|
|
// contains the audiodma structure.
|
|
|
|
audio_dma_state[dma->dma_channel] = dma;
|
|
|
|
MP_STATE_PORT(playing_audio)[dma->dma_channel] = dma->sample;
|
|
|
|
}
|
|
|
|
|
2018-05-02 15:15:25 -07:00
|
|
|
|
2018-03-12 16:09:13 -07:00
|
|
|
if (audiosample_bits_per_sample(sample) == 16) {
|
|
|
|
dma->beat_size = 2;
|
|
|
|
dma->bytes_per_sample = 2;
|
2018-05-02 15:15:25 -07:00
|
|
|
} else {
|
|
|
|
dma->beat_size = 1;
|
|
|
|
dma->bytes_per_sample = 1;
|
|
|
|
if (single_channel) {
|
|
|
|
output_register_address += 1;
|
|
|
|
}
|
2018-03-12 16:09:13 -07:00
|
|
|
}
|
|
|
|
// Transfer both channels at once.
|
|
|
|
if (!single_channel && audiosample_channel_count(sample) == 2) {
|
|
|
|
dma->beat_size *= 2;
|
|
|
|
}
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
#ifdef SAM_D5X_E5X
|
2020-07-07 10:11:29 -05:00
|
|
|
int irq = dma->event_channel < 4 ? EVSYS_0_IRQn + dma->event_channel : EVSYS_4_IRQn;
|
2021-03-15 19:27:36 +05:30
|
|
|
#else
|
2020-07-07 10:11:29 -05:00
|
|
|
int irq = EVSYS_IRQn;
|
2021-03-15 19:27:36 +05:30
|
|
|
#endif
|
2020-07-07 10:11:29 -05:00
|
|
|
|
|
|
|
NVIC_DisableIRQ(irq);
|
|
|
|
NVIC_ClearPendingIRQ(irq);
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
DmacDescriptor *first_descriptor = dma_descriptor(dma_channel);
|
2018-03-12 16:09:13 -07:00
|
|
|
setup_audio_descriptor(first_descriptor, dma->beat_size, output_spacing, output_register_address);
|
|
|
|
if (single_buffer) {
|
|
|
|
first_descriptor->DESCADDR.reg = 0;
|
|
|
|
if (dma->loop) {
|
2021-03-15 19:27:36 +05:30
|
|
|
first_descriptor->DESCADDR.reg = (uint32_t)first_descriptor;
|
2018-03-12 16:09:13 -07:00
|
|
|
}
|
|
|
|
} else {
|
2021-03-15 19:27:36 +05:30
|
|
|
first_descriptor->DESCADDR.reg = (uint32_t)dma->second_descriptor;
|
2018-03-12 16:09:13 -07:00
|
|
|
setup_audio_descriptor(dma->second_descriptor, dma->beat_size, output_spacing, output_register_address);
|
2021-03-15 19:27:36 +05:30
|
|
|
dma->second_descriptor->DESCADDR.reg = (uint32_t)first_descriptor;
|
2018-03-12 16:09:13 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Load the first two blocks up front.
|
|
|
|
audio_dma_load_next_block(dma);
|
|
|
|
if (!single_buffer) {
|
|
|
|
audio_dma_load_next_block(dma);
|
|
|
|
}
|
|
|
|
|
|
|
|
dma_configure(dma_channel, dma_trigger_source, true);
|
2019-08-06 21:42:01 -05:00
|
|
|
audio_dma_enable_channel(dma_channel);
|
2018-03-12 16:09:13 -07:00
|
|
|
|
2020-07-07 10:11:29 -05:00
|
|
|
NVIC_EnableIRQ(irq);
|
|
|
|
|
2018-03-12 16:09:13 -07:00
|
|
|
return AUDIO_DMA_OK;
|
|
|
|
}
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
void audio_dma_stop(audio_dma_t *dma) {
|
2019-08-06 21:59:16 -05:00
|
|
|
uint8_t channel = dma->dma_channel;
|
|
|
|
if (channel < AUDIO_DMA_CHANNEL_COUNT) {
|
|
|
|
audio_dma_disable_channel(channel);
|
|
|
|
disable_event_channel(dma->event_channel);
|
|
|
|
MP_STATE_PORT(playing_audio)[channel] = NULL;
|
|
|
|
audio_dma_state[channel] = NULL;
|
2021-04-23 09:46:33 -05:00
|
|
|
dma_free_channel(dma->dma_channel);
|
2019-08-06 21:59:16 -05:00
|
|
|
}
|
2018-03-12 16:09:13 -07:00
|
|
|
dma->dma_channel = AUDIO_DMA_CHANNEL_COUNT;
|
|
|
|
}
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
void audio_dma_pause(audio_dma_t *dma) {
|
2018-05-07 17:47:29 -07:00
|
|
|
dma_suspend_channel(dma->dma_channel);
|
|
|
|
}
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
void audio_dma_resume(audio_dma_t *dma) {
|
2018-05-07 17:47:29 -07:00
|
|
|
dma_resume_channel(dma->dma_channel);
|
|
|
|
}
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
bool audio_dma_get_paused(audio_dma_t *dma) {
|
2018-05-07 17:47:29 -07:00
|
|
|
if (dma->dma_channel >= AUDIO_DMA_CHANNEL_COUNT) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint32_t status = dma_transfer_status(dma->dma_channel);
|
|
|
|
|
|
|
|
return (status & DMAC_CHINTFLAG_SUSP) != 0;
|
|
|
|
}
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
void audio_dma_init(audio_dma_t *dma) {
|
2018-03-12 16:09:13 -07:00
|
|
|
dma->dma_channel = AUDIO_DMA_CHANNEL_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
void audio_dma_reset(void) {
|
|
|
|
for (uint8_t i = 0; i < AUDIO_DMA_CHANNEL_COUNT; i++) {
|
|
|
|
audio_dma_state[i] = NULL;
|
2018-06-14 18:47:40 -04:00
|
|
|
audio_dma_pending[i] = false;
|
samd: audio_dma: Track channel allocation
Previously, we depended on allocated channels to always be
"dma_channel_enabled". However, (A) sometimes, many operations
would take place between find_free_audio_dma_channel and
audio_dma_enable_channel, and (B) some debugging I did led me to believe
that "dma_channel_enabled" would become false when the hardware ended
a scheduled DMA transaction, but while a CP object would still think it
owned the DMA channel.
((B) is not documented in the datasheet and I am not 100% convinced that
my debugging session was not simply missing where we were disabling the
channel, but in either case, it shows a need to directly track allocated
separately from enabled)
Therefore,
* Add audio_dma_{allocate,free}_channel.
* audio_dma_free_channel implies audio_dma_disable_channel
* track via a new array audio_dma_allocated[]
* clear all allocated flags on soft-reboot
* Convert find_free_audio_dma_channel to audio_dma_allocate_channel
* use audio_dma_allocated[] instead of dma_channel_enabled() to check
availability
* remove find_free_audio_dma_channel
* For each one, find a matching audio_dma_disable_channel to convert
to audio_dma_free_channel
Closes: #2058
2019-08-28 16:55:17 -05:00
|
|
|
audio_dma_allocated[i] = false;
|
2019-08-06 21:42:01 -05:00
|
|
|
audio_dma_disable_channel(i);
|
2018-03-12 16:09:13 -07:00
|
|
|
dma_descriptor(i)->BTCTRL.bit.VALID = false;
|
|
|
|
MP_STATE_PORT(playing_audio)[i] = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
bool audio_dma_get_playing(audio_dma_t *dma) {
|
2018-03-12 16:09:13 -07:00
|
|
|
if (dma->dma_channel >= AUDIO_DMA_CHANNEL_COUNT) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint32_t status = dma_transfer_status(dma->dma_channel);
|
2018-05-07 17:47:29 -07:00
|
|
|
if ((status & DMAC_CHINTFLAG_TCMPL) != 0 || (status & DMAC_CHINTFLAG_TERR) != 0) {
|
2018-03-12 16:09:13 -07:00
|
|
|
audio_dma_stop(dma);
|
|
|
|
}
|
|
|
|
|
2018-05-07 17:47:29 -07:00
|
|
|
return (status & DMAC_CHINTFLAG_TERR) == 0;
|
2018-03-12 16:09:13 -07:00
|
|
|
}
|
|
|
|
|
2020-07-07 10:11:29 -05:00
|
|
|
// WARN(tannewt): DO NOT print from here, or anything it calls. Printing calls
|
|
|
|
// background tasks such as this and causes a stack overflow.
|
|
|
|
STATIC void dma_callback_fun(void *arg) {
|
2021-03-15 19:27:36 +05:30
|
|
|
audio_dma_t *dma = arg;
|
2020-07-07 10:11:29 -05:00
|
|
|
if (dma == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
audio_dma_load_next_block(dma);
|
|
|
|
}
|
2018-06-14 18:47:40 -04:00
|
|
|
|
2020-07-07 10:11:29 -05:00
|
|
|
void evsyshandler_common(void) {
|
2018-03-12 16:09:13 -07:00
|
|
|
for (uint8_t i = 0; i < AUDIO_DMA_CHANNEL_COUNT; i++) {
|
2021-03-15 19:27:36 +05:30
|
|
|
audio_dma_t *dma = audio_dma_state[i];
|
2018-03-12 16:09:13 -07:00
|
|
|
if (dma == NULL) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
bool block_done = event_interrupt_active(dma->event_channel);
|
|
|
|
if (!block_done) {
|
|
|
|
continue;
|
|
|
|
}
|
2021-03-15 19:27:36 +05:30
|
|
|
background_callback_add(&dma->callback, dma_callback_fun, (void *)dma);
|
2018-03-12 16:09:13 -07:00
|
|
|
}
|
|
|
|
}
|
2020-07-07 10:11:29 -05:00
|
|
|
|
|
|
|
#ifdef SAM_D5X_E5X
|
2021-03-15 19:27:36 +05:30
|
|
|
void EVSYS_0_Handler(void) {
|
|
|
|
evsyshandler_common();
|
|
|
|
}
|
|
|
|
void EVSYS_1_Handler(void) {
|
|
|
|
evsyshandler_common();
|
|
|
|
}
|
|
|
|
void EVSYS_2_Handler(void) {
|
|
|
|
evsyshandler_common();
|
|
|
|
}
|
|
|
|
void EVSYS_3_Handler(void) {
|
|
|
|
evsyshandler_common();
|
|
|
|
}
|
|
|
|
void EVSYS_4_Handler(void) {
|
|
|
|
evsyshandler_common();
|
|
|
|
}
|
2020-07-07 10:11:29 -05:00
|
|
|
#else
|
2021-03-15 19:27:36 +05:30
|
|
|
void EVSYS_Handler(void) {
|
|
|
|
evsyshandler_common();
|
|
|
|
}
|
2020-07-07 10:11:29 -05:00
|
|
|
#endif
|
|
|
|
|
2019-08-07 21:29:24 -05:00
|
|
|
#endif
|