/* Copyright (C) 2009 Jonathon Fowler This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program 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 General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ /** * OggVorbis source support for MultiVoc */ #include "compat.h" #ifdef HAVE_VORBIS #include "pitch.h" #include "multivoc.h" #include "_multivc.h" #define OV_EXCLUDE_STATIC_CALLBACKS #if defined __APPLE__ # include #elif defined GEKKO # define USING_TREMOR # include #else # include "vorbis/vorbisfile.h" #endif #define BLOCKSIZE MV_MIXBUFFERSIZE typedef struct { void * ptr; size_t length; size_t pos; OggVorbis_File vf; char block[BLOCKSIZE]; int32_t lastbitstream; } vorbis_data; // designed with multiple calls in mind static void MV_GetVorbisCommentLoops(VoiceNode *voice, vorbis_comment *vc) { if (vc == NULL) return; const char *vc_loopstart = NULL; const char *vc_loopend = NULL; const char *vc_looplength = NULL; for (int comment = 0; comment < vc->comments; ++comment) { auto entry = (const char *)vc->user_comments[comment]; if (entry != NULL && entry[0] != '\0') { const char *value = Bstrchr(entry, '='); if (!value) continue; const size_t field = value - entry; value += 1; for (int t = 0; t < loopStartTagCount && vc_loopstart == NULL; ++t) { auto tag = loopStartTags[t]; if (field == Bstrlen(tag) && Bstrncasecmp(entry, tag, field) == 0) vc_loopstart = value; } for (int t = 0; t < loopEndTagCount && vc_loopend == NULL; ++t) { auto tag = loopEndTags[t]; if (field == Bstrlen(tag) && Bstrncasecmp(entry, tag, field) == 0) vc_loopend = value; } for (int t = 0; t < loopLengthTagCount && vc_looplength == NULL; ++t) { auto tag = loopLengthTags[t]; if (field == Bstrlen(tag) && Bstrncasecmp(entry, tag, field) == 0) vc_looplength = value; } } } if (vc_loopstart != NULL) { const ogg_int64_t ov_loopstart = Batol(vc_loopstart); if (ov_loopstart >= 0) // a loop starting at 0 is valid { voice->LoopStart = (const char *) (intptr_t) ov_loopstart; voice->LoopSize = 1; } } if (vc_loopend != NULL) { if (voice->LoopSize > 0) { const ogg_int64_t ov_loopend = Batol(vc_loopend); if (ov_loopend > 0) // a loop ending at 0 is invalid voice->LoopEnd = (const char *) (intptr_t) ov_loopend; } } if (vc_looplength != NULL) { if (voice->LoopSize > 0 && voice->LoopEnd == 0) { const ogg_int64_t ov_looplength = Batol(vc_looplength); if (ov_looplength > 0) // a loop of length 0 is invalid voice->LoopEnd = (const char *) ((intptr_t) ov_looplength + (intptr_t) voice->LoopStart); } } } // callbacks static size_t read_vorbis(void *ptr, size_t size, size_t nmemb, void *datasource) { auto vorb = (vorbis_data *)datasource; errno = 0; if (vorb->length == vorb->pos) return 0; int32_t nread = 0; for (; nmemb > 0; nmemb--, nread++) { int32_t bytes = vorb->length - vorb->pos; if ((signed)size < bytes) bytes = (int32_t)size; memcpy(ptr, (uint8_t *)vorb->ptr + vorb->pos, bytes); vorb->pos += bytes; ptr = (uint8_t *)ptr + bytes; if (vorb->length == vorb->pos) { nread++; break; } } return nread; } static int seek_vorbis(void *datasource, ogg_int64_t offset, int whence) { auto vorb = (vorbis_data *)datasource; switch (whence) { case SEEK_SET: vorb->pos = 0; break; case SEEK_CUR: break; case SEEK_END: vorb->pos = vorb->length; break; } vorb->pos += offset; if (vorb->pos > vorb->length) vorb->pos = vorb->length; return vorb->pos; } static int close_vorbis(void *datasource) { UNREFERENCED_PARAMETER(datasource); return 0; } static long tell_vorbis(void *datasource) { auto vorb = (vorbis_data *)datasource; return vorb->pos; } static ov_callbacks vorbis_callbacks = { read_vorbis, seek_vorbis, close_vorbis, tell_vorbis }; int32_t MV_GetVorbisPosition(VoiceNode *voice) { auto vd = (vorbis_data *) voice->rawdataptr; return ov_pcm_tell(&vd->vf); } void MV_SetVorbisPosition(VoiceNode *voice, int32_t position) { auto vd = (vorbis_data *) voice->rawdataptr; ov_pcm_seek(&vd->vf, position); } /*--------------------------------------------------------------------- Function: MV_GetNextVorbisBlock Controls playback of OggVorbis data ---------------------------------------------------------------------*/ static playbackstatus MV_GetNextVorbisBlock(VoiceNode *voice) { int bitstream; int32_t bytesread = 0; auto vd = (vorbis_data *)voice->rawdataptr; do { #ifdef USING_TREMOR int32_t bytes = ov_read(&vd->vf, vd->block + bytesread, BLOCKSIZE - bytesread, &bitstream); #else int32_t bytes = ov_read(&vd->vf, vd->block + bytesread, BLOCKSIZE - bytesread, 0, 2, 1, &bitstream); #endif // fprintf(stderr, "ov_read = %d\n", bytes); if (bytes > 0) { ogg_int64_t currentPosition; bytesread += bytes; if ((ogg_int64_t)(intptr_t)voice->LoopEnd > 0 && (currentPosition = ov_pcm_tell(&vd->vf)) >= (ogg_int64_t)(intptr_t)voice->LoopEnd) { bytesread -= (currentPosition - (ogg_int64_t)(intptr_t)voice->LoopEnd) * voice->channels * 2; // (voice->bits>>3) int const err = ov_pcm_seek(&vd->vf, (ogg_int64_t)(intptr_t)voice->LoopStart); if (err != 0) { MV_Printf("MV_GetNextVorbisBlock ov_pcm_seek: LOOP_START %l, LOOP_END %l, err %d\n", (ogg_int64_t)(intptr_t)voice->LoopStart, (ogg_int64_t)(intptr_t)voice->LoopEnd, err); } } continue; } else if (bytes == OV_HOLE) continue; else if (bytes == 0) { if (voice->LoopSize > 0) { int const err = ov_pcm_seek(&vd->vf, (ogg_int64_t)(intptr_t)voice->LoopStart); if (err != 0) { MV_Printf("MV_GetNextVorbisBlock ov_pcm_seek: LOOP_START %l, err %d\n", (ogg_int64_t)(intptr_t)voice->LoopStart, err); } else continue; } else { break; } } else if (bytes < 0) { MV_Printf("MV_GetNextVorbisBlock ov_read: err %d\n", bytes); return NoMoreData; } } while (bytesread < BLOCKSIZE); if (bytesread == 0) return NoMoreData; if (bitstream != vd->lastbitstream) { vorbis_info *vi = 0; vi = ov_info(&vd->vf, -1); if (!vi || (vi->channels != 1 && vi->channels != 2)) return NoMoreData; voice->channels = vi->channels; voice->SamplingRate = vi->rate; voice->RateScale = (voice->SamplingRate * voice->PitchScale) / MV_MixRate; voice->FixedPointBufferSize = (voice->RateScale * MV_MIXBUFFERSIZE) - voice->RateScale; MV_SetVoiceMixMode(voice); vd->lastbitstream = bitstream; } uint32_t const samples = bytesread / ((voice->bits/8) * voice->channels); voice->position = 0; voice->sound = vd->block; voice->BlockLength = 0; voice->length = samples << 16; #ifdef GEKKO // If libtremor had the three additional ov_read() parameters that libvorbis has, // this would be better handled using the endianness parameter. int16_t *data = (int16_t *)(vd->block); // assumes signed 16-bit for (bytesread = 0; bytesread < BLOCKSIZE / 2; ++bytesread) data[bytesread] = (data[bytesread] & 0xff) << 8 | ((data[bytesread] & 0xff00) >> 8); #endif return KeepPlaying; } /*--------------------------------------------------------------------- Function: MV_PlayVorbis3D Begin playback of sound data at specified angle and distance from listener. ---------------------------------------------------------------------*/ int32_t MV_PlayVorbis3D(char *ptr, uint32_t length, int32_t loophow, int32_t pitchoffset, int32_t angle, int32_t distance, int32_t priority, float volume, intptr_t callbackval) { if (!MV_Installed) { MV_SetErrorCode(MV_NotInstalled); return MV_Error; } if (distance < 0) { distance = -distance; angle += MV_NUMPANPOSITIONS / 2; } int const vol = MIX_VOLUME(distance); // Ensure angle is within 0 - 127 angle &= MV_MAXPANPOSITION; return MV_PlayVorbis(ptr, length, loophow, -1, pitchoffset, max(0, 255 - distance), MV_PanTable[angle][vol].left, MV_PanTable[angle][vol].right, priority, volume, callbackval); } /*--------------------------------------------------------------------- Function: MV_PlayVorbis Begin playback of sound data with the given sound levels and priority. ---------------------------------------------------------------------*/ int32_t MV_PlayVorbis(char *ptr, uint32_t length, int32_t loopstart, int32_t loopend, int32_t pitchoffset, int32_t vol, int32_t left, int32_t right, int32_t priority, float volume, intptr_t callbackval) { UNREFERENCED_PARAMETER(loopend); if (!MV_Installed) { MV_SetErrorCode(MV_NotInstalled); return MV_Error; } auto vd = (vorbis_data *)Xcalloc(1, sizeof(vorbis_data)); if (!vd) { MV_SetErrorCode(MV_InvalidFile); return MV_Error; } vd->ptr = ptr; vd->pos = 0; vd->length = length; vd->lastbitstream = -1; int32_t status = ov_open_callbacks((void *)vd, &vd->vf, 0, 0, vorbis_callbacks); if (status < 0) { Xfree(vd); MV_Printf("MV_PlayVorbis: err %d\n", status); MV_SetErrorCode(MV_InvalidFile); return MV_Error; } vorbis_info *vi = ov_info(&vd->vf, 0); if (!vi) { ov_clear(&vd->vf); Xfree(vd); MV_SetErrorCode(MV_InvalidFile); return MV_Error; } if (vi->channels != 1 && vi->channels != 2) { ov_clear(&vd->vf); Xfree(vd); MV_SetErrorCode(MV_InvalidFile); return MV_Error; } // Request a voice from the voice pool VoiceNode *voice = MV_AllocVoice(priority); if (voice == NULL) { ov_clear(&vd->vf); Xfree(vd); MV_SetErrorCode(MV_NoVoices); return MV_Error; } voice->wavetype = FMT_VORBIS; voice->bits = 16; voice->channels = vi->channels; voice->rawdataptr = (void *)vd; voice->GetSound = MV_GetNextVorbisBlock; voice->NextBlock = vd->block; voice->LoopCount = 0; voice->BlockLength = 0; voice->length = 0; voice->next = NULL; voice->prev = NULL; voice->priority = priority; voice->callbackval = callbackval; voice->LoopStart = 0; voice->LoopEnd = 0; voice->LoopSize = (loopstart >= 0 ? 1 : 0); // load loop tags from metadata MV_GetVorbisCommentLoops(voice, ov_comment(&vd->vf, 0)); voice->Paused = FALSE; MV_SetVoicePitch(voice, vi->rate, pitchoffset); MV_SetVoiceMixMode(voice); MV_SetVoiceVolume(voice, vol, left, right, volume); MV_PlayVoice(voice); return voice->handle; } void MV_ReleaseVorbisVoice( VoiceNode * voice ) { if (voice->wavetype != FMT_VORBIS) return; auto vd = (vorbis_data *)voice->rawdataptr; voice->rawdataptr = 0; voice->length = 0; voice->sound = nullptr; ov_clear(&vd->vf); Xfree(vd); } #else #include "_multivc.h" int32_t MV_PlayVorbis(char *ptr, uint32_t ptrlength, int32_t loopstart, int32_t loopend, int32_t pitchoffset, int32_t vol, int32_t left, int32_t right, int32_t priority, float volume, intptr_t callbackval) { UNREFERENCED_PARAMETER(ptr); UNREFERENCED_PARAMETER(ptrlength); UNREFERENCED_PARAMETER(loopstart); UNREFERENCED_PARAMETER(loopend); UNREFERENCED_PARAMETER(pitchoffset); UNREFERENCED_PARAMETER(vol); UNREFERENCED_PARAMETER(left); UNREFERENCED_PARAMETER(right); UNREFERENCED_PARAMETER(priority); UNREFERENCED_PARAMETER(volume); UNREFERENCED_PARAMETER(callbackval); MV_Printf("MV_PlayVorbis: OggVorbis support not included in this binary.\n"); return -1; } int32_t MV_PlayVorbis3D(char *ptr, uint32_t ptrlength, int32_t loophow, int32_t pitchoffset, int32_t angle, int32_t distance, int32_t priority, float volume, intptr_t callbackval) { UNREFERENCED_PARAMETER(ptr); UNREFERENCED_PARAMETER(ptrlength); UNREFERENCED_PARAMETER(loophow); UNREFERENCED_PARAMETER(pitchoffset); UNREFERENCED_PARAMETER(angle); UNREFERENCED_PARAMETER(distance); UNREFERENCED_PARAMETER(priority); UNREFERENCED_PARAMETER(volume); UNREFERENCED_PARAMETER(callbackval); MV_Printf("MV_PlayVorbis: OggVorbis support not included in this binary.\n"); return -1; } #endif //HAVE_VORBIS