| /*************************************************************************** |
| * __________ __ ___. |
| * Open \______ \ ____ ____ | | _\_ |__ _______ ___ |
| * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ / |
| * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < < |
| * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \ |
| * \/ \/ \/ \/ \/ |
| * $Id$ |
| * |
| * Copyright (C) 2005 David Bryant |
| * |
| * All files in this archive are subject to the GNU General Public License. |
| * See the file COPYING in the source tree root for full license agreement. |
| * |
| * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY |
| * KIND, either express or implied. |
| * |
| ****************************************************************************/ |
| |
| #include "codec.h" |
| |
| #include <codecs/libwavpack/wavpack.h> |
| #include "playback.h" |
| #include "lib/codeclib.h" |
| #include "dsp.h" |
| |
| static struct codec_api *rb; |
| static struct codec_api *ci; |
| |
| #define BUFFER_SIZE 4096 |
| |
| static long temp_buffer [BUFFER_SIZE] IDATA_ATTR; |
| |
| static long read_callback (void *buffer, long bytes) |
| { |
| long retval = ci->read_filebuf (buffer, bytes); |
| ci->id3->offset = ci->curpos; |
| return retval; |
| } |
| |
| #ifndef SIMULATOR |
| extern char iramcopy[]; |
| extern char iramstart[]; |
| extern char iramend[]; |
| #endif |
| |
| /* this is the codec entry point */ |
| enum codec_status codec_start(struct codec_api* api) |
| { |
| WavpackContext *wpc; |
| char error [80]; |
| int bps, nchans, sr_100; |
| |
| /* Generic codec initialisation */ |
| TEST_CODEC_API(api); |
| |
| rb = api; |
| ci = api; |
| |
| #ifndef SIMULATOR |
| rb->memcpy(iramstart, iramcopy, iramend-iramstart); |
| #endif |
| |
| ci->configure(CODEC_SET_FILEBUF_LIMIT, (int *)(1024*1024*10)); |
| ci->configure(CODEC_SET_FILEBUF_WATERMARK, (int *)(1024*512)); |
| ci->configure(CODEC_SET_FILEBUF_CHUNKSIZE, (int *)(1024*128)); |
| |
| ci->configure(DSP_DITHER, (bool *)false); |
| ci->configure(DSP_SET_STEREO_MODE, (int *)STEREO_INTERLEAVED); |
| ci->configure(DSP_SET_SAMPLE_DEPTH, (int *)(16)); |
| |
| next_track: |
| |
| if (codec_init(api)) |
| return CODEC_ERROR; |
| |
| while (!*rb->taginfo_ready && !ci->stop_codec) |
| ci->sleep(1); |
| |
| if (ci->id3->frequency != NATIVE_FREQUENCY) { |
| ci->configure(DSP_SET_FREQUENCY, (long *)(ci->id3->frequency)); |
| ci->configure(CODEC_DSP_ENABLE, (bool *)true); |
| } else { |
| ci->configure(CODEC_DSP_ENABLE, (bool *)false); |
| } |
| |
| /* Create a decoder instance */ |
| wpc = WavpackOpenFileInput (read_callback, error); |
| |
| if (!wpc) |
| return CODEC_ERROR; |
| |
| bps = WavpackGetBytesPerSample (wpc); |
| nchans = WavpackGetReducedChannels (wpc); |
| sr_100 = ci->id3->frequency / 100; |
| |
| ci->set_elapsed (0); |
| |
| /* The main decoder loop */ |
| |
| while (1) { |
| long nsamples; |
| |
| if (ci->seek_time && ci->taginfo_ready && ci->id3->length) { |
| int curpos_ms = WavpackGetSampleIndex (wpc) / sr_100 * 10; |
| int n, d, skip; |
| |
| if (ci->seek_time > curpos_ms) { |
| n = ci->seek_time - curpos_ms; |
| d = ci->id3->length - curpos_ms; |
| skip = (int)((long long)(ci->filesize - ci->curpos) * n / d); |
| ci->seek_buffer (ci->curpos + skip); |
| } |
| else { |
| n = curpos_ms - ci->seek_time; |
| d = curpos_ms; |
| skip = (int)((long long) ci->curpos * n / d); |
| ci->seek_buffer (ci->curpos - skip); |
| } |
| |
| wpc = WavpackOpenFileInput (read_callback, error); |
| ci->seek_time = 0; |
| |
| if (!wpc) |
| break; |
| |
| ci->set_elapsed (WavpackGetSampleIndex (wpc) / sr_100 * 10); |
| rb->yield (); |
| } |
| |
| nsamples = WavpackUnpackSamples (wpc, temp_buffer, BUFFER_SIZE / 2); |
| |
| if (!nsamples || ci->stop_codec || ci->reload_codec) |
| break; |
| |
| /* convert mono to stereo here, in place */ |
| |
| if (nchans == 1) { |
| long *dst = temp_buffer + (nsamples * 2); |
| long *src = temp_buffer + nsamples; |
| long count = nsamples; |
| |
| while (count--) { |
| *--dst = *--src; |
| *--dst = *src; |
| if (!(count & 0x7f)) |
| rb->yield (); |
| } |
| } |
| |
| if (bps == 1) { |
| short *dst = (short *) temp_buffer; |
| long *src = temp_buffer; |
| long count = nsamples; |
| |
| while (count--) { |
| *dst++ = *src++ << 8; |
| *dst++ = *src++ << 8; |
| if (!(count & 0x7f)) |
| rb->yield (); |
| } |
| } |
| else if (bps == 2) { |
| short *dst = (short *) temp_buffer; |
| long *src = temp_buffer; |
| long count = nsamples; |
| |
| while (count--) { |
| *dst++ = *src++; |
| *dst++ = *src++; |
| if (!(count & 0x7f)) |
| rb->yield (); |
| } |
| } |
| else { |
| short *dst = (short *) temp_buffer; |
| int shift = (bps - 2) * 8; |
| long *src = temp_buffer; |
| long count = nsamples; |
| |
| while (count--) { |
| *dst++ = *src++ >> shift; |
| *dst++ = *src++ >> shift; |
| if (!(count & 0x7f)) |
| rb->yield (); |
| } |
| } |
| |
| if (ci->stop_codec || ci->reload_codec) |
| break; |
| |
| while (!ci->pcmbuf_insert ((char *) temp_buffer, nsamples * 4)) |
| rb->sleep (1); |
| |
| ci->set_elapsed (WavpackGetSampleIndex (wpc) / sr_100 * 10); |
| } |
| |
| if (ci->request_next_track()) |
| goto next_track; |
| |
| return CODEC_OK; |
| } |