diff decoders/aiff.c @ 34:938ef560c7bf

Initial add. Thanks, Torbj�rn!
author Ryan C. Gordon <icculus@icculus.org>
date Thu, 20 Sep 2001 07:51:42 +0000
parents
children 0d5ff5679523
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/decoders/aiff.c	Thu Sep 20 07:51:42 2001 +0000
@@ -0,0 +1,368 @@
+/*
+ * SDL_sound -- An abstract sound format decoding API.
+ * Copyright (C) 2001  Ryan C. Gordon.
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ */
+
+/*
+ * AIFF decoder for SDL_sound
+ *
+ * [Insert something profound about the AIFF file format here.]
+ *
+ * This code was ripped from a decoder I had written for SDL_mixer, which was
+ * based on SDL_mixer's old AIFF music loader. (This loader was unfortunately
+ * completely broken, but it was still useful because all the pieces were
+ * still there, so to speak.)
+ *
+ * When rewriting it for SDL_sound, I changed its structure to be more like
+ * the WAV loader Ryan wrote. Had they not both been part of the same project
+ * it would have been embarrassing how similar they are.
+ *
+ * It is not the most feature-complete AIFF loader the world has ever seen.
+ * For instance, it only makes a token attempt at implementing the AIFF-C
+ * standard; basically the parts of it that I can easily understand and test.
+ * It's a start, though.
+ *
+ * Please see the file LICENSE in the source's root directory.
+ *
+ *  This file was written by Torbjörn Andersson. (d91tan@Update.UU.SE)
+ */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <assert.h>
+
+#include "SDL.h"
+#include "SDL_endian.h"
+#include "SDL_sound.h"
+
+#define __SDL_SOUND_INTERNAL__
+#include "SDL_sound_internal.h"
+
+#if (!defined SOUND_SUPPORTS_AIFF)
+#error SOUND_SUPPORTS_AIFF must be defined.
+#endif
+
+
+static int AIFF_open(Sound_Sample *sample, const char *ext);
+static void AIFF_close(Sound_Sample *sample);
+static Uint32 AIFF_read(Sound_Sample *sample);
+
+const Sound_DecoderFunctions __Sound_DecoderFunctions_AIFF =
+{
+    {
+        "AIFF",
+        "Audio Interchange File Format",
+        "Torbjörn Andersson <d91tan@Update.UU.SE>",
+        "http://www.icculus.org/SDL_sound/"
+    },
+
+    AIFF_open,      /* open() method       */
+    AIFF_close,     /* close() method       */
+    AIFF_read       /* read() method       */
+};
+
+
+    /* this is what we store in our internal->decoder_private field... */
+typedef struct {
+    Sint32 bytesLeft;
+} aiff_t;
+
+
+    /* Chunk management code... */
+
+#define formID 0x4d524f46  /* "FORM", in ascii. */
+#define aiffID 0x46464941  /* "AIFF", in ascii. */
+#define aifcID 0x43464941  /* "AIFC", in ascii. */
+#define ssndID 0x444e5353  /* "SSND", in ascii. */
+#define commID 0x4d4d4f43  /* "COMM", in ascii. */
+
+#define noneID 0x454e4f4e  /* "NONE", in ascii. */
+
+typedef struct
+{
+    Uint32 ckID;
+    Uint32 ckDataSize;
+    Uint16 numChannels;
+    Uint32 numSampleFrames;
+    Uint16 sampleSize;
+    Uint32 sampleRate;
+        /*
+         * We don't handle AIFF-C compressed audio yet, but for those
+         * interested the allowed compression types are supposed to be
+         *
+         *   compressionType   compressionName   meaning
+         *   ---------------------------------------------------------------
+         *   'NONE'            "not compressed"  uncompressed, that is,
+         *                                        straight digitized samples
+         *   'ACE2'            "ACE 2-to-1"      2-to-1 IIGS ACE (Audio
+         *                                        Compression / Expansion)
+         *   'ACE8'            "ACE 8-to-3"      8-to-3 IIGS ACE (Audio
+         *                                        Compression / Expansion)
+         *   'MAC3'            "MACE 3-to-1"     3-to-1 Macintosh Audio
+         *                                        Compression / Expansion
+         *   'MAC6'            "MACE 6-to-1"     6-to-1 Macintosh Audio
+         *                                        Compression / Expansion
+         *
+         * A pstring is a "Pascal-style string", that is, "one byte followed
+         * by test bytes followed when needed by one pad byte. The total
+         * number of bytes in a pstring must be even. The pad byte is
+         * included when the number of text bytes is even, so the total of
+         * text bytes + one count byte + one pad byte will be even. This pad
+         * byte is not reflected in the count."
+         *
+         * As for how these compression algorithms work, your guess is as
+         * good as mine.
+         */
+    Uint32 compressionType;
+#if 0
+    pstring compressionName;
+#endif
+} comm_t;
+
+
+/*
+ * Sample rate is encoded as an "80 bit IEEE Standard 754 floating point
+ * number (Standard Apple Numeric Environment [SANE] data type Extended)".
+ * Whose bright idea was that?
+ *
+ * This function was adapted from libsndfile, and while I do know a little
+ * bit about the IEEE floating point standard I don't pretend to fully
+ * understand this.
+ */
+
+static Uint32 SANE_to_Uint32 (Uint8 *sanebuf)
+{
+    /* Is the frequency outside of what we can represent with Uint32? */
+    if ( (sanebuf[0] & 0x80)
+      || (sanebuf[0] <= 0x3F)
+      || (sanebuf[0] > 0x40)
+      || (sanebuf[0] == 0x40 && sanebuf[1] > 0x1C) )
+        return 0;
+
+    return ((sanebuf[2] << 23) | (sanebuf[3] << 15) | (sanebuf[4] << 7)
+        | (sanebuf[5] >> 1)) >> (29 - sanebuf[1]);
+} /* SANE_to_Uint32 */
+
+
+/*
+ * Read in a comm_t from disk. This makes this process safe regardless of
+ *  the processor's byte order or how the comm_t structure is packed.
+ */
+
+static int read_comm_chunk(SDL_RWops *rw, comm_t *comm)
+{
+    Uint8 sampleRate[10];
+
+    /* skip reading the chunk ID, since it was already read at this point... */
+    comm->ckID = commID;
+
+    if (SDL_RWread(rw, &comm->ckDataSize, sizeof (comm->ckDataSize), 1) != 1)
+        return(0);
+    comm->ckDataSize = SDL_SwapBE32(comm->ckDataSize);
+
+    if (SDL_RWread(rw, &comm->numChannels, sizeof (comm->numChannels), 1) != 1)
+        return(0);
+    comm->numChannels = SDL_SwapBE16(comm->numChannels);
+
+    if (SDL_RWread(rw, &comm->numSampleFrames, sizeof (comm->numSampleFrames), 1) != 1)
+        return(0);
+    comm->numSampleFrames = SDL_SwapBE32(comm->numSampleFrames);
+
+    if (SDL_RWread(rw, &comm->sampleSize, sizeof (comm->sampleSize), 1) != 1)
+        return(0);
+    comm->sampleSize = SDL_SwapBE16(comm->sampleSize);
+
+    if (SDL_RWread(rw, sampleRate, sizeof(sampleRate), 1) != 1)
+        return(0);
+    comm->sampleRate = SANE_to_Uint32(sampleRate);
+
+    if (comm->ckDataSize > sizeof(comm->numChannels)
+                         + sizeof(comm->numSampleFrames)
+                         + sizeof(comm->sampleSize)
+                         + sizeof(sampleRate))
+    {
+        if (SDL_RWread(rw, &comm->compressionType, sizeof (comm->compressionType), 1) != 1)
+            return(0);
+        comm->compressionType = SDL_SwapBE32(comm->compressionType);
+    } /* if */
+    else
+        comm->compressionType = noneID;
+
+    return(1);
+} /* read_comm_chunk */
+
+typedef struct
+{
+    Uint32 ckID;
+    Uint32 ckDataSize;
+    Uint32 offset;
+    Uint32 blockSize;
+    /*
+     * Then, comm->numSampleFrames sample frames. (It's better to get the
+     * length from numSampleFrames than from ckDataSize.)
+     */
+} ssnd_t;
+
+
+static int read_ssnd_chunk(SDL_RWops *rw, ssnd_t *ssnd)
+{
+    /* skip reading the chunk ID, since it was already read at this point... */
+    ssnd->ckID = ssndID;
+
+    if (SDL_RWread(rw, &ssnd->ckDataSize, sizeof (ssnd->ckDataSize), 1) != 1)
+        return(0);
+    ssnd->ckDataSize = SDL_SwapBE32(ssnd->ckDataSize);
+
+    if (SDL_RWread(rw, &ssnd->offset, sizeof(ssnd->offset), 1) != 1)
+        return(0);
+    ssnd->offset = SDL_SwapBE32(ssnd->offset);
+
+    if (SDL_RWread(rw, &ssnd->blockSize, sizeof(ssnd->blockSize), 1) != 1)
+        return(0);
+    ssnd->blockSize = SDL_SwapBE32(ssnd->blockSize);
+
+    /* Leave the SDL_RWops position indicator at the start of the samples */
+    if (SDL_RWseek(rw, (int) ssnd->offset, SEEK_CUR) == -1) /* !!! FIXME: Int? Really? */
+        return(0);
+
+    return(1);
+} /* read_ssnd_chunk */
+
+
+static int find_chunk(SDL_RWops *rw, Uint32 id)
+{
+    Sint32 siz = 0;
+    Uint32 _id = 0;
+
+    while (1)
+    {
+        BAIL_IF_MACRO(SDL_RWread(rw, &_id, sizeof (_id), 1) != 1, NULL, 0);
+        if (SDL_SwapLE32(_id) == id)
+            return(1);
+
+        BAIL_IF_MACRO(SDL_RWread(rw, &siz, sizeof (siz), 1) != 1, NULL, 0);
+        siz = SDL_SwapBE32(siz);
+        assert(siz > 0);
+        BAIL_IF_MACRO(SDL_RWseek(rw, siz, SEEK_CUR) == -1, NULL, 0);
+    } /* while */
+
+    return(0);  /* shouldn't hit this, but just in case... */
+} /* find_chunk */
+
+
+static int AIFF_open(Sound_Sample *sample, const char *ext)
+{
+    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;
+    SDL_RWops *rw = internal->rw;
+    Uint32 chunk_id;
+    int bytes_per_sample;
+    long pos;
+    comm_t c;
+    ssnd_t s;
+    aiff_t *a;
+
+    BAIL_IF_MACRO(SDL_ReadLE32(rw) != formID, "AIFF: Not a FORM file.", 0);
+        SDL_ReadBE32(rw);  /* throw the length away; we don't need it. */
+
+    chunk_id = SDL_ReadLE32(rw);
+    BAIL_IF_MACRO(chunk_id != aiffID && chunk_id != aifcID,
+        "AIFF: Not an AIFF or AIFC file.", 0);
+
+    /* Chunks may appear in any order, so we establish base camp here. */
+    pos = SDL_RWtell(rw);
+
+    BAIL_IF_MACRO(!find_chunk(rw, commID), "AIFF: No common chunk.", 0);
+    BAIL_IF_MACRO(!read_comm_chunk(rw, &c), "AIFF: Can't read common chunk.", 0);
+
+    /* !!! FIXME: This will have to change for compression types... */
+    BAIL_IF_MACRO(c.compressionType != noneID, "AIFF: Unsupported encoding.", 0);
+
+    BAIL_IF_MACRO(c.sampleRate == 0, "AIFF: Unsupported sample rate.", 0);
+
+    sample->actual.channels = (Uint8) c.numChannels;
+    sample->actual.rate = c.sampleRate;
+
+    if (c.sampleSize <= 8)
+    {
+        sample->actual.format = AUDIO_S8;
+        bytes_per_sample = 1;
+    } /* if */
+    else if (c.sampleSize <= 16)
+    {
+        sample->actual.format = AUDIO_S16MSB;
+        bytes_per_sample = 2;
+    } /* if */
+    else
+        BAIL_MACRO("AIFF: Unsupported sample size.", 0);
+
+    SDL_RWseek(rw, pos, SEEK_SET);
+
+    BAIL_IF_MACRO(!find_chunk(rw, ssndID), "AIFF: No sound data chunk.", 0);
+    BAIL_IF_MACRO(!read_ssnd_chunk(rw, &s), "AIFF: Can't read sound data chunk.", 0);
+
+    a = (aiff_t *) malloc(sizeof(aiff_t));
+    BAIL_IF_MACRO(a == NULL, ERR_OUT_OF_MEMORY, 0);
+    a->bytesLeft = bytes_per_sample * c.numSampleFrames;
+    internal->decoder_private = (void *) a;
+
+    _D(("AIFF: Accepting data stream.\n"));
+    return(1); /* we'll handle this data. */
+} /* AIFF_open */
+
+
+static void AIFF_close(Sound_Sample *sample)
+{
+    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;
+    free(internal->decoder_private);
+} /* WAV_close */
+
+
+static Uint32 AIFF_read(Sound_Sample *sample)
+{
+    Uint32 retval;
+    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;
+    aiff_t *a = (aiff_t *) internal->decoder_private;
+    Uint32 max = (internal->buffer_size < (Uint32) a->bytesLeft) ?
+                    internal->buffer_size : (Uint32) a->bytesLeft;
+
+    assert(max > 0);
+
+        /*
+         * We don't actually do any decoding, so we read the AIFF data
+         *  directly into the internal buffer...
+         */
+    retval = SDL_RWread(internal->rw, internal->buffer, 1, max);
+
+    a->bytesLeft -= retval;
+
+        /* Make sure the read went smoothly... */
+    if ((retval == 0) || (a->bytesLeft == 0))
+        sample->flags |= SOUND_SAMPLEFLAG_EOF;
+
+    else if (retval == -1)
+        sample->flags |= SOUND_SAMPLEFLAG_ERROR;
+
+        /* (next call this EAGAIN may turn into an EOF or error.) */
+    else if (retval < internal->buffer_size)
+        sample->flags |= SOUND_SAMPLEFLAG_EAGAIN;
+
+    return(retval);
+} /* AIFF_read */
+
+
+/* end of aiff.c ... */