audio/pcm_decode.c: skip extra chunk of wav header
Summary: - The wav header parser in /dev/audio/pcm device driver expects the 'data' chunk is placed just after the 'fmt ' chunk. - Because the wav files generated by FFmpeg places 'LIST' chunk which contains the music track information between 'fmt ' and 'data' chunks, nxplayer cannot playback the files. - This patch skips extra chunks after 'fmt ' chunk to find the 'data' chunk. Impact: - All architectures which support /dev/audio/pcm device. Testing: - Tested by Raspberry Pi Pico audio driver. - nxplayer can playback the wav files which are created by FFmpeg after applying this patch.
This commit is contained in:
parent
7f307f9765
commit
9061c997fc
1 changed files with 80 additions and 48 deletions
|
@ -124,17 +124,10 @@ static void pcm_dump(FAR const struct wav_header_s *wav);
|
||||||
# define pcm_dump(w)
|
# define pcm_dump(w)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_ENDIAN_BIG
|
|
||||||
static uint16_t pcm_leuint16(uint16_t value);
|
|
||||||
static uint16_t pcm_leuint32(uint32_t value);
|
|
||||||
#else
|
|
||||||
# define pcm_leuint16(v) (v)
|
|
||||||
# define pcm_leuint32(v) (v)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef CONFIG_AUDIO_FORMAT_RAW
|
#ifndef CONFIG_AUDIO_FORMAT_RAW
|
||||||
static inline bool pcm_validwav(FAR const struct wav_header_s *wav);
|
static inline bool pcm_validwav(FAR const struct wav_header_s *wav);
|
||||||
static bool pcm_parsewav(FAR struct pcm_decode_s *priv, uint8_t *data);
|
static ssize_t pcm_parsewav(FAR struct pcm_decode_s *priv, uint8_t *data,
|
||||||
|
apb_samp_t len);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef CONFIG_AUDIO_EXCLUDE_FFORWARD
|
#ifndef CONFIG_AUDIO_EXCLUDE_FFORWARD
|
||||||
|
@ -270,37 +263,37 @@ static void pcm_dump(FAR const struct wav_header_s *wav)
|
||||||
* Name: pcm_leuint16
|
* Name: pcm_leuint16
|
||||||
*
|
*
|
||||||
* Description:
|
* Description:
|
||||||
* Get a 16-bit value stored in little endian order for a big-endian
|
* Get a 16-bit value stored in little endian order. Unaligned address is
|
||||||
* machine.
|
* acceptable.
|
||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
#ifdef CONFIG_ENDIAN_BIG
|
static uint16_t pcm_leuint16(FAR const uint16_t *ptr)
|
||||||
static uint16_t pcm_leuint16(uint16_t value)
|
|
||||||
{
|
{
|
||||||
return (((value & 0x00ff) << 8) |
|
FAR const uint8_t *p = (FAR const uint8_t *)ptr;
|
||||||
((value >> 8) & 0x00ff));
|
|
||||||
|
return ((p[0] << 0) |
|
||||||
|
(p[1] << 8));
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Name: pcm_leuint16
|
* Name: pcm_leuint32
|
||||||
*
|
*
|
||||||
* Description:
|
* Description:
|
||||||
* Get a 16-bit value stored in little endian order for a big-endian
|
* Get a 32-bit value stored in little endian order. Unaligned address is
|
||||||
* machine.
|
* acceptable.
|
||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
#ifdef CONFIG_ENDIAN_BIG
|
static uint32_t pcm_leuint32(FAR const uint32_t *ptr)
|
||||||
static uint16_t pcm_leuint32(uint32_t value)
|
|
||||||
{
|
{
|
||||||
return (((value & 0x000000ff) << 24) |
|
FAR const uint8_t *p = (FAR const uint8_t *)ptr;
|
||||||
((value & 0x0000ff00) << 8) |
|
|
||||||
((value & 0x00ff0000) >> 8) |
|
return ((p[0] << 0) |
|
||||||
((value & 0xff000000) >> 24));
|
(p[1] << 8) |
|
||||||
|
(p[2] << 16) |
|
||||||
|
(p[3] << 24));
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Name: pcm_validwav
|
* Name: pcm_validwav
|
||||||
|
@ -319,8 +312,7 @@ static inline bool pcm_validwav(FAR const struct wav_header_s *wav)
|
||||||
wav->fmt.format == WAV_FMT_FORMAT &&
|
wav->fmt.format == WAV_FMT_FORMAT &&
|
||||||
wav->fmt.nchannels < 256 &&
|
wav->fmt.nchannels < 256 &&
|
||||||
wav->fmt.align < 256 &&
|
wav->fmt.align < 256 &&
|
||||||
wav->fmt.bpsamp < 256 &&
|
wav->fmt.bpsamp < 256);
|
||||||
wav->data.chunkid == WAV_DATA_CHUNKID);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
|
@ -333,31 +325,67 @@ static inline bool pcm_validwav(FAR const struct wav_header_s *wav)
|
||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
static bool pcm_parsewav(FAR struct pcm_decode_s *priv, uint8_t *data)
|
static ssize_t pcm_parsewav(FAR struct pcm_decode_s *priv, uint8_t *data,
|
||||||
|
apb_samp_t len)
|
||||||
{
|
{
|
||||||
FAR const struct wav_header_s *wav = (FAR const struct wav_header_s *)data;
|
FAR const struct wav_header_s *wav = (FAR const struct wav_header_s *)data;
|
||||||
|
FAR const struct wav_datachunk_s *dchunk;
|
||||||
struct wav_header_s localwav;
|
struct wav_header_s localwav;
|
||||||
bool ret;
|
size_t ret = sizeof(struct wav_header_s);
|
||||||
|
|
||||||
|
if (len < sizeof(struct wav_header_s))
|
||||||
|
{
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
/* Transfer the purported WAV file header into our stack storage,
|
/* Transfer the purported WAV file header into our stack storage,
|
||||||
* correcting for endian issues as needed.
|
* correcting for endian issues as needed.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
localwav.hdr.chunkid = pcm_leuint32(wav->hdr.chunkid);
|
localwav.hdr.chunkid = pcm_leuint32(&wav->hdr.chunkid);
|
||||||
localwav.hdr.chunklen = pcm_leuint32(wav->hdr.chunklen);
|
localwav.hdr.chunklen = pcm_leuint32(&wav->hdr.chunklen);
|
||||||
localwav.hdr.format = pcm_leuint32(wav->hdr.format);
|
localwav.hdr.format = pcm_leuint32(&wav->hdr.format);
|
||||||
|
|
||||||
localwav.fmt.chunkid = pcm_leuint32(wav->fmt.chunkid);
|
localwav.fmt.chunkid = pcm_leuint32(&wav->fmt.chunkid);
|
||||||
localwav.fmt.chunklen = pcm_leuint32(wav->fmt.chunklen);
|
localwav.fmt.chunklen = pcm_leuint32(&wav->fmt.chunklen);
|
||||||
localwav.fmt.format = pcm_leuint16(wav->fmt.format);
|
localwav.fmt.format = pcm_leuint16(&wav->fmt.format);
|
||||||
localwav.fmt.nchannels = pcm_leuint16(wav->fmt.nchannels);
|
localwav.fmt.nchannels = pcm_leuint16(&wav->fmt.nchannels);
|
||||||
localwav.fmt.samprate = pcm_leuint32(wav->fmt.samprate);
|
localwav.fmt.samprate = pcm_leuint32(&wav->fmt.samprate);
|
||||||
localwav.fmt.byterate = pcm_leuint32(wav->fmt.byterate);
|
localwav.fmt.byterate = pcm_leuint32(&wav->fmt.byterate);
|
||||||
localwav.fmt.align = pcm_leuint16(wav->fmt.align);
|
localwav.fmt.align = pcm_leuint16(&wav->fmt.align);
|
||||||
localwav.fmt.bpsamp = pcm_leuint16(wav->fmt.bpsamp);
|
localwav.fmt.bpsamp = pcm_leuint16(&wav->fmt.bpsamp);
|
||||||
|
|
||||||
localwav.data.chunkid = pcm_leuint32(wav->data.chunkid);
|
/* Find the data chunk */
|
||||||
localwav.data.chunklen = pcm_leuint32(wav->data.chunklen);
|
|
||||||
|
dchunk = &wav->data;
|
||||||
|
|
||||||
|
for (; ; )
|
||||||
|
{
|
||||||
|
/* NOTE: The data chunk is possible to be not word-aligned if extra
|
||||||
|
* chunks exist before it.
|
||||||
|
*/
|
||||||
|
|
||||||
|
localwav.data.chunkid = pcm_leuint32(&dchunk->chunkid);
|
||||||
|
localwav.data.chunklen = pcm_leuint32(&dchunk->chunklen);
|
||||||
|
|
||||||
|
if (localwav.data.chunkid == WAV_DATA_CHUNKID)
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Not data chunk. Skip it. */
|
||||||
|
|
||||||
|
ret += localwav.data.chunklen + 8;
|
||||||
|
if (ret >= len)
|
||||||
|
{
|
||||||
|
/* Data chunk not found */
|
||||||
|
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
dchunk = (FAR const struct wav_datachunk_s *)
|
||||||
|
((uintptr_t)dchunk + localwav.data.chunklen + 8);
|
||||||
|
}
|
||||||
|
|
||||||
/* Dump the converted wave header information */
|
/* Dump the converted wave header information */
|
||||||
|
|
||||||
|
@ -365,8 +393,11 @@ static bool pcm_parsewav(FAR struct pcm_decode_s *priv, uint8_t *data)
|
||||||
|
|
||||||
/* Check if the file is a valid PCM WAV header */
|
/* Check if the file is a valid PCM WAV header */
|
||||||
|
|
||||||
ret = pcm_validwav(&localwav);
|
if (!pcm_validwav(&localwav))
|
||||||
if (ret)
|
{
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
/* Yes... pick off the relevant format values and save then in the
|
/* Yes... pick off the relevant format values and save then in the
|
||||||
* device structure.
|
* device structure.
|
||||||
|
@ -1017,6 +1048,7 @@ static int pcm_enqueuebuffer(FAR struct audio_lowerhalf_s *dev,
|
||||||
FAR struct pcm_decode_s *priv = (FAR struct pcm_decode_s *)dev;
|
FAR struct pcm_decode_s *priv = (FAR struct pcm_decode_s *)dev;
|
||||||
FAR struct audio_lowerhalf_s *lower;
|
FAR struct audio_lowerhalf_s *lower;
|
||||||
apb_samp_t bytesleft;
|
apb_samp_t bytesleft;
|
||||||
|
ssize_t headersize;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
DEBUGASSERT(priv);
|
DEBUGASSERT(priv);
|
||||||
|
@ -1070,8 +1102,8 @@ static int pcm_enqueuebuffer(FAR struct audio_lowerhalf_s *dev,
|
||||||
/* Parse and verify the candidate PCM WAV file header */
|
/* Parse and verify the candidate PCM WAV file header */
|
||||||
|
|
||||||
#ifndef CONFIG_AUDIO_FORMAT_RAW
|
#ifndef CONFIG_AUDIO_FORMAT_RAW
|
||||||
if (bytesleft >= sizeof(struct wav_header_s) &&
|
headersize = pcm_parsewav(priv, &apb->samp[apb->curbyte], bytesleft);
|
||||||
pcm_parsewav(priv, &apb->samp[apb->curbyte]))
|
if (headersize > 0)
|
||||||
{
|
{
|
||||||
struct audio_caps_s caps;
|
struct audio_caps_s caps;
|
||||||
|
|
||||||
|
@ -1101,7 +1133,7 @@ static int pcm_enqueuebuffer(FAR struct audio_lowerhalf_s *dev,
|
||||||
|
|
||||||
/* Bump up the data offset */
|
/* Bump up the data offset */
|
||||||
|
|
||||||
apb->curbyte += sizeof(struct wav_header_s);
|
apb->curbyte += headersize;
|
||||||
#endif
|
#endif
|
||||||
#ifndef CONFIG_AUDIO_EXCLUDE_FFORWARD
|
#ifndef CONFIG_AUDIO_EXCLUDE_FFORWARD
|
||||||
audinfo("Begin streaming: apb=%p curbyte=%d nbytes=%d\n",
|
audinfo("Begin streaming: apb=%p curbyte=%d nbytes=%d\n",
|
||||||
|
|
Loading…
Reference in a new issue