view decoders/mikmod.c @ 558:3b054685eac9

Tagging 1.0.2 release.
author Ryan C. Gordon <icculus@icculus.org>
date Tue, 27 Jan 2009 17:09:03 -0500
parents 2e8907ff98e9
children
line wrap: on
line source

/*
 * 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
 */

/*
 * Module player for SDL_sound. This driver handles anything MikMod does, and
 *  is based on SDL_mixer.
 *
 * Please see the file LICENSE.txt in the source's root directory.
 *
 *  This file written by Torbjörn Andersson (d91tan@Update.UU.SE)
 */

#if HAVE_CONFIG_H
#  include <config.h>
#endif

#ifdef SOUND_SUPPORTS_MIKMOD

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#include "SDL_sound.h"

#define __SDL_SOUND_INTERNAL__
#include "SDL_sound_internal.h"

#include "mikmod.h"


static int MIKMOD_init(void);
static void MIKMOD_quit(void);
static int MIKMOD_open(Sound_Sample *sample, const char *ext);
static void MIKMOD_close(Sound_Sample *sample);
static Uint32 MIKMOD_read(Sound_Sample *sample);
static int MIKMOD_rewind(Sound_Sample *sample);
static int MIKMOD_seek(Sound_Sample *sample, Uint32 ms);

static const char *extensions_mikmod[] =
{
    "669",   /* Composer 669                                                */
    "AMF",   /* DMP Advanced Module Format                                  */
    "DSM",   /* DSIK internal format                                        */
    "FAR",   /* Farandole module                                            */
    "GDM",   /* General DigiMusic module                                    */
    "IMF",   /* Imago Orpheus module                                        */
    "IT",    /* Impulse tracker                                             */
    "M15",   /* 15 instrument MOD / Ultimate Sound Tracker (old M15 format) */
    "MED",   /* Amiga MED module                                            */
    "MOD",   /* Generic MOD (Protracker, StarTracker, FastTracker, etc)     */
    "MTM",   /* MTM module                                                  */
    "OKT",   /* Oktalyzer module                                            */
    "S3M",   /* Screamtracker module                                        */
    "STM",   /* Screamtracker 2 module                                      */
    "STX",   /* STMIK 0.2 module                                            */
    "ULT",   /* Ultratracker module                                         */
    "UNI",   /* UNIMOD - libmikmod's and APlayer's internal module format   */
    "XM",    /* Fasttracker module                                          */
    NULL
};

const Sound_DecoderFunctions __Sound_DecoderFunctions_MIKMOD =
{
    {
        extensions_mikmod,
        "Play modules through MikMod",
        "Torbjörn Andersson <d91tan@Update.UU.SE>",
        "http://mikmod.raphnet.net/"
    },

    MIKMOD_init,       /*   init() method */
    MIKMOD_quit,       /*   quit() method */
    MIKMOD_open,       /*   open() method */
    MIKMOD_close,      /*  close() method */
    MIKMOD_read,       /*   read() method */
    MIKMOD_rewind,     /* rewind() method */
    MIKMOD_seek        /*   seek() method */
};


/* Make MikMod read from a RWops... */

typedef struct MRWOPSREADER {
    MREADER core;
    Sound_Sample *sample;
    int end;
} MRWOPSREADER;

static BOOL _mm_RWopsReader_eof(MREADER *reader)
{
    MRWOPSREADER *rwops_reader = (MRWOPSREADER *) reader;
    Sound_Sample *sample = rwops_reader->sample;
    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;
    int pos = SDL_RWtell(internal->rw);

    if (rwops_reader->end == pos)
        return(1);

    return(0);
} /* _mm_RWopsReader_eof */


static BOOL _mm_RWopsReader_read(MREADER *reader, void *ptr, size_t size)
{
    MRWOPSREADER *rwops_reader = (MRWOPSREADER *) reader;
    Sound_Sample *sample = rwops_reader->sample;
    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;
    return(SDL_RWread(internal->rw, ptr, size, 1));
} /* _mm_RWopsReader_Read */


static int _mm_RWopsReader_get(MREADER *reader)
{
    char buf;
    MRWOPSREADER *rwops_reader = (MRWOPSREADER *) reader;
    Sound_Sample *sample = rwops_reader->sample;
    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;

    if (SDL_RWread(internal->rw, &buf, 1, 1) != 1)
        return(EOF);

    return((int) buf);
} /* _mm_RWopsReader_get */


static BOOL _mm_RWopsReader_seek(MREADER *reader, long offset, int whence)
{
    MRWOPSREADER *rwops_reader = (MRWOPSREADER *) reader;
    Sound_Sample *sample = rwops_reader->sample;
    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;

    return(SDL_RWseek(internal->rw, offset, whence));
} /* _mm_RWopsReader_seek */


static long _mm_RWopsReader_tell(MREADER *reader)
{
    MRWOPSREADER *rwops_reader = (MRWOPSREADER *) reader;
    Sound_Sample *sample = rwops_reader->sample;
    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;

    return(SDL_RWtell(internal->rw));
} /* _mm_RWopsReader_tell */


static MREADER *_mm_new_rwops_reader(Sound_Sample *sample)
{
    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;

    MRWOPSREADER *reader = (MRWOPSREADER *) malloc(sizeof (MRWOPSREADER));
    if (reader != NULL)
    {
        int failed_seek = 1;
        int here;
        reader->core.Eof  = _mm_RWopsReader_eof;
        reader->core.Read = _mm_RWopsReader_read;
        reader->core.Get  = _mm_RWopsReader_get;
        reader->core.Seek = _mm_RWopsReader_seek;
        reader->core.Tell = _mm_RWopsReader_tell;
        reader->sample = sample;

        /* RWops does not explicitly support an eof check, so we shall find
           the end manually - this requires seek support for the RWop */
        here = SDL_RWtell(internal->rw);
        if (here != -1)
        {
            reader->end = SDL_RWseek(internal->rw, 0, SEEK_END);
            if (reader->end != -1)
            {
                /* Move back */
                if (SDL_RWseek(internal->rw, here, SEEK_SET) != -1)
                    failed_seek = 0;
            } /* if */
        } /* if */

        if (failed_seek)
        {
            free(reader);
            reader = NULL;
        } /* if */
    } /* if */

    return((MREADER *) reader);
} /* _mm_new_rwops_reader */


static void _mm_delete_rwops_reader(MREADER *reader)
{
    /* SDL_sound will delete the RWops and sample at a higher level... */
    if (reader != NULL)
        free(reader);
} /* _mm_delete_rwops_reader */



static int MIKMOD_init(void)
{
    MikMod_RegisterDriver(&drv_nos);
    
        /*
         * Quick and dirty hack to prevent an infinite loop problem
         *  found when using SDL_mixer and SDL_sound together and
         *  both have MikMod compiled in. So, check to see if
         *  MikMod has already been registered first before calling
         *  RegisterAllLoaders()
         */
    if (MikMod_InfoLoader() == NULL)
    {
        MikMod_RegisterAllLoaders();
    } /* if */

        /*
         * Both DMODE_SOFT_MUSIC and DMODE_16BITS should be set by default,
         * so this is just for clarity. I haven't experimented with any of
         * the other flags. There are a few which are said to give better
         * sound quality.
         */
    md_mode |= (DMODE_SOFT_MUSIC | DMODE_16BITS);
    md_mixfreq = 0;
    md_reverb = 1;

    BAIL_IF_MACRO(MikMod_Init(""), MikMod_strerror(MikMod_errno), 0);

    return(1);  /* success. */
} /* MIKMOD_init */


static void MIKMOD_quit(void)
{
    MikMod_Exit();
    md_mixfreq = 0;
} /* MIKMOD_quit */


static int MIKMOD_open(Sound_Sample *sample, const char *ext)
{
    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;
    MREADER *reader;
    MODULE *module;
    Uint32 i; /* temp counter for time computation */
    double segment_time = 0.0; /* temp holder for time */
    
    reader = _mm_new_rwops_reader(sample);
    BAIL_IF_MACRO(reader == NULL, ERR_OUT_OF_MEMORY, 0);
    module = Player_LoadGeneric(reader, 64, 0);
    _mm_delete_rwops_reader(reader);
    BAIL_IF_MACRO(module == NULL, "MIKMOD: Not a module file.", 0);

    module->extspd  = 1;
    module->panflag = 1;
    module->wrap    = 0;
    module->loop    = 0;

    if (md_mixfreq == 0)
        md_mixfreq = (!sample->desired.rate) ? 44100 : sample->desired.rate;

    sample->actual.channels = 2;
    sample->actual.rate = md_mixfreq;
    sample->actual.format = AUDIO_S16SYS;
    internal->decoder_private = (void *) module;

    Player_Start(module);
    Player_SetPosition(0);

    sample->flags = SOUND_SAMPLEFLAG_NONE;

    /*
     *   module->sngtime = current song time in 2^-10 seconds
     *   internal->total_time = (module->sngtime * 1000) / (1<<10)
     */
    internal->total_time = (module->sngtime * 1000) / (1<<10);

    SNDDBG(("MIKMOD: Name: %s\n", module->songname));
    SNDDBG(("MIKMOD: Type: %s\n", module->modtype));
    SNDDBG(("MIKMOD: Accepting data stream\n"));


    /* 
     * This is a quick and dirty way for getting the play time
     * of a file. This will often be wrong because the tracker format
     * allows for so much. If you want a better one, use ModPlug,
     * demand that the Mikmod people write better functionality,
     * or write a more complicated version of the code below.
     *
     * There are two dumb ways to compute the length. The really
     * dumb way is to look at the header and take the initial
     * speed/tempo  values. However, speed values can change throughout
     * the file. The slightly smarter way is to iterate through
     * all the positions and add up each segment's time based
     * on the idea that each segment will give us its own
     * speed value. The hope is that this is more accurate.
     * However, this demands that the file be seekable
     * and that we can change the position of the sample.
     * Depending on the assumptions of SDL_sound, this block
     * of code should be enabled or disabled. If disabled,
     * you still can make the computations doing the first method.
     * For now, we will assume it's acceptable to seek a Mod file
     * since this is essentially how Modplug also does it.
     *
     * Keep in mind that this will be incorrect for loops, jumps, short
     * patterns and other features.
     */
    sample->flags |= SOUND_SAMPLEFLAG_CANSEEK;

    /* 
     * For each position (which corresponds to a particular pattern),
     * get the speed values and compute the time length of the segment
     */
    internal->total_time = 0;
    for (i = 0; i < module->numpos; i++)
    {
        Player_SetPosition(i);
        /* Must call update, or the speed values won't get reset */
        MikMod_Update();
        /* Now the magic formula:
         * Multiply the number of positions by the
         * Number of rows (usually 64 but can be different) by the
         * time it takes to read one row (1/50)
         * by the speed by the 
         * magic reference beats per minute / the beats per minute
         * 
         * We're using positions instead of patterns because in our
         * test cases, this seems to be the correct value for the 
         * number of sections you hear during normal playback.
         * They typically map to a fewer number of patterns
         * where some patterns get replayed multiple times
         * in a song (think chorus). Since we're in a for-loop,
         * the multiplication is implicit while we're adding
         * all the segments.
         * 
         * From a tracker format spec, it seems that 64 rows
         * is the normal (00-3F), but I've seen songs that 
         * either have less or are jumping positions in the
         * middle of a pattern. It looks like Mikmod might
         * reveal this number for us.
         *
         * According to the spec, it seems that a speed of 1
         * corresponds to reading 1 row in 50 ticks. However,
         * I'm not sure if ticks are real seconds or this
         * notion of second units: 
         * Assuming that it's just normal seconds, we get 1/50 = 0.02.
         *
         * The current speed and current tempo (beats per minute) 
         * we can just grab. However, we need a magic number 
         * to figure out what the tempo is based on. Our primitive
         * stopwatch results and intuition seem to imply 120-130bpm 
         * is the magic number. Looking at the majority of tracker
         * files I have, 125 seems to be the common value. Furthermore
         * most (if not all) of my Future Crew .S3M (Scream Tracker 3)
         * files also use 125. Since they invented that format, 
         * I'll also assume that's the base number.
         */
        if(module->bpm == 0)
        {
            /* 
             * Should never get here, but I don't want any
             * divide by zero errors
             */
            continue;
        } /* if */
        segment_time += (module->numrow * .02 * module->sngspd *
                          125.0 / module->bpm);
    } /* for */
    /* Now convert to milliseconds and store the value */
    internal->total_time = (Sint32)(segment_time * 1000);

    /* Reset the sample to the beginning */
    Player_SetPosition(0);
    MikMod_Update();

    return(1); /* we'll handle this data. */
} /* MIKMOD_open */


static void MIKMOD_close(Sound_Sample *sample)
{
    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;
    MODULE *module = (MODULE *) internal->decoder_private;

    Player_Free(module);
} /* MIKMOD_close */


static Uint32 MIKMOD_read(Sound_Sample *sample)
{
    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;
    MODULE *module = (MODULE *) internal->decoder_private;

        /* Switch to the current module, stopping any previous one. */
    Player_Start(module);
    if (!Player_Active())
    {
        sample->flags |= SOUND_SAMPLEFLAG_EOF;
        return(0);
    } /* if */
    return((Uint32) VC_WriteBytes(internal->buffer, internal->buffer_size));
} /* MIKMOD_read */


static int MIKMOD_rewind(Sound_Sample *sample)
{
    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;
    MODULE *module = (MODULE *) internal->decoder_private;

    Player_Start(module);
    Player_SetPosition(0);
    return(1);
} /* MIKMOD_rewind */


static int MIKMOD_seek(Sound_Sample *sample, Uint32 ms)
{
    Sound_SampleInternal *internal = (Sound_SampleInternal *) sample->opaque;
    MODULE *module = (MODULE *) internal->decoder_private;
    double last_time = 0.0;
    double current_time = 0.0;
    double target_time;
    Uint32 i; 

        /*
         * Heaven may know what the argument to Player_SetPosition() is.
         * I, however, haven't the faintest idea.
         */
    Player_Start(module);

        /*
         * Mikmod only lets you seek to the beginning of a pattern.
         * This means we'll get very coarse grain seeks. The 
         * value we pass to SetPosition is a value between 0 
         * and the number of positions in the file. The
         * dumb approach would be to take our total_time that
         * we've already calculated and divide it up by the 
         * number of positions and seek to the position that results.
         * However, because songs can alter their speed/tempo during
         * playback, different patterns in the song can take 
         * up different amounts of time. So the slightly
         * smarter approach is to repeat what was done in the
         * total_time computation and traverse through the file
         * until we find the closest position.
         * The follwing is basically cut and paste from the 
         * open function.
         */
    if (ms == 0)  /* Check end conditions to simplify things */
    {
        Player_SetPosition(0);
        return(1);
    } /* if */

    if (ms >= internal->total_time)
        Player_SetPosition(module->numpos);

    /* Convert time to seconds (double) to make comparisons easier */
    target_time = ms / 1000.0;
    
    for (i = 0; i < module->numpos; i++)
    {
        Player_SetPosition(i);
        /* Must call update, or the speed values won't get reset */
        MikMod_Update();
        /* Divide by zero check */
        if(module->bpm == 0)
            continue;
        last_time = current_time;
        /* See the notes in the open function about the formula */
        current_time += (module->numrow * .02 
            * module->sngspd * 125.0 / module->bpm);
        if(target_time <= current_time)
            break; /* We now have our interval, so break out */
    } /* for */
    
    if( (target_time-last_time) > (current_time-target_time) )
    {
        /* The target time is closer to the higher position, so go there */
        Player_SetPosition(i+1);
    } /* if */
    else
    {
        /* The target time is closer to the lower position, so go there */
        Player_SetPosition(i);
    } /* else */

    return(1);
} /* MIKMOD_seek */

#endif /* SOUND_SUPPORTS_MIKMOD */


/* end of mikmod.c ... */