1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/drivers/dfplayer/dfplayer_internal.c
2020-11-05 11:32:41 +01:00

458 lines
14 KiB
C

/*
* Copyright 2019 Marian Buschsieweke
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @ingroup drivers_dfplayer
* @{
*
* @file
* @brief Implementation DFPlayer Mini Device Driver
* @author Marian Buschsieweke <marian.buschsieweke@ovgu.de>
*
* @}
*/
#include <assert.h>
#include <errno.h>
#include <inttypes.h>
#include <stdint.h>
#include <string.h>
#include "dfplayer.h"
#include "dfplayer_constants.h"
#include "dfplayer_internal.h"
#include "periph/gpio.h"
#include "periph/uart.h"
#include "thread.h"
#include "xtimer.h"
#define ENABLE_DEBUG 0
#include "debug.h"
/**
* @brief Initial value of the frame check sequence
*/
static const uint16_t fcs_init = -(DFPLAYER_VERSION + DFPLAYER_LEN);
/**
* @brief Identify the source from an insert/eject event
*
* @param dev DFPlayer device descriptor
*
* @return The source that was inserted / ejected
* @retval DFPLAYER_SOURCE_NUMOF Unknown source
*/
static dfplayer_source_t _get_inserted_ejected_source(dfplayer_t *dev)
{
switch (dev->buf[3]) {
case DFPLAYER_DEVICE_USB:
DEBUG("[dfplayer] Inserted/ejected USB storage device\n");
return DFPLAYER_SOURCE_USB;
case DFPLAYER_DEVICE_SDCARD:
DEBUG("[dfplayer] Inserted/ejected SD card\n");
return DFPLAYER_SOURCE_SDCARD;
}
DEBUG("[dfplayer] Insert/eject event with unknown source\n");
return DFPLAYER_SOURCE_NUMOF;
}
/**
* @brief Handle a playback completed event
*
* @param dev DFPlayer device descriptor
* @param src Medium the track was played from
*/
static void _handle_playback_completed(dfplayer_t *dev, dfplayer_source_t src)
{
uint16_t track = (((uint16_t)dev->buf[2]) << 8) | dev->buf[3];
DEBUG("[dfplayer] Playback of track %" PRIu16 " on medium %u completed\n",
track, (unsigned)src);
dev->flags |= DFPLAYER_FLAG_NO_ACK_BUG;
/* Note: At least some revisions report playback completed more than once,
* maybe to increase probability of the message reaching the MCU. This
* de-duplicates the message by ignoring follow up messages for 100ms.
* Filtering by track number and medium wouldn't work here, as the same
* song might be played in repeat mode.
*/
uint32_t now_us = xtimer_now_usec();
if (dev->cb_done && (now_us - dev->last_event_us > DFPLAYER_TIMEOUT_MS * US_PER_MS)) {
dev->cb_done(dev, src, track);
}
dev->last_event_us = now_us;
}
/**
* @brief Parse the bootup completed frame and init available sources
*
* @param dev DFPlayer device descriptor
*/
static void _handle_bootup_completed(dfplayer_t *dev)
{
if (dev->buf[3] & DFPLAYER_MASK_USB) {
dev->srcs |= 0x01 << DFPLAYER_SOURCE_USB;
}
if (dev->buf[3] & DFPLAYER_MASK_SDCARD) {
dev->srcs |= 0x01 << DFPLAYER_SOURCE_SDCARD;
}
if (dev->buf[3] & DFPLAYER_MASK_FLASH) {
dev->srcs |= 0x01 << DFPLAYER_SOURCE_FLASH;
}
/* Unblock caller of dfplayer_reset() */
mutex_unlock(&dev->sync);
}
/**
* @brief Handle a notification message
*/
static void _handle_event_notification(dfplayer_t *dev)
{
switch (dev->buf[0]) {
case DFPLAYER_NOTIFY_INSERT:
DEBUG("[dfplayer] Insert event\n");
{
dfplayer_source_t src = _get_inserted_ejected_source(dev);
if (src < DFPLAYER_SOURCE_NUMOF) {
dev->srcs |= (dfplayer_source_set_t)(0x01 << src);
}
}
if (dev->cb_src) {
dev->cb_src(dev, dev->srcs);
}
return;
case DFPLAYER_NOTIFY_EJECT:
DEBUG("[dfplayer] Eject event\n");
{
dfplayer_source_t src = _get_inserted_ejected_source(dev);
if (src < DFPLAYER_SOURCE_NUMOF) {
dev->srcs &= ~((dfplayer_source_set_t)(0x01 << src));
}
}
if (dev->cb_src) {
dev->cb_src(dev, dev->srcs);
}
return;
case DFPLAYER_NOTIFY_DONE_USB:
_handle_playback_completed(dev, DFPLAYER_SOURCE_USB);
return;
case DFPLAYER_NOTIFY_DONE_SDCARD:
_handle_playback_completed(dev, DFPLAYER_SOURCE_SDCARD);
return;
case DFPLAYER_NOTIFY_DONE_FLASH:
_handle_playback_completed(dev, DFPLAYER_SOURCE_FLASH);
return;
case DFPLAYER_NOTIFY_READY:
_handle_bootup_completed(dev);
return;
default:
DEBUG("[dfplayer] Unknown notification (%02x)\n", dev->buf[0]);
}
}
/**
* @brief Parse the frame received from the DFPlayer Mini
*
* @param dev Device descriptor of the DFPlayer the frame received from
*
* The frame is stored in the buffer of the device descriptor
*/
static void _parse_frame(dfplayer_t *dev)
{
assert(dev->len == DFPLAYER_LEN);
switch (dev->buf[0] & DFPLAYER_CLASS_MASK) {
case DFPLAYER_CLASS_NOTIFY:
_handle_event_notification(dev);
return;
case DFPLAYER_CLASS_RESPONSE:
/* Unblock thread waiting for response */
mutex_unlock(&dev->sync);
return;
}
DEBUG("[dfplayer] Got frame of unknown class\n");
}
/**
* @brief Function called when a byte was received over UART (ISR-context)
*
* @param _dev The corresponding device descriptor
* @param data The received byte of data
*/
void dfplayer_uart_rx_cb(void *_dev, uint8_t data)
{
dfplayer_t *dev = _dev;
switch (dev->state) {
case DFPLAYER_RX_STATE_START:
if (data == DFPLAYER_START) {
dev->state = DFPLAYER_RX_STATE_VERSION;
return;
}
break;
case DFPLAYER_RX_STATE_VERSION:
if (data == DFPLAYER_VERSION) {
dev->state = DFPLAYER_RX_STATE_LENGTH;
return;
}
break;
case DFPLAYER_RX_STATE_LENGTH:
if (data == DFPLAYER_LEN) {
dev->len = 0;
dev->state = DFPLAYER_RX_STATE_DATA;
return;
}
else {
DEBUG("[dfplayer] Got frame with length %" PRIu8 ", but all "
"frames should have length 6\n", data);
}
break;
case DFPLAYER_RX_STATE_DATA:
/* We are a bit more liberal here and allow the end symbol to
* appear in the payload of the frame, as the data sheet does not
* mention any sort of escaping to prevent it from appearing in the
* frame's payload. If bytes get lost and an and of frame symbol
* is mistaken for a payload byte, this will be almost certainly
* detected, as additionally a second end of frame symbol would
* need to appear at the right position *and* the frame check
* sequence need to match
*/
if ((data == DFPLAYER_END) && (dev->len == DFPLAYER_LEN)) {
uint16_t fcs_exp = fcs_init;
fcs_exp -= dev->buf[0] + dev->buf[1] + dev->buf[2] + dev->buf[3];
uint16_t fcs = (((uint16_t)dev->buf[4]) << 8) | dev->buf[5];
if (fcs == fcs_exp) {
DEBUG("[dfplayer] Got 0x%02x, 0x%02x, 0x%02x, 0x%02x\n",
dev->buf[0], dev->buf[1], dev->buf[2],
dev->buf[3]);
_parse_frame(dev);
}
else {
DEBUG("[dfplayer] Checksum mismatch");
}
}
else if (dev->len < sizeof(dev->buf)) {
dev->buf[dev->len++] = data;
return;
}
else {
DEBUG("[dfplayer] Frame overflown\n");
}
break;
default:
break;
}
dev->state = DFPLAYER_RX_STATE_START;
}
static int _send(dfplayer_t *dev, uint8_t cmd, uint8_t p1, uint8_t p2,
uint32_t timeout_us)
{
int retval;
if (dev->flags & DFPLAYER_FLAG_NO_ACK_BUG) {
/* Hardware bug: The next command will not be ack'ed, unless it is
* a query command. We can clear the flag, as we issue now a fake query,
* if needed.
*/
dev->flags &= ~(DFPLAYER_FLAG_NO_ACK_BUG);
if (cmd < DFPLAYER_LOWEST_QUERY) {
/* Command is a control command, we query the volume and ignore the
* result as work around */
retval = _send(dev, DFPLAYER_CMD_GET_VOLUME, 0, 0,
DFPLAYER_TIMEOUT_MS * US_PER_MS);
if (retval) {
/* pass through error */
return retval;
}
}
}
uint16_t fcs = fcs_init - (cmd + DFPLAYER_ACK + p1 + p2);
uint8_t frame[] = {
DFPLAYER_START, DFPLAYER_VERSION, DFPLAYER_LEN, cmd, DFPLAYER_ACK,
p1, p2, (uint8_t)(fcs>>8), (uint8_t)fcs, DFPLAYER_END
};
for (unsigned i = 0; i < DFPLAYER_RETRIES; i++) {
retval = 0;
DEBUG("[dfplayer] About to exchange frame\n");
/* Enforce that mutex is locked, so that xtimer_mutex_lock_timeout()
* will not return immediately. */
mutex_trylock(&dev->sync);
uart_write(dev->uart, frame, sizeof(frame));
if (xtimer_mutex_lock_timeout(&dev->sync, timeout_us)) {
DEBUG("[dfplayer] Response timed out\n");
retval = -ETIMEDOUT;
}
else {
uint8_t code = dev->buf[0];
if (code == DFPLAYER_RESPONSE_ERROR) {
switch (dev->buf[3]) {
case DFPLAYER_ERROR_BUSY:
DEBUG("[dfplayer] Error: Module is busy\n");
retval = -EAGAIN;
break;
case DFPLAYER_ERROR_FRAME:
DEBUG("[dfplayer] Error: DFPlayer received incomplete "
"frame\n");
retval = -EIO;
break;
case DFPLAYER_ERROR_FCS:
DEBUG("[dfplayer] Error: DFPlayer received corrupt frame "
"(FCS mismatch)\n");
retval = -EIO;
break;
default:
DEBUG("[dfplayer] Unknown error!\n");
/* This should never be reached according the datasheet */
retval = -EIO;
break;
}
}
}
/* wait to work around HW bug */
xtimer_msleep(DFPLAYER_SEND_DELAY_MS);
if (!retval) {
break;
}
}
return retval;
}
int dfplayer_transceive(dfplayer_t *dev, uint16_t *resp,
uint8_t cmd, uint8_t p1, uint8_t p2)
{
if (!dev) {
return -EINVAL;
}
mutex_lock(&dev->mutex);
int retval = _send(dev, cmd, p1, p2, DFPLAYER_TIMEOUT_MS * US_PER_MS);
if (retval) {
mutex_unlock(&dev->mutex);
return retval;
}
if (resp) {
*resp = (((uint16_t)dev->buf[2]) << 8) | (uint16_t)dev->buf[3];
}
mutex_unlock(&dev->mutex);
return 0;
}
int dfplayer_reset(dfplayer_t *dev)
{
if (!dev) {
return -EINVAL;
}
mutex_lock(&dev->mutex);
int retval = _send(dev, DFPLAYER_CMD_RESET, 0, 0,
DFPLAYER_TIMEOUT_MS * US_PER_MS);
if (retval) {
mutex_unlock(&dev->mutex);
return retval;
}
/* Enforce that mutex is locked, so that xtimer_mutex_lock_timeout()
* will not return immediately. */
mutex_trylock(&dev->sync);
const uint32_t bootup_timeout = DFPLAYER_BOOTUP_TIME_MS * US_PER_MS;
if (xtimer_mutex_lock_timeout(&dev->sync, bootup_timeout)) {
mutex_unlock(&dev->mutex);
DEBUG("[dfplayer] Waiting for device to boot timed out\n");
return -ETIMEDOUT;
}
uint8_t code = dev->buf[0];
mutex_unlock(&dev->mutex);
if (code != DFPLAYER_NOTIFY_READY) {
DEBUG("[dfplayer] Got unexpected response after reset\n");
return -EIO;
}
return 0;
}
int dfplayer_file_cmd(dfplayer_t *dev, uint8_t cmd, uint8_t p1, uint8_t p2)
{
int retval = _send(dev, cmd, p1, p2, DFPLAYER_TIMEOUT_MS * US_PER_MS);
if (retval) {
return retval;
}
/* Enforce that mutex is locked, so that xtimer_mutex_lock_timeout()
* will not return immediately. */
mutex_trylock(&dev->sync);
const uint32_t timeout_us = DFPLAYER_TIMEOUT_MS * US_PER_MS;
if (xtimer_mutex_lock_timeout(&dev->sync, timeout_us)) {
/* For commands DFPLAYER_CMD_PLAY_FROM_MP3 (0x12) and
* DFPLAYER_CMD_PLAY_ADVERT (0x13) a second reply is only generated on
* failure. A timeout could be either:
* a) Success. DFPlayer is playing the selected file
* or
* b) Failure, but the reply got lost (or was rejected due to mismatch
* of the frame check sequence)
*
* We just check if the DFPlayer is actually playing
*/
if (gpio_is_valid(dev->busy_pin)) {
/* Using BUSY pin to check if device is playing */
if (gpio_read(dev->busy_pin)) {
/* Device not playing, file does not exist */
retval = -ENOENT;
}
retval = 0;
}
else {
/* BUSY pin not connected, query status instead */
retval = _send(dev, DFPLAYER_CMD_GET_STATUS, 0, 0, timeout_us);
if (!retval) {
uint8_t status = dev->buf[3];
retval = (status & DFPLAYER_STATUS_PLAYING) ? 0 : -ENOENT;
}
}
}
uint8_t code = dev->buf[0];
uint8_t error = dev->buf[3];
if (retval) {
return retval;
}
if (code == DFPLAYER_RESPONSE_ERROR) {
/* The DFPlayer already acknowledged successful reception of the
* command, so we expect that the only cause for an error is that the
* file was not found. But better check anyway, the device is strange */
if (error == DFPLAYER_ERROR_NO_SUCH_FILE) {
return -ENOENT;
}
DEBUG("[dfplayer] Got unexpected error message\n");
return -EIO;
}
return 0;
}