ref: 6159713b2a0f55e2cd41797ed7aa3a5a4c0a06a3
dir: /src/support/cmixer.c/
/* Derivative work of cmixer by rxi (https://github.com/rxi/cmixer) Copyright (c) 2017 rxi Copyright (c) 2023 Iliyas Jorio Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #include "cmixer.h" #include "ibxm.h" #include <SDL.h> #include <stdint.h> #include <stdbool.h> #include <limits.h> #include <stdio.h> #define MAX_CONCURRENT_VOICES 8 #define BUFFER_SIZE 512 #define CM_DIE(message) \ do { \ char buf[256]; \ snprintf(buf, sizeof(buf), "%s:%d: %s", __func__, __LINE__, (message)); \ SDL_ShowSimpleMessageBox(SDL_MESSAGEBOX_ERROR, "cmixer", buf, NULL); \ abort(); \ } while(0) #define CM_ASSERT(assertion, message) do { if (!(assertion)) CM_DIE(message); } while(0) #define FX_BITS (12L) #define FX_UNIT (1L << FX_BITS) #define FX_MASK (FX_UNIT - 1) #define BUFFER_MASK (BUFFER_SIZE - 1) enum { PCMFORMAT_NULL = 0x00, PCMFORMAT_1CH_8 = 0x11, PCMFORMAT_2CH_8 = 0x21, PCMFORMAT_1CH_LE16 = 0x12, PCMFORMAT_2CH_LE16 = 0x22, PCMFORMAT_1CH_BE16 = PCMFORMAT_1CH_LE16 | 0x80, PCMFORMAT_2CH_BE16 = PCMFORMAT_2CH_LE16 | 0x80, }; struct CMWavStream { uint8_t pcmformat; int idx; char* data; size_t dataLength; bool ownData; uint32_t cookie; }; struct CMModStream { struct module* module; struct replay* replay; char* moduleFileMemory; int* replayBuffer; int replayBufferOffset; int replayBufferSamples; double playbackSpeedMult; uint32_t cookie; }; struct CMVoice { int16_t pcmbuf[BUFFER_SIZE]; // Internal buffer with raw stereo PCM int sampleRate; // Stream's native samplerate int sampleCount; // Stream's length in frames int sustainOffset; // Offset of the sustain loop in frames int end; // End index for the current play-through int state; // Current state (playing|paused|stopped) int64_t position; // Current playhead position (fixed point) int lgain, rgain; // Left and right gain (fixed point) int rate; // Playback rate (fixed point) int nextfill; // Next frame idx where the buffer needs to be filled bool loop; // Whether the voice will loop when `end` is reached bool rewind; // Whether the voice will rewind before playing bool active; // Whether the voice is part of `voices` list bool interpolate; // Interpolated resampling when played back at a non-native rate double gain; // Gain set by `cm_set_gain()` double pan; // Pan set by `cm_set_pan()` struct { void (*fillBuffer)(struct CMVoice* voice, int16_t* into, int len); void (*completed)(struct CMVoice* voice); void (*rewind)(struct CMVoice* voice); void (*free)(struct CMVoice* voice); } callbacks; uint32_t cookie; union { struct CMWavStream wav; struct CMModStream mod; }; }; typedef struct CMVoice CMVoice; typedef struct CMWavStream CMWavStream; typedef struct CMModStream CMModStream; static void CMVoice_RemoveFromMixer(CMVoice* voice); static void CMVoice_AddToMix(CMVoice* voice, int len, int32_t* dst); static inline CMWavStream* CMWavStream_Check(CMVoice* voice); static void StreamWav(CMVoice* voice, int16_t* output, int length); static void RewindWav(CMVoice* voice); static void FreeWav(CMVoice* voice); static inline CMModStream* CMModStream_Check(CMVoice* voice); static void StreamMod(CMVoice* voice, int16_t* output, int length); static void FreeMod(CMVoice* voice); //----------------------------------------------------------------------------- // Utilities static inline int DoubleToFixed(double f) { return (int) (f * FX_UNIT); } static inline double FixedToDouble(int f) { return (double) f / FX_UNIT; } static inline int FixedLerp(int a, int b, int p) { return a + (((b - a) * p) >> FX_BITS); } static inline int16_t UnpackI16BE(const void* data) { #if __BIG_ENDIAN__ // no-op on big-endian systems return *(const uint16_t*) data; #else const uint8_t* p = (uint8_t*) data; return ( p[0] << 8 ) | ( p[1] ); #endif } static inline int16_t UnpackI16LE(const void* data) { #if __BIG_ENDIAN__ const uint8_t* p = (uint8_t*) data; return ( p[0] ) | ( p[1] << 8 ); #else // no-op on little-endian systems return *(const uint16_t*) data; #endif } static inline int MinInt(int a, int b) { return a < b ? a : b; } static inline int MaxInt(int a, int b) { return a < b ? b : a; } static inline int ClampInt(int x, int a, int b) { return x < a ? a : x > b ? b : x; } static inline double ClampDouble(double x, double a, double b) { return x < a ? a : x > b ? b : x; } static char* LoadFile(const char* filename, size_t* outSize) { FILE* ifs = fopen(filename, "rb"); if (!ifs) return NULL; fseek(ifs, 0, SEEK_END); long filesize = ftell(ifs); fseek(ifs, 0, SEEK_SET); void* bytes = SDL_malloc(filesize); fread(bytes, 1, filesize, ifs); fclose(ifs); if (outSize) *outSize = filesize; return (char*)bytes; } static uint8_t BuildPCMFormat(int bitdepth, int channels, bool bigEndian) { return ((!!bigEndian) << 7) | (channels << 4) | (bitdepth / 8); } //----------------------------------------------------------------------------- // Global mixer static struct Mixer { SDL_mutex* sdlAudioMutex; CMVoice* voices[MAX_CONCURRENT_VOICES]; // List of active (playing) voices int32_t pcmmixbuf[BUFFER_SIZE]; // Internal master buffer int samplerate; // Master samplerate int gain; // Master gain (fixed point) } gMixer; static void Mixer_Init(struct Mixer* mixer, int samplerate); static void Mixer_Process(struct Mixer* mixer, int16_t* dst, int len); static void Mixer_Lock(struct Mixer* mixer); static void Mixer_Unlock(struct Mixer* mixer); static void Mixer_SetMasterGain(struct Mixer* mixer, double newGain); //----------------------------------------------------------------------------- // Global init/shutdown static bool sdlAudioSubSystemInited = false; static SDL_AudioDeviceID sdlDeviceID = 0; static void MixerCallback(void* udata, Uint8* stream, int size) { struct Mixer* mixer = (struct Mixer*) udata; Mixer_Process(mixer, (int16_t*) stream, size / 2); } void cmixer_InitWithSDL(void) { CM_ASSERT(!sdlAudioSubSystemInited, "SDL audio subsystem already inited"); if (0 != SDL_InitSubSystem(SDL_INIT_AUDIO)) CM_DIE(SDL_GetError()); sdlAudioSubSystemInited = true; // Init SDL audio SDL_AudioSpec fmt = { .freq = 44100, .format = AUDIO_S16SYS, .channels = 2, .samples = 1024, .callback = MixerCallback, .userdata = &gMixer, }; SDL_AudioSpec got; sdlDeviceID = SDL_OpenAudioDevice(NULL, 0, &fmt, &got, SDL_AUDIO_ALLOW_FREQUENCY_CHANGE); CM_ASSERT(sdlDeviceID, SDL_GetError()); // Init library Mixer_Init(&gMixer, got.freq); Mixer_SetMasterGain(&gMixer, 0.5); // Start audio SDL_PauseAudioDevice(sdlDeviceID, 0); } void cmixer_ShutdownWithSDL() { if (sdlDeviceID) { SDL_CloseAudioDevice(sdlDeviceID); sdlDeviceID = 0; } if (gMixer.sdlAudioMutex) { SDL_DestroyMutex(gMixer.sdlAudioMutex); gMixer.sdlAudioMutex = NULL; } if (sdlAudioSubSystemInited) { SDL_QuitSubSystem(SDL_INIT_AUDIO); sdlAudioSubSystemInited = false; } } double cmixer_GetMasterGain() { return FixedToDouble(gMixer.gain); } void cmixer_SetMasterGain(double newGain) { Mixer_SetMasterGain(&gMixer, newGain); } //----------------------------------------------------------------------------- // Global mixer impl static void Mixer_Init(struct Mixer* mixer, int newSamplerate) { SDL_memset(mixer, 0, sizeof(mixer)); mixer->sdlAudioMutex = SDL_CreateMutex(); mixer->samplerate = newSamplerate; mixer->gain = FX_UNIT; } static void Mixer_Lock(struct Mixer* mixer) { SDL_LockMutex(mixer->sdlAudioMutex); } static void Mixer_Unlock(struct Mixer* mixer) { SDL_UnlockMutex(mixer->sdlAudioMutex); } static void Mixer_SetMasterGain(struct Mixer* mixer, double newGain) { if (newGain < 0) newGain = 0; mixer->gain = DoubleToFixed(newGain); } static int Mixer_AddVoice(struct Mixer* mixer, CMVoice* voice) { CM_ASSERT(voice->callbacks.fillBuffer, "fill buffer callback not set"); // Look for a free slot for (int i = 0; i < MAX_CONCURRENT_VOICES; i++) { CM_ASSERT(mixer->voices[i] != voice, "voice added twice to mixer"); if (!mixer->voices[i]) { mixer->voices[i] = voice; return i; } } return -1; } static void Mixer_RemoveVoice(struct Mixer* mixer, CMVoice* voice) { for (int i = 0; i < MAX_CONCURRENT_VOICES; i++) { if (mixer->voices[i] == voice) { mixer->voices[i] = NULL; break; } } } static void Mixer_Process(struct Mixer* mixer, int16_t* dst, int len) { // Process in chunks of BUFFER_SIZE if `len` is larger than BUFFER_SIZE while (len > BUFFER_SIZE) { Mixer_Process(mixer, dst, BUFFER_SIZE); dst += BUFFER_SIZE; len -= BUFFER_SIZE; } // Zeroset internal buffer SDL_memset(mixer->pcmmixbuf, 0, len * sizeof(mixer->pcmmixbuf[0])); // Process active voices Mixer_Lock(mixer); for (int i = 0; i < MAX_CONCURRENT_VOICES; i++) { CMVoice* voice = mixer->voices[i]; if (!voice) continue; CMVoice_AddToMix(voice, len, mixer->pcmmixbuf); // Remove voice from list if it is no longer playing if (voice->state != CM_STATE_PLAYING) { voice->active = false; mixer->voices[i] = NULL; } } Mixer_Unlock(mixer); // Copy internal buffer to destination and clip for (int i = 0; i < len; i++) { int x = (mixer->pcmmixbuf[i] * mixer->gain) >> FX_BITS; dst[i] = ClampInt(x, -32768, 32767); } } //----------------------------------------------------------------------------- // Voice implementation static inline CMVoice* CMVoice_Check(void* ptr) { CMVoice* voice = (CMVoice*) ptr; CM_ASSERT(voice->cookie == 'VOIX', "VOIX cookie not found"); return voice; } static CMVoice* CMVoice_New(int sampleRate, int sampleCount) { CMVoice* voice = SDL_calloc(1, sizeof(CMVoice)); voice->cookie = 'VOIX'; voice->sampleRate = 0; voice->sampleCount = 0; voice->end = 0; voice->state = CM_STATE_STOPPED; voice->position = 0; voice->lgain = 0; voice->rgain = 0; voice->rate = 0; voice->nextfill = 0; voice->loop = false; voice->rewind = true; voice->interpolate = false; voice->gain = 0; voice->pan = 0; voice->active = false; voice->sampleRate = sampleRate; voice->sampleCount = sampleCount; voice->sustainOffset = 0; CMVoice_SetGain(voice, 1); CMVoice_SetPan(voice, 0); CMVoice_SetPitch(voice, 1); CMVoice_SetLoop(voice, false); CMVoice_Stop(voice); return voice; } void CMVoice_Free(CMVoice* voice) { CMVoice_Check(voice); CMVoice_RemoveFromMixer(voice); if (voice->callbacks.free) voice->callbacks.free(voice); voice->cookie = 'DEAD'; SDL_free(voice); } static void CMVoice_RemoveFromMixer(CMVoice* voice) { CMVoice_Check(voice); Mixer_Lock(&gMixer); if (voice->active) { Mixer_RemoveVoice(&gMixer, voice); voice->active = false; } Mixer_Unlock(&gMixer); } void CMVoice_Rewind(CMVoice* voice) { if (voice->callbacks.rewind) voice->callbacks.rewind(voice); voice->position = 0; voice->rewind = false; voice->end = voice->sampleCount; voice->nextfill = 0; } static void CMVoice_AddToMix(CMVoice* voice, int len, int32_t* dst) { CMVoice_Check(voice); // check pointer validity // Do rewind if flag is set if (voice->rewind) { CMVoice_Rewind(voice); } // Don't process if not playing if (voice->state != CM_STATE_PLAYING) { return; } // Process audio while (len > 0) { // Get current position frame int frame = (int) (voice->position >> FX_BITS); // Fill buffer if required if (frame + 3 >= voice->nextfill) { int fillOffset = (voice->nextfill * 2) & BUFFER_MASK; int fillLength = BUFFER_SIZE / 2; voice->callbacks.fillBuffer(voice, voice->pcmbuf + fillOffset, fillLength); voice->nextfill += BUFFER_SIZE / 4; } // Handle reaching the end of the playthrough if (frame >= voice->end) { // As streams continuously fill the raw buffer in a loop, // increment the end idx by one length // and continue reading from it another playthrough voice->end = frame + voice->sampleCount; // Set state and stop processing if we're not set to loop if (!voice->loop) { voice->state = CM_STATE_STOPPED; if (voice->callbacks.completed) voice->callbacks.completed(voice); break; } } // Work out how many frames we should process in the loop int n = MinInt(voice->nextfill - 2, voice->end) - frame; int count = (n << FX_BITS) / voice->rate; count = MaxInt(count, 1); count = MinInt(count, len / 2); len -= count * 2; // Add audio to master buffer if (voice->rate == FX_UNIT) { // Add audio to buffer -- basic n = frame * 2; for (int i = 0; i < count; i++) { dst[0] += (voice->pcmbuf[(n ) & BUFFER_MASK] * voice->lgain) >> FX_BITS; dst[1] += (voice->pcmbuf[(n + 1) & BUFFER_MASK] * voice->rgain) >> FX_BITS; n += 2; dst += 2; } voice->position += count * FX_UNIT; } else if (voice->interpolate) { // Resample audio (with linear interpolation) and add to buffer for (int i = 0; i < count; i++) { n = (int) (voice->position >> FX_BITS) * 2; int p = voice->position & FX_MASK; int a = voice->pcmbuf[(n ) & BUFFER_MASK]; int b = voice->pcmbuf[(n + 2) & BUFFER_MASK]; dst[0] += (FixedLerp(a, b, p) * voice->lgain) >> FX_BITS; n++; a = voice->pcmbuf[(n ) & BUFFER_MASK]; b = voice->pcmbuf[(n + 2) & BUFFER_MASK]; dst[1] += (FixedLerp(a, b, p) * voice->rgain) >> FX_BITS; voice->position += voice->rate; dst += 2; } } else { // Resample audio (without interpolation) and add to buffer for (int i = 0; i < count; i++) { n = (int) (voice->position >> FX_BITS) * 2; dst[0] += (voice->pcmbuf[(n ) & BUFFER_MASK] * voice->lgain) >> FX_BITS; dst[1] += (voice->pcmbuf[(n + 1) & BUFFER_MASK] * voice->rgain) >> FX_BITS; voice->position += voice->rate; dst += 2; } } } } double CMVoice_GetLength(const CMVoice* voice) { return voice->sampleCount / (double) voice->sampleRate; } double CMVoice_GetPosition(const CMVoice* voice) { return ((voice->position >> FX_BITS) % voice->sampleCount) / (double) voice->sampleRate; } int CMVoice_GetState(const CMVoice* voice) { return voice->state; } static void CMVoice_RecalcGains(CMVoice* voice) { double l = voice->gain * (voice->pan <= 0. ? 1. : 1. - voice->pan); double r = voice->gain * (voice->pan >= 0. ? 1. : 1. + voice->pan); voice->lgain = DoubleToFixed(l); voice->rgain = DoubleToFixed(r); } void CMVoice_SetGain(CMVoice* voice, double newGain) { voice->gain = newGain; CMVoice_RecalcGains(voice); } void CMVoice_SetPan(CMVoice* voice, double newPan) { voice->pan = ClampDouble(newPan, -1.0, 1.0); CMVoice_RecalcGains(voice); } void CMVoice_SetPitch(CMVoice* voice, double newPitch) { double newRate; if (newPitch > 0.) { newRate = (double)voice->sampleRate / (double) gMixer.samplerate * newPitch; } else { newRate = 0.001; } voice->rate = DoubleToFixed(newRate); } void CMVoice_SetLoop(CMVoice* voice, int newLoop) { voice->loop = newLoop; } void CMVoice_SetInterpolation(CMVoice* voice, int newInterpolation) { voice->interpolate = newInterpolation; } void CMVoice_Play(CMVoice* voice) { CMVoice_Check(voice); // check pointer validity if (voice->sampleCount == 0) { // Don't attempt to play an empty voice as this would result // in instant starvation when filling mixer buffer return; } Mixer_Lock(&gMixer); if (!voice->active) { int rc = Mixer_AddVoice(&gMixer, voice); if (rc < 0) { // couldn't add voice } else { voice->state = CM_STATE_PLAYING; voice->active = true; } } Mixer_Unlock(&gMixer); } void CMVoice_Pause(CMVoice* voice) { voice->state = CM_STATE_PAUSED; } void CMVoice_TogglePause(CMVoice* voice) { if (voice->state == CM_STATE_PAUSED) CMVoice_Play(voice); else if (voice->state == CM_STATE_PLAYING) CMVoice_Pause(voice); } void CMVoice_Stop(CMVoice* voice) { voice->state = CM_STATE_STOPPED; voice->rewind = true; } //----------------------------------------------------------------------------- // WavStream implementation static inline CMWavStream* CMWavStream_Check(CMVoice* voice) { CM_ASSERT(voice->cookie == 'VOIX', "VOIX cookie not found"); CM_ASSERT(voice->wav.cookie == 'WAVS', "WAVS cookie not found"); return &voice->wav; } static CMWavStream* InstallWavStream(CMVoice* voice) { CMWavStream* wav = &voice->wav; wav->cookie = 'WAVS'; wav->pcmformat = PCMFORMAT_NULL; wav->idx = 0; voice->callbacks.fillBuffer = StreamWav; voice->callbacks.rewind = RewindWav; voice->callbacks.free = FreeWav; return wav; } static void FreeWav(CMVoice* voice) { CMWavStream* wav = CMWavStream_Check(voice); if (!wav->data) { return; } if (wav->ownData) { SDL_free(wav->data); } wav->data = NULL; wav->dataLength = 0; wav->ownData = false; wav->cookie = 'DEAD'; } static void RewindWav(CMVoice* voice) { CMWavStream* wav = CMWavStream_Check(voice); wav->idx = 0; } static void StreamWav(CMVoice* voice, int16_t* dst, int fillLength) { CMWavStream* wav = CMWavStream_Check(voice); int x, n; fillLength /= 2; const int16_t* data16 = (const int16_t*) wav->data; const uint8_t* data8 = (const uint8_t*) wav->data; #define WAV_PROCESS_LOOP(X) \ while (n--) \ { \ X \ dst += 2; \ wav->idx++; \ } while (fillLength > 0) { n = MinInt(fillLength, voice->sampleCount - wav->idx); fillLength -= n; switch (wav->pcmformat) { case PCMFORMAT_1CH_BE16: WAV_PROCESS_LOOP({ dst[0] = dst[1] = UnpackI16BE(&data16[wav->idx]); }); break; case PCMFORMAT_2CH_BE16: WAV_PROCESS_LOOP({ x = wav->idx * 2; dst[0] = UnpackI16BE(&data16[x]); dst[1] = UnpackI16BE(&data16[x + 1]); }); break; case PCMFORMAT_1CH_LE16: WAV_PROCESS_LOOP({ dst[0] = dst[1] = UnpackI16LE(&data16[wav->idx]); }); break; case PCMFORMAT_2CH_LE16: WAV_PROCESS_LOOP({ x = wav->idx * 2; dst[0] = UnpackI16LE(&data16[x]); dst[1] = UnpackI16LE(&data16[x + 1]); }); break; case PCMFORMAT_1CH_8: case PCMFORMAT_1CH_8 | 0x80: // with big-endian flag WAV_PROCESS_LOOP({ dst[0] = dst[1] = (data8[wav->idx] - 128) << 8; }); break; case PCMFORMAT_2CH_8: case PCMFORMAT_2CH_8 | 0x80: // with big-endian flag WAV_PROCESS_LOOP({ x = wav->idx * 2; dst[0] = (data8[x] - 128) << 8; dst[1] = (data8[x + 1] - 128) << 8; }); break; default: CM_DIE("unknown pcmformat"); break; } // Loop back and continue filling buffer if we didn't fill the buffer if (fillLength > 0) { wav->idx = voice->sustainOffset; } } #undef WAV_PROCESS_LOOP } //----------------------------------------------------------------------------- // LoadWAVFromFile static const char* FindRIFFChunk(const char* data, size_t len, const char* id, int* size) { // TODO : Error handling on malformed wav file size_t idlen = SDL_strlen(id); const char* p = data + 12; next: *size = *((uint32_t*)(p + 4)); if (SDL_memcmp(p, id, idlen)) { p += 8 + *size; if (p > data + len) return NULL; goto next; } return p + 8; } CMVoice* CMVoice_LoadWAV(const char* path) { int sz; size_t len = 0; char *const data = LoadFile(path, &len); const char* p = (char*)data; // Check header if (SDL_memcmp(p, "RIFF", 4) || SDL_memcmp(p + 8, "WAVE", 4)) CM_DIE("not a WAVE file"); // Find fmt subchunk p = FindRIFFChunk(data, len, "fmt ", &sz); CM_ASSERT(p, "no fmt chunk in WAVE"); // Load fmt info int format = *((uint16_t*)(p)); int channels = *((uint16_t*)(p + 2)); int samplerate = *((uint32_t*)(p + 4)); int bitdepth = *((uint16_t*)(p + 14)); CM_ASSERT(format == 1, "unsupported WAVE format"); CM_ASSERT(channels == 1 || channels == 2, "unsupported channel count"); CM_ASSERT(bitdepth == 8 || bitdepth == 16, "unsupported bitdepth"); CM_ASSERT(samplerate != 0, "weird samplerate"); // Find data subchunk p = FindRIFFChunk(data, len, "data", &sz); CM_ASSERT(p, "no data chunk in WAVE"); const char* sampleData = p; int sampleDataLength = sz; int samplecount = (sampleDataLength / (bitdepth / 8)) / channels; CMVoice* voice = CMVoice_New(samplerate, samplecount); CMWavStream* wav = InstallWavStream(voice); wav->pcmformat = BuildPCMFormat(bitdepth, channels, 0); wav->data = SDL_malloc(sampleDataLength); wav->dataLength = sampleDataLength; wav->ownData = true; SDL_memcpy(wav->data, sampleData, sampleDataLength); SDL_free(data); CM_ASSERT(wav->pcmformat != 0, "weird pcmformat"); return voice; } //----------------------------------------------------------------------------- // ModStream static inline CMModStream* CMModStream_Check(CMVoice* voice) { CM_ASSERT(voice->cookie == 'VOIX', "VOIX cookie not found"); CM_ASSERT(voice->mod.cookie == 'MODS', "MODS cookie not found"); return &voice->mod; } CMVoice* CMVoice_LoadMOD(const char* path) { char errors[64]; errors[0] = '\0'; size_t moduleFileSize = 0; char* moduleFile = LoadFile(path, &moduleFileSize); struct data d = { .buffer = moduleFile, .length = (int)moduleFileSize }; CMVoice* voice = CMVoice_New(gMixer.samplerate, INT_MAX); voice->callbacks.fillBuffer = StreamMod; voice->callbacks.free = FreeMod; voice->mod.cookie = 'MODS'; voice->mod.replayBuffer = SDL_calloc(1, 2048 * 8 * sizeof(voice->mod.replayBuffer[0])); voice->mod.replayBufferOffset = 0; voice->mod.replayBufferSamples = 0; voice->mod.playbackSpeedMult = 1.0; voice->mod.moduleFileMemory = moduleFile; voice->mod.module = module_load(&d, errors); voice->mod.replay = new_replay(voice->mod.module, gMixer.samplerate, 0); CM_ASSERT(!errors[0], errors); return voice; } static void FreeMod(CMVoice* voice) { CMModStream* mod = CMModStream_Check(voice); dispose_module(mod->module); SDL_free(mod->moduleFileMemory); SDL_free(mod->replayBuffer); mod->cookie = 'DEAD'; } void CMVoice_SetMODPlaybackSpeed(CMVoice* voice, double speed) { CMModStream* mod = CMModStream_Check(voice); mod->playbackSpeedMult = speed; } static void StreamMod(CMVoice* voice, int16_t* output, int length) { CMModStream* mod = CMModStream_Check(voice); length /= 2; while (length > 0) { // refill replay buffer if exhausted if (mod->replayBufferSamples == 0) { mod->replayBufferOffset = 0; mod->replayBufferSamples = replay_get_audio(mod->replay, mod->replayBuffer, 0, (int)(mod->playbackSpeedMult * 100.0)); } // number of stereo samples to copy from replay buffer to output buffer int nToCopy = MinInt(mod->replayBufferSamples, length); int* input = &mod->replayBuffer[mod->replayBufferOffset * 2]; // Copy samples for (int i = 0; i < nToCopy * 2; i++) { int sample = *(input++); sample = ClampInt(sample, -32768, 32767); *(output++) = sample; } mod->replayBufferOffset += nToCopy; mod->replayBufferSamples -= nToCopy; length -= nToCopy; } }