c73894213d
Prevents cutoff of tracks, especially short ones: * Extend looped tracks by fade length to fade at start of loop repeat. * No fade occurs for non-repeating track only having an intro. * Uses id3.tail_trim field to store fade duration. Use libGME built-in elapsed time reporting instead of custom calculation: * libGME already reports in milliseconds. * Don't advance time counter when Repeat == One. It just runs the progress over the length limit. Fix a comment about sample rate and set the reported bitrate to be accurate for 44.1 kHz stereo. Change-Id: I3ede22bda0f9a941a3fef751f4d678eb0027344c
158 lines
4.3 KiB
C
158 lines
4.3 KiB
C
|
|
/* Ripped off from Game_Music_Emu 0.5.2. http://www.slack.net/~ant/ */
|
|
/* Inflate code taken from WikiViewer plugin by Adam Gashlin */
|
|
|
|
#include <codecs/lib/codeclib.h>
|
|
|
|
#include "libgme/blargg_endian.h"
|
|
#include "libgme/vgm_emu.h"
|
|
#include "libgme/inflate/mallocer.h"
|
|
#include "libgme/inflate/inflate.h"
|
|
|
|
CODEC_HEADER
|
|
|
|
/* Maximum number of bytes to process in one iteration */
|
|
#define CHUNK_SIZE (1024*4)
|
|
#define MAINMEMBUF 0
|
|
|
|
static int16_t samples[CHUNK_SIZE] IBSS_ATTR;
|
|
static struct Vgm_Emu vgm_emu;
|
|
|
|
static void *inflatebuf; /* heap for gunzip */
|
|
static char *songbuf; /* destination for uncompressed song */
|
|
static uint32_t songbuflen=0; /* size of the song buffer */
|
|
static uint32_t songlen=0; /* used size of the song buffer */
|
|
|
|
static void codec_vgz_update_length(void)
|
|
{
|
|
ci->id3->length = Track_get_length( &vgm_emu );
|
|
ci->id3->tail_trim = 4 * 1000;
|
|
|
|
if (vgm_emu.info.loop_length <= 0)
|
|
ci->id3->tail_trim = 0;
|
|
|
|
ci->id3->length += ci->id3->tail_trim;
|
|
}
|
|
|
|
static void codec_update_fade(void)
|
|
{
|
|
/* for REPEAT_ONE we disable track limits */
|
|
Track_set_fade(&vgm_emu,
|
|
ci->loop_track() ? indefinite_count :
|
|
ci->id3->length - ci->id3->tail_trim,
|
|
ci->id3->tail_trim);
|
|
}
|
|
|
|
static void codec_update_elapsed(void)
|
|
{
|
|
ci->set_elapsed(ci->loop_track() ? 0 : Track_tell(&vgm_emu));
|
|
}
|
|
|
|
/****************** rockbox interface ******************/
|
|
|
|
/* this is the codec entry point */
|
|
enum codec_status codec_main(enum codec_entry_call_reason reason)
|
|
{
|
|
if (reason == CODEC_LOAD) {
|
|
/* we only render 16 bits */
|
|
ci->configure(DSP_SET_SAMPLE_DEPTH, 16);
|
|
|
|
/* 44 Khz, Interleaved stereo */
|
|
ci->configure(DSP_SET_FREQUENCY, 44100);
|
|
ci->configure(DSP_SET_STEREO_MODE, STEREO_INTERLEAVED);
|
|
|
|
Vgm_init(&vgm_emu);
|
|
Vgm_set_sample_rate(&vgm_emu, 44100);
|
|
}
|
|
|
|
return CODEC_OK;
|
|
}
|
|
|
|
/* this is called for each file to process */
|
|
enum codec_status codec_run(void)
|
|
{
|
|
blargg_err_t err;
|
|
uint8_t *buf;
|
|
size_t n;
|
|
intptr_t param;
|
|
|
|
DEBUGF("VGM: next_track\n");
|
|
if (codec_init()) {
|
|
return CODEC_ERROR;
|
|
}
|
|
|
|
codec_set_replaygain(ci->id3);
|
|
|
|
/* Read the entire file */
|
|
DEBUGF("VGM: request file\n");
|
|
ci->seek_buffer(0);
|
|
buf = ci->request_buffer(&n, ci->filesize);
|
|
if (!buf) {
|
|
DEBUGF("VGM: file load failed\n");
|
|
return CODEC_ERROR;
|
|
}
|
|
|
|
/* If couldn't get the whole buffer
|
|
will trim file and put and 'end_command'
|
|
at the end*/
|
|
if (n < (size_t)ci->filesize) {
|
|
DEBUGF("VGM: file was trimmed\n");
|
|
}
|
|
|
|
/* If is gzipped decompress it */
|
|
if ( get_le16( buf ) == 0x8b1f ) {
|
|
wpw_init_mempool(MAINMEMBUF);
|
|
inflatebuf=wpw_malloc(MAINMEMBUF,0x13500);
|
|
|
|
/* Will use available remaining memory
|
|
as output buffer */
|
|
songbuflen=wpw_available(MAINMEMBUF);
|
|
songbuf=wpw_malloc(MAINMEMBUF,songbuflen);
|
|
|
|
songlen=decompress(buf,n,songbuf,songbuflen,0,inflatebuf);
|
|
|
|
if ((err = Vgm_load_mem(&vgm_emu, songbuf, songlen, true))) {
|
|
DEBUGF("VGM: Vgm_load_mem failed (%s)\n", err);
|
|
return CODEC_ERROR;
|
|
}
|
|
|
|
/* Since metadata parser doesn't support VGZ
|
|
will set song length here */
|
|
codec_vgz_update_length();
|
|
}
|
|
else if ((err = Vgm_load_mem(&vgm_emu, buf, n, false))) {
|
|
DEBUGF("VGM: Vgm_load failed_mem (%s)\n", err);
|
|
return CODEC_ERROR;
|
|
}
|
|
|
|
Vgm_start_track(&vgm_emu);
|
|
|
|
ci->set_elapsed(0);
|
|
codec_update_fade();
|
|
|
|
/* The main decoder loop */
|
|
while (1) {
|
|
enum codec_command_action action = ci->get_command(¶m);
|
|
|
|
if (action == CODEC_ACTION_HALT)
|
|
break;
|
|
|
|
if (action == CODEC_ACTION_SEEK_TIME) {
|
|
Track_seek(&vgm_emu, param);
|
|
codec_update_elapsed();
|
|
ci->seek_complete();
|
|
|
|
/* Set fade again in case we seek to start of song */
|
|
codec_update_fade();
|
|
}
|
|
|
|
/* Generate audio buffer */
|
|
err = Vgm_play(&vgm_emu, CHUNK_SIZE, samples);
|
|
if (err || Track_ended(&vgm_emu)) break;
|
|
|
|
ci->pcmbuf_insert(samples, NULL, CHUNK_SIZE / 2);
|
|
codec_update_elapsed();
|
|
}
|
|
|
|
return CODEC_OK;
|
|
}
|