mirror of
https://github.com/ZDoom/raze-gles.git
synced 2025-01-26 08:50:55 +00:00
- SW sound refactoring complete, not tested yet.
This commit is contained in:
parent
a087d566ee
commit
591ace496f
5 changed files with 210 additions and 347 deletions
|
@ -640,7 +640,7 @@ FSoundChan *SoundEngine::StartSound(int type, const void *source,
|
||||||
|
|
||||||
void SoundEngine::RestartChannel(FSoundChan *chan)
|
void SoundEngine::RestartChannel(FSoundChan *chan)
|
||||||
{
|
{
|
||||||
assert(chan->ChanFlags & CHANF_EVICTED);
|
if (!(chan->ChanFlags & CHANF_EVICTED)) return;
|
||||||
|
|
||||||
FSoundChan *ochan;
|
FSoundChan *ochan;
|
||||||
sfxinfo_t *sfx = &S_sfx[chan->SoundID];
|
sfxinfo_t *sfx = &S_sfx[chan->SoundID];
|
||||||
|
@ -1061,6 +1061,16 @@ void SoundEngine::ChangeSoundVolume(int sourcetype, const void *source, int chan
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SoundEngine::SetVolume(FSoundChan* chan, float volume)
|
||||||
|
{
|
||||||
|
if (volume < 0.0) volume = 0.0;
|
||||||
|
else if (volume > 1.0) volume = 1.0;
|
||||||
|
|
||||||
|
assert(chan != nullptr);
|
||||||
|
GSnd->ChannelVolume(chan, volume);
|
||||||
|
chan->Volume = volume;
|
||||||
|
}
|
||||||
|
|
||||||
//==========================================================================
|
//==========================================================================
|
||||||
//
|
//
|
||||||
// S_ChangeSoundPitch
|
// S_ChangeSoundPitch
|
||||||
|
@ -1165,6 +1175,31 @@ bool SoundEngine::IsSourcePlayingSomething (int sourcetype, const void *actor, i
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//==========================================================================
|
||||||
|
//
|
||||||
|
// S_EvictChannel
|
||||||
|
//
|
||||||
|
// Forcibly evicts one single channel.
|
||||||
|
//
|
||||||
|
//==========================================================================
|
||||||
|
|
||||||
|
void SoundEngine::EvictChannel(FSoundChan *chan)
|
||||||
|
{
|
||||||
|
if (!(chan->ChanFlags & CHANF_EVICTED))
|
||||||
|
{
|
||||||
|
chan->ChanFlags |= CHANF_EVICTED;
|
||||||
|
if (chan->SysChannel != NULL)
|
||||||
|
{
|
||||||
|
if (!(chan->ChanFlags & CHANF_ABSTIME))
|
||||||
|
{
|
||||||
|
chan->StartTime = GSnd ? GSnd->GetPosition(chan) : 0;
|
||||||
|
chan->ChanFlags |= CHANF_ABSTIME;
|
||||||
|
}
|
||||||
|
StopChannel(chan);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
//==========================================================================
|
//==========================================================================
|
||||||
//
|
//
|
||||||
// S_EvictAllChannels
|
// S_EvictAllChannels
|
||||||
|
@ -1181,21 +1216,7 @@ void SoundEngine::EvictAllChannels()
|
||||||
for (chan = Channels; chan != NULL; chan = next)
|
for (chan = Channels; chan != NULL; chan = next)
|
||||||
{
|
{
|
||||||
next = chan->NextChan;
|
next = chan->NextChan;
|
||||||
|
EvictChannel(chan);
|
||||||
if (!(chan->ChanFlags & CHANF_EVICTED))
|
|
||||||
{
|
|
||||||
chan->ChanFlags |= CHANF_EVICTED;
|
|
||||||
if (chan->SysChannel != NULL)
|
|
||||||
{
|
|
||||||
if (!(chan->ChanFlags & CHANF_ABSTIME))
|
|
||||||
{
|
|
||||||
chan->StartTime = GSnd ? GSnd->GetPosition(chan) : 0;
|
|
||||||
chan->ChanFlags |= CHANF_ABSTIME;
|
|
||||||
}
|
|
||||||
StopChannel(chan);
|
|
||||||
}
|
|
||||||
// assert(chan->NextChan == next);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1338,7 +1359,7 @@ void SoundEngine::ChannelEnded(FISoundChannel *ichan)
|
||||||
if (schan != NULL)
|
if (schan != NULL)
|
||||||
{
|
{
|
||||||
// If the sound was stopped with GSnd->StopSound(), then we know
|
// If the sound was stopped with GSnd->StopSound(), then we know
|
||||||
// it wasn't evicted. Otherwise, if it's looping, it must have
|
// it wasn't evicted. Otherwise, if it's looping or declared virtual, it must have
|
||||||
// been evicted. If it's not looping, then it was evicted if it
|
// been evicted. If it's not looping, then it was evicted if it
|
||||||
// didn't reach the end of its playback.
|
// didn't reach the end of its playback.
|
||||||
if (schan->ChanFlags & CHANF_FORGETTABLE)
|
if (schan->ChanFlags & CHANF_FORGETTABLE)
|
||||||
|
|
|
@ -174,8 +174,8 @@ struct FSoundChan : public FISoundChannel
|
||||||
uint8_t SourceType;
|
uint8_t SourceType;
|
||||||
float LimitRange;
|
float LimitRange;
|
||||||
const void *Source;
|
const void *Source;
|
||||||
float Point[3]; // Sound is not attached to any source.
|
float Point[3]; // Sound is not attached to any source, can also be used as auxiliary storage for sounds with a real source.
|
||||||
int UserData; // One word of data for user-specific data
|
int UserData[2]; // storage for user-specific data (Shadow Warrior's intermittent sounds use this as a counter.
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -217,7 +217,7 @@ enum // This cannot be remain as this, but for now it has to suffice.
|
||||||
SOURCE_Actor, // Sound is coming from an actor.
|
SOURCE_Actor, // Sound is coming from an actor.
|
||||||
SOURCE_Ambient, // Sound is coming from a blood ambient definition.
|
SOURCE_Ambient, // Sound is coming from a blood ambient definition.
|
||||||
SOURCE_Unattached, // Sound is not attached to any particular emitter.
|
SOURCE_Unattached, // Sound is not attached to any particular emitter.
|
||||||
SOURCE_Player, // SW player sound (player in SW is not connected to a sprite so needs to be special.)
|
SOURCE_Player, // SW player sound (player in SW maintains its own position separately from the sprite so needs to be special.)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -253,7 +253,6 @@ private:
|
||||||
void LinkChannel(FSoundChan* chan, FSoundChan** head);
|
void LinkChannel(FSoundChan* chan, FSoundChan** head);
|
||||||
void UnlinkChannel(FSoundChan* chan);
|
void UnlinkChannel(FSoundChan* chan);
|
||||||
void ReturnChannel(FSoundChan* chan);
|
void ReturnChannel(FSoundChan* chan);
|
||||||
void RestartChannel(FSoundChan* chan);
|
|
||||||
void RestoreEvictedChannel(FSoundChan* chan);
|
void RestoreEvictedChannel(FSoundChan* chan);
|
||||||
|
|
||||||
bool IsChannelUsed(int sourcetype, const void* actor, int channel, int* seen);
|
bool IsChannelUsed(int sourcetype, const void* actor, int channel, int* seen);
|
||||||
|
@ -273,6 +272,7 @@ protected:
|
||||||
|
|
||||||
public:
|
public:
|
||||||
virtual ~SoundEngine() = default;
|
virtual ~SoundEngine() = default;
|
||||||
|
void EvictChannel(FSoundChan *chan);
|
||||||
void EvictAllChannels();
|
void EvictAllChannels();
|
||||||
|
|
||||||
void StopChannel(FSoundChan* chan);
|
void StopChannel(FSoundChan* chan);
|
||||||
|
@ -289,8 +289,10 @@ public:
|
||||||
|
|
||||||
void StopAllChannels(void);
|
void StopAllChannels(void);
|
||||||
void SetPitch(FSoundChan* chan, float dpitch);
|
void SetPitch(FSoundChan* chan, float dpitch);
|
||||||
|
void SetVolume(FSoundChan* chan, float vol);
|
||||||
|
|
||||||
FSoundChan* GetChannel(void* syschan);
|
FSoundChan* GetChannel(void* syschan);
|
||||||
|
void RestartChannel(FSoundChan* chan);
|
||||||
void RestoreEvictedChannels();
|
void RestoreEvictedChannels();
|
||||||
void CalcPosVel(FSoundChan* chan, FVector3* pos, FVector3* vel);
|
void CalcPosVel(FSoundChan* chan, FVector3* pos, FVector3* vel);
|
||||||
|
|
||||||
|
|
|
@ -213,8 +213,6 @@ JS_SpriteSetup(void)
|
||||||
else if (tag == AMBIENT_SOUND)
|
else if (tag == AMBIENT_SOUND)
|
||||||
{
|
{
|
||||||
change_sprite_stat(SpriteNum, STAT_AMBIENT);
|
change_sprite_stat(SpriteNum, STAT_AMBIENT);
|
||||||
// PlaySound(sp->lotag, sp, v3df_ambient
|
|
||||||
// | v3df_init | v3df_doppler);
|
|
||||||
}
|
}
|
||||||
else if (tag == TAG_ECHO_SOUND)
|
else if (tag == TAG_ECHO_SOUND)
|
||||||
{
|
{
|
||||||
|
|
|
@ -101,6 +101,34 @@ AMB_INFO ambarray[] =
|
||||||
#define MAX_AMBIENT_SOUNDS 82
|
#define MAX_AMBIENT_SOUNDS 82
|
||||||
|
|
||||||
|
|
||||||
|
class SWSoundEngine : public SoundEngine
|
||||||
|
{
|
||||||
|
// client specific parts of the sound engine go in this class.
|
||||||
|
void CalcPosVel(int type, const void* source, const float pt[3], int channum, int chanflags, FSoundID chanSound, FVector3* pos, FVector3* vel, FSoundChan* chan) override;
|
||||||
|
TArray<uint8_t> ReadSound(int lumpnum);
|
||||||
|
|
||||||
|
public:
|
||||||
|
SWSoundEngine()
|
||||||
|
{
|
||||||
|
S_Rolloff.RolloffType = ROLLOFF_Doom;
|
||||||
|
S_Rolloff.MinDistance = 0; // These are the default values, SW uses a few different rolloff settings.
|
||||||
|
S_Rolloff.MaxDistance = 1187;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
//==========================================================================
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//==========================================================================
|
||||||
|
|
||||||
|
TArray<uint8_t> SWSoundEngine::ReadSound(int lumpnum)
|
||||||
|
{
|
||||||
|
auto wlump = fileSystem.OpenFileReader(lumpnum);
|
||||||
|
return wlump.Read();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
//==========================================================================
|
//==========================================================================
|
||||||
//
|
//
|
||||||
//
|
//
|
||||||
|
@ -109,6 +137,9 @@ AMB_INFO ambarray[] =
|
||||||
|
|
||||||
void InitFX(void)
|
void InitFX(void)
|
||||||
{
|
{
|
||||||
|
if (soundEngine) return; // just in case.
|
||||||
|
soundEngine = new SWSoundEngine;
|
||||||
|
|
||||||
auto &S_sfx = soundEngine->GetSounds();
|
auto &S_sfx = soundEngine->GetSounds();
|
||||||
S_sfx.Resize(countof(voc));
|
S_sfx.Resize(countof(voc));
|
||||||
for (auto& sfx : S_sfx) { sfx.Clear(); sfx.lumpnum = sfx_empty; }
|
for (auto& sfx : S_sfx) { sfx.Clear(); sfx.lumpnum = sfx_empty; }
|
||||||
|
@ -219,23 +250,11 @@ FRolloffInfo GetRolloff(int basedist)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//==========================================================================
|
||||||
|
//
|
||||||
|
//
|
||||||
extern USERp User[MAXSPRITES];
|
//
|
||||||
void DumpSounds(void);
|
//==========================================================================
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// Global vars used by ambient sounds to set spritenum of ambient sounds for later lookups in
|
|
||||||
// the sprite array so FAFcansee can know the sound sprite's current sector location
|
|
||||||
SWBOOL Use_SoundSpriteNum = FALSE;
|
|
||||||
int16_t SoundSpriteNum = -1; // Always set this back to -1 for proper validity checking!
|
|
||||||
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Play a sound
|
|
||||||
////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
int RandomizeAmbientSpecials(int handle)
|
int RandomizeAmbientSpecials(int handle)
|
||||||
{
|
{
|
||||||
|
@ -244,339 +263,161 @@ int RandomizeAmbientSpecials(int handle)
|
||||||
{
|
{
|
||||||
56,57,58,59,60,61,62,63,64,65,66,67
|
56,57,58,59,60,61,62,63,64,65,66,67
|
||||||
};
|
};
|
||||||
short i;
|
int i;
|
||||||
|
|
||||||
// If ambient sound is found in the array, randomly pick a new sound
|
// If ambient sound is found in the array, randomly pick a new sound
|
||||||
for (i = 0; i < MAXRNDAMB; i++)
|
for (i = 0; i < MAXRNDAMB; i++)
|
||||||
{
|
{
|
||||||
if (handle == ambrand[i])
|
if (handle == ambarray[ambrand[i]].diginame)
|
||||||
return ambrand[STD_RANDOM_RANGE(MAXRNDAMB - 1)];
|
return ambrand[STD_RANDOM_RANGE(MAXRNDAMB - 1)];
|
||||||
}
|
}
|
||||||
|
|
||||||
return handle; // Give back the sound, no new one was found
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//==========================================================================
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//==========================================================================
|
||||||
|
|
||||||
void
|
|
||||||
DoTimedSound(VOC3D_INFOp p)
|
void DoTimedSound(FSoundChan *chan)
|
||||||
{
|
{
|
||||||
p->tics += synctics;
|
chan->UserData[1] += synctics; // This got called 5x a second, incrementing by 3 each time, meaning that 15 units are one second. Now it gets called 40x a second.
|
||||||
|
|
||||||
if (p->tics >= p->maxtics)
|
if (chan->UserData[1] >= chan->UserData[0] * 8) // account for the 8x higher update frequency.
|
||||||
{
|
{
|
||||||
if (!FX_SoundValidAndActive(p->handle))
|
if (chan->ChanFlags & CHANF_EVICTED)
|
||||||
{
|
{
|
||||||
// Check for special case ambient sounds
|
// Check for special case ambient sounds. Since the sound is stopped and doesn't occupy a real channel at this time we can just swap out the sound ID before restarting it.
|
||||||
p->num = RandomizeAmbientSpecials(p->num);
|
int ambid = RandomizeAmbientSpecials(chan->SoundID);
|
||||||
|
if (ambid != -1)
|
||||||
// Sound was bumped from active sounds list, try to play again.
|
|
||||||
// Don't bother if voices are already maxed out.
|
|
||||||
if (FX_SoundsPlaying() < snd_numvoices)
|
|
||||||
{
|
{
|
||||||
if (p->flags & v3df_follow)
|
chan->SoundID = FSoundID(ambarray[ambid].maxtics);
|
||||||
{
|
chan->UserData[0] = STD_RANDOM_RANGE(ambarray[ambid].maxtics);
|
||||||
//PlaySound(p->num, p->x, p->y, p->z, p->flags); todo
|
|
||||||
p->deleted = TRUE; // Mark old sound for deletion
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
//PlaySound(p->num, &p->fx, &p->fy, &p->fz, p->flags); todo
|
|
||||||
p->deleted = TRUE; // Mark old sound for deletion
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
p->tics = 0;
|
soundEngine->RestartChannel(chan);
|
||||||
|
}
|
||||||
|
|
||||||
|
chan->UserData[1] = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
///////////////////////////////////////////////
|
//==========================================================================
|
||||||
// Main function to update 3D sound array
|
//
|
||||||
///////////////////////////////////////////////
|
//
|
||||||
typedef struct
|
//
|
||||||
|
//==========================================================================
|
||||||
|
|
||||||
|
void SWSoundEngine::CalcPosVel(int type, const void* source, const float pt[3], int channum, int chanflags, FSoundID chanSound, FVector3* pos, FVector3* vel, FSoundChan* chan)
|
||||||
{
|
{
|
||||||
VOC3D_INFOp p;
|
if (pos != nullptr)
|
||||||
short dist;
|
|
||||||
uint8_t priority;
|
|
||||||
} TVOC_INFO, * TVOC_INFOp;
|
|
||||||
|
|
||||||
void
|
|
||||||
DoUpdateSounds(void)
|
|
||||||
{
|
|
||||||
VOC3D_INFOp p;
|
|
||||||
SWBOOL looping;
|
|
||||||
int pitch = 0, pitchmax;
|
|
||||||
int delta;
|
|
||||||
short dist, angle;
|
|
||||||
SWBOOL deletesound = FALSE;
|
|
||||||
|
|
||||||
TVOC_INFO TmpVocArray[32];
|
|
||||||
int i;
|
|
||||||
|
|
||||||
// Zero out the temporary array
|
|
||||||
// Zero out the temporary array
|
|
||||||
//memset(&TmpVocArray[0],0,sizeof(TmpVocArray));
|
|
||||||
for (i = 0; i < 32; i++)
|
|
||||||
{
|
|
||||||
TmpVocArray[i].p = NULL;
|
|
||||||
TmpVocArray[i].dist = 0;
|
|
||||||
TmpVocArray[i].priority = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
p = nullptr;// voc3dstart;
|
|
||||||
|
|
||||||
while (p)
|
|
||||||
{
|
|
||||||
ASSERT(p->num >= 0 && p->num < DIGI_MAX);
|
|
||||||
|
|
||||||
looping = p->vp->voc_flags & vf_loop;
|
|
||||||
|
|
||||||
// //DSPRINTF(ds,"sound %d FX_SoundActive = %d\n,",p->num,FX_SoundActive(p->handle));
|
|
||||||
// MONO_PRINT(ds);
|
|
||||||
|
|
||||||
// If sprite owner is dead, kill this sound as long as it isn't ambient
|
|
||||||
if (looping && p->owner == -1 && !TEST(p->flags, v3df_ambient))
|
|
||||||
{
|
|
||||||
SET(p->flags, v3df_kill);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Is the sound slated for death? Kill it, otherwise play it.
|
|
||||||
if (p->flags & v3df_kill)
|
|
||||||
{
|
|
||||||
if (FX_SoundValidAndActive(p->handle))
|
|
||||||
{
|
|
||||||
FX_StopSound(p->handle); // Make sure to stop active sounds
|
|
||||||
p->handle = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
//DSPRINTF(ds,"%d had v3df_kill.\n",p->num);
|
|
||||||
//MONO_PRINT(ds);
|
|
||||||
p->deleted = TRUE;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (!FX_SoundValidAndActive(p->handle) && !looping)
|
|
||||||
{
|
|
||||||
if (p->flags & v3df_intermit)
|
|
||||||
{
|
|
||||||
DoTimedSound(p);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
//if(p->owner == -1 && !TEST(p->flags,v3df_ambient))
|
|
||||||
{
|
|
||||||
//DSPRINTF(ds,"%d is now inactive.\n",p->num);
|
|
||||||
//MONO_PRINT(ds);
|
|
||||||
p->deleted = TRUE;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (FX_SoundValidAndActive(p->handle))
|
|
||||||
{
|
|
||||||
if (p->flags & v3df_follow)
|
|
||||||
{
|
|
||||||
dist = SoundDist(*p->x, *p->y, *p->z, p->vp->voc_distance);
|
|
||||||
angle = SoundAngle(*p->x, *p->y);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (p->fx == 0 && p->fy == 0 && p->fz == 0)
|
|
||||||
dist = 0;
|
|
||||||
else
|
|
||||||
dist = SoundDist(p->fx, p->fy, p->fz, p->vp->voc_distance);
|
|
||||||
angle = SoundAngle(p->fx, p->fy);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Can the ambient sound see the player? If not, tone it down some.
|
|
||||||
if ((p->vp->voc_flags & vf_loop) && p->owner != -1)
|
|
||||||
{
|
{
|
||||||
PLAYERp pp = Player + screenpeek;
|
PLAYERp pp = Player + screenpeek;
|
||||||
SPRITEp sp = &sprite[p->owner];
|
FVector3 campos = GetSoundPos((vec3_t*)pp);
|
||||||
|
|
||||||
//MONO_PRINT("Checking sound cansee");
|
//S_GetCamera(&campos, nullptr, &camsect);
|
||||||
if (!FAFcansee(sp->x, sp->y, sp->z, sp->sectnum, pp->posx, pp->posy, pp->posz, pp->cursectnum))
|
if (vel) vel->Zero();
|
||||||
|
|
||||||
|
if (type == SOURCE_Unattached)
|
||||||
{
|
{
|
||||||
//MONO_PRINT("Reducing sound distance");
|
pos->X = pt[0];
|
||||||
dist += ((dist / 2) + (dist / 4)); // Play more quietly
|
pos->Y = pt[1];
|
||||||
if (dist > 255) dist = 255;
|
pos->Z = pt[2];
|
||||||
|
}
|
||||||
// Special cases
|
else if (type == SOURCE_Actor || type == SOURCE_Player)
|
||||||
if (p->num == 76 && TEST(p->flags, v3df_ambient))
|
|
||||||
{
|
{
|
||||||
dist = 255; // Cut off whipping sound, it's secret
|
vec3_t* vpos = type == SOURCE_Actor ? &((SPRITEp)source)->pos : (vec3_t*)&((PLAYERp)source)->posx;
|
||||||
}
|
FVector3 npos = GetSoundPos(vpos);
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (dist >= 255 && p->vp->voc_distance == DIST_NORMAL)
|
|
||||||
{
|
|
||||||
FX_StopSound(p->handle); // Make sure to stop active
|
|
||||||
p->handle = 0;
|
|
||||||
// sounds
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Handle Panning Left and Right
|
|
||||||
if (!(p->flags & v3df_dontpan))
|
|
||||||
FX_Pan3D(p->handle, angle, dist);
|
|
||||||
else
|
|
||||||
FX_Pan3D(p->handle, 0, dist);
|
|
||||||
|
|
||||||
// Handle Doppler Effects
|
|
||||||
#define DOPPLERMAX 400
|
|
||||||
if (!(p->flags & v3df_doppler) && FX_SoundActive(p->handle))
|
|
||||||
{
|
|
||||||
pitch -= (dist - p->doplr_delta);
|
|
||||||
|
|
||||||
if (p->vp->pitch_lo != 0 && p->vp->pitch_hi != 0)
|
|
||||||
{
|
|
||||||
if (abs(p->vp->pitch_lo) > abs(p->vp->pitch_hi))
|
|
||||||
pitchmax = abs(p->vp->pitch_lo);
|
|
||||||
else
|
|
||||||
pitchmax = abs(p->vp->pitch_hi);
|
|
||||||
|
|
||||||
}
|
|
||||||
else
|
|
||||||
pitchmax = DOPPLERMAX;
|
|
||||||
|
|
||||||
if (pitch > pitchmax)
|
|
||||||
pitch = pitchmax;
|
|
||||||
if (pitch < -pitchmax)
|
|
||||||
pitch = -pitchmax;
|
|
||||||
|
|
||||||
p->doplr_delta = dist; // Save new distance to
|
|
||||||
// struct
|
|
||||||
FX_SetPitch(p->handle, pitch);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (!FX_SoundValidAndActive(p->handle) && looping)
|
|
||||||
{
|
|
||||||
if (p->flags & v3df_follow)
|
|
||||||
{
|
|
||||||
dist = SoundDist(*p->x, *p->y, *p->z, p->vp->voc_distance);
|
|
||||||
angle = SoundAngle(*p->x, *p->y);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
dist = SoundDist(p->fx, p->fy, p->fz, p->vp->voc_distance);
|
|
||||||
angle = SoundAngle(p->fx, p->fy);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sound was bumped from active sounds list, try to play
|
|
||||||
// again.
|
|
||||||
// Don't bother if voices are already maxed out.
|
|
||||||
// Sort looping vocs in order of priority and distance
|
|
||||||
//if (FX_SoundsPlaying() < snd_numvoices && dist <= 255)
|
|
||||||
if (dist <= 255)
|
|
||||||
{
|
|
||||||
for (i = 0; i < min((int)SIZ(TmpVocArray), *snd_numvoices); i++)
|
|
||||||
{
|
|
||||||
if (p->priority >= TmpVocArray[i].priority)
|
|
||||||
{
|
|
||||||
if (!TmpVocArray[i].p || dist < TmpVocArray[i].dist)
|
|
||||||
{
|
|
||||||
ASSERT(p->num >= 0 && p->num < DIGI_MAX);
|
|
||||||
TmpVocArray[i].p = p;
|
|
||||||
TmpVocArray[i].dist = dist;
|
|
||||||
TmpVocArray[i].priority = p->priority;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} // !FX_SoundActive
|
|
||||||
} // if(p->flags & v3df_kill)
|
|
||||||
|
|
||||||
p = p->next;
|
|
||||||
} // while(p)
|
|
||||||
|
|
||||||
// Process all the looping sounds that said they wanted to get back in
|
|
||||||
// Only update these sounds 5x per second! Woo hoo!, aren't we optimized now?
|
|
||||||
//if(MoveSkip8==0)
|
|
||||||
// {
|
|
||||||
for (i = 0; i < min((int)SIZ(TmpVocArray), *snd_numvoices); i++)
|
|
||||||
{
|
|
||||||
int handle;
|
|
||||||
|
|
||||||
p = TmpVocArray[i].p;
|
|
||||||
|
|
||||||
//if (FX_SoundsPlaying() >= snd_numvoices || !p) break;
|
|
||||||
if (!p) break;
|
|
||||||
|
|
||||||
ASSERT(p->num >= 0 && p->num < DIGI_MAX);
|
|
||||||
|
|
||||||
if (p->flags & v3df_follow)
|
|
||||||
{
|
|
||||||
if (p->owner == -1)
|
|
||||||
{
|
|
||||||
// Terminate the sound without aborting.
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
Use_SoundSpriteNum = TRUE;
|
|
||||||
SoundSpriteNum = p->owner;
|
|
||||||
|
|
||||||
//handle = PlaySound(p->num, p->x, p->y, p->z, p->flags); todo
|
|
||||||
//if(handle >= 0 || TEST(p->flags,v3df_ambient)) // After a valid PlaySound, it's ok to use voc3dend
|
|
||||||
//voc3dend->owner = p->owner; // todo Transfer the owner
|
|
||||||
p->deleted = TRUE;
|
|
||||||
|
|
||||||
Use_SoundSpriteNum = FALSE;
|
|
||||||
SoundSpriteNum = -1;
|
|
||||||
|
|
||||||
//MONO_PRINT("TmpVocArray playing a follow sound");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (p->owner == -1)
|
|
||||||
{
|
|
||||||
// Terminate the sound without aborting.
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
Use_SoundSpriteNum = TRUE;
|
|
||||||
SoundSpriteNum = p->owner;
|
|
||||||
|
|
||||||
//handle = PlaySound(p->num, &p->fx, &p->fy, &p->fz, p->flags); todo
|
|
||||||
//if(handle >= 0 || TEST(p->flags,v3df_ambient))
|
|
||||||
//voc3dend->owner = p->owner; // todo Transfer the owner
|
|
||||||
p->deleted = TRUE;
|
|
||||||
|
|
||||||
Use_SoundSpriteNum = FALSE;
|
|
||||||
SoundSpriteNum = -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
|
|
||||||
|
|
||||||
// Can the ambient sound see the player? If not, tone it down some.
|
// Can the ambient sound see the player? If not, tone it down some.
|
||||||
if ((vp->voc_flags & vf_loop) && Use_SoundSpriteNum && SoundSpriteNum >= 0)
|
if ((chanflags & CHANF_LOOP) && (chanflags & CHANEXF_AMBIENT) && type == SOURCE_Actor)
|
||||||
{
|
{
|
||||||
PLAYERp pp = Player+screenpeek;
|
auto sp = (SPRITEp)source;
|
||||||
|
|
||||||
//MONO_PRINT("PlaySound:Checking sound cansee");
|
//MONO_PRINT("PlaySound:Checking sound cansee");
|
||||||
if (!FAFcansee(tx, ty, tz, sp->sectnum,pp->posx, pp->posy, pp->posz, pp->cursectnum))
|
if (!FAFcansee(vpos->x, vpos->y, vpos->z, sp->sectnum, pp->posx, pp->posy, pp->posz, pp->cursectnum))
|
||||||
{
|
{
|
||||||
//MONO_PRINT("PlaySound:Reducing sound distance");
|
//MONO_PRINT("PlaySound:Reducing sound distance");
|
||||||
sound_dist += ((sound_dist/2)+(sound_dist/4)); // Play more quietly
|
auto distvec = npos - campos;
|
||||||
if (sound_dist > 255) sound_dist = 255;
|
|
||||||
|
|
||||||
// Special Cases
|
// Special Cases
|
||||||
if (num == DIGI_WHIPME) sound_dist = 255;
|
npos = campos + distvec * 1.75f; // Play more quietly
|
||||||
|
chanflags |= CHANEXF_NODOPPLER; // Ambient sounds should be stationary, but let's make sure that no doppler gets applied after messing up the position.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
*/
|
|
||||||
|
|
||||||
|
if (chanflags & CHANEXF_DONTPAN)
|
||||||
|
{
|
||||||
|
// For unpanned sounds the volume must be set directly and the position taken from the listener.
|
||||||
|
*pos = campos;
|
||||||
|
auto sdist = SoundDist(vpos->x, vpos->y, vpos->z, voc[chanSound].voc_distance);
|
||||||
|
SetVolume(chan, (255 - sdist) * (1 / 255.f));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
*pos = npos;
|
||||||
|
// Doppler only makes sense for sounds that are actually positioned in the world (i.e. not in combination with DONTPAN)
|
||||||
|
if (!(chanflags & CHANEXF_NODOPPLER) && vel)
|
||||||
|
{
|
||||||
|
// Hack alert. Velocity may only be set if a) the sound is already running and b) an actual sound channel is modified.
|
||||||
|
// It remains to be seen if this is actually workable. I have my doubts. The velocity should be taken from a stable source.
|
||||||
|
if (chan && !(chanflags & (CHANF_JUSTSTARTED|CHANF_EVICTED)))
|
||||||
|
{
|
||||||
|
*vel = (npos - FVector3(pt[0], pt[1], pt[2])) * 40; // SW ticks 40 times a second.
|
||||||
|
chan->Point[0] = npos.X;
|
||||||
|
chan->Point[1] = npos.Y;
|
||||||
|
chan->Point[2] = npos.Z;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ((chanflags & CHANF_LISTENERZ) && campos != nullptr && type != SOURCE_None)
|
||||||
|
{
|
||||||
|
pos->Y = campos.Y;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//==========================================================================
|
||||||
|
//
|
||||||
|
// Main function to update 3D sound array
|
||||||
|
//
|
||||||
|
//==========================================================================
|
||||||
|
|
||||||
|
void DoUpdateSounds(void)
|
||||||
|
{
|
||||||
|
soundEngine->EnumerateChannels([](FSoundChan* chan)
|
||||||
|
{
|
||||||
|
if (chan->ChanFlags & EChanFlags::FromInt(CHANEXF_INTERMIT))
|
||||||
|
{
|
||||||
|
DoTimedSound(chan);
|
||||||
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////
|
if (chan->SoundID == DIGI_WHIPME && chan->SourceType == SOURCE_Actor)
|
||||||
////////////////////////////////////////////////////////////////////////////
|
{
|
||||||
|
auto sp = (SPRITEp)chan->Source;
|
||||||
|
PLAYERp pp = Player + screenpeek;
|
||||||
|
if (!FAFcansee(sp->pos.x, sp->pos.y, sp->pos.z, sp->sectnum, pp->posx, pp->posy, pp->posz, pp->cursectnum))
|
||||||
|
{
|
||||||
|
soundEngine->EvictChannel(chan);
|
||||||
|
}
|
||||||
|
else if (chan->ChanFlags & CHANF_EVICTED)
|
||||||
|
{
|
||||||
|
soundEngine->RestartChannel(chan);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
soundEngine->UpdateSounds((int)totalclock);
|
||||||
|
}
|
||||||
|
|
||||||
|
//==========================================================================
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//==========================================================================
|
||||||
|
|
||||||
float S_ConvertPitch(int lpitch)
|
float S_ConvertPitch(int lpitch)
|
||||||
{
|
{
|
||||||
|
@ -592,7 +433,6 @@ float S_ConvertPitch(int lpitch)
|
||||||
int _PlaySound(int num, SPRITEp sp, PLAYERp pp, vec3_t *pos, Voc3D_Flags flags, int channel)
|
int _PlaySound(int num, SPRITEp sp, PLAYERp pp, vec3_t *pos, Voc3D_Flags flags, int channel)
|
||||||
{
|
{
|
||||||
VOC_INFOp vp;
|
VOC_INFOp vp;
|
||||||
VOC3D_INFOp v3p;
|
|
||||||
int pitch = 0;
|
int pitch = 0;
|
||||||
short angle, sound_dist;
|
short angle, sound_dist;
|
||||||
int tx, ty, tz;
|
int tx, ty, tz;
|
||||||
|
@ -682,10 +522,12 @@ int _PlaySound(int num, SPRITEp sp, PLAYERp pp, vec3_t *pos, Voc3D_Flags flags,
|
||||||
{
|
{
|
||||||
cflags |= EChanFlags::FromInt(CHANEXF_DONTPAN); // beware of hackery to emulate this.
|
cflags |= EChanFlags::FromInt(CHANEXF_DONTPAN); // beware of hackery to emulate this.
|
||||||
}
|
}
|
||||||
|
/*
|
||||||
if (flags & v3df_init)
|
if (flags & v3df_init)
|
||||||
{
|
{
|
||||||
cflags |= CHANF_VIRTUAL; // don't start right away but keep the channel around until explicitly deleted.
|
// this only gets used for starting looped sounds that are outside hearing range - something the sound engine handles transparently.
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
if (vp->voc_flags & vf_loop)
|
if (vp->voc_flags & vf_loop)
|
||||||
{
|
{
|
||||||
cflags |= CHANF_LOOP; // with the new sound engine these can just be started and don't have to be stopped ever.
|
cflags |= CHANF_LOOP; // with the new sound engine these can just be started and don't have to be stopped ever.
|
||||||
|
@ -707,7 +549,8 @@ int _PlaySound(int num, SPRITEp sp, PLAYERp pp, vec3_t *pos, Voc3D_Flags flags,
|
||||||
{
|
{
|
||||||
chan->ChanFlags |= CHANF_VIRTUAL | EChanFlags::FromInt(CHANEXF_INTERMIT); // for intermittent sounds this must be set after starting the sound so that it actually plays.
|
chan->ChanFlags |= CHANF_VIRTUAL | EChanFlags::FromInt(CHANEXF_INTERMIT); // for intermittent sounds this must be set after starting the sound so that it actually plays.
|
||||||
}
|
}
|
||||||
chan->UserData = maxtics; // counter for intermittent delay.
|
chan->UserData[0] = maxtics; // counter for intermittent delay.
|
||||||
|
chan->UserData[1] = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -766,14 +609,14 @@ void DeleteNoSoundOwner(short spritenum)
|
||||||
//==========================================================================
|
//==========================================================================
|
||||||
//
|
//
|
||||||
// This is called from KillSprite to kill a follow sound with no valid sprite owner
|
// This is called from KillSprite to kill a follow sound with no valid sprite owner
|
||||||
// Stops and active sound with the follow bit set, even play once sounds.
|
// Stops any active sound with the follow bit set, even play once sounds.
|
||||||
//
|
//
|
||||||
//==========================================================================
|
//==========================================================================
|
||||||
|
|
||||||
void DeleteNoFollowSoundOwner(short spritenum)
|
void DeleteNoFollowSoundOwner(short spritenum)
|
||||||
{
|
{
|
||||||
SPRITEp sp = &sprite[spritenum];
|
SPRITEp sp = &sprite[spritenum];
|
||||||
soundEngine->StopSound(SOURCE_Actor, sp, -1);
|
soundEngine->StopSound(SOURCE_Actor, sp, -1); // all non-follow sounds are SOURCE_Unattached
|
||||||
}
|
}
|
||||||
|
|
||||||
//==========================================================================
|
//==========================================================================
|
||||||
|
@ -803,7 +646,6 @@ void StopAmbientSound(void)
|
||||||
|
|
||||||
void StartAmbientSound(void)
|
void StartAmbientSound(void)
|
||||||
{
|
{
|
||||||
VOC3D_INFOp p;
|
|
||||||
short i, nexti;
|
short i, nexti;
|
||||||
extern SWBOOL InMenuLevel;
|
extern SWBOOL InMenuLevel;
|
||||||
|
|
||||||
|
|
|
@ -58,21 +58,19 @@ enum
|
||||||
v3df_init = 64, // 1 = First pass of sound, don't play it.
|
v3df_init = 64, // 1 = First pass of sound, don't play it.
|
||||||
// This is mainly used for intermittent sounds
|
// This is mainly used for intermittent sounds
|
||||||
v3df_nolookup = 128, // don't use ambient table lookup
|
v3df_nolookup = 128, // don't use ambient table lookup
|
||||||
v3df_listenerz = 256 // ignore height when positioning sound (was mostly hacked by providing bad cooordinates to the code before.)
|
|
||||||
};
|
};
|
||||||
typedef int Voc3D_Flags;
|
typedef int Voc3D_Flags;
|
||||||
|
|
||||||
struct VOCstruct;
|
struct VOCstruct;
|
||||||
typedef struct VOCstruct VOC_INFO, *VOC_INFOp;
|
typedef struct VOCstruct VOC_INFO, *VOC_INFOp;
|
||||||
|
#if 0
|
||||||
struct VOC3Dstruct;
|
struct VOC3Dstruct;
|
||||||
typedef struct VOC3Dstruct VOC3D_INFO, *VOC3D_INFOp;
|
typedef struct VOC3Dstruct VOC3D_INFO, *VOC3D_INFOp;
|
||||||
|
#endif
|
||||||
struct ambientstruct;
|
struct ambientstruct;
|
||||||
typedef struct ambientstruct AMB_INFO, *AMB_INFOp;
|
typedef struct ambientstruct AMB_INFO, *AMB_INFOp;
|
||||||
|
|
||||||
|
|
||||||
extern VOC3D_INFOp voc3dstart;
|
|
||||||
extern VOC3D_INFOp voc3dend;
|
|
||||||
|
|
||||||
void DoUpdateSounds(void);
|
void DoUpdateSounds(void);
|
||||||
void Terminate3DSounds(void);
|
void Terminate3DSounds(void);
|
||||||
|
|
||||||
|
@ -151,6 +149,7 @@ struct VOCstruct
|
||||||
|
|
||||||
// JIMSOUND3D(tm) variables section //////////////////////////////////////////
|
// JIMSOUND3D(tm) variables section //////////////////////////////////////////
|
||||||
|
|
||||||
|
#if 0
|
||||||
struct VOC3Dstruct
|
struct VOC3Dstruct
|
||||||
{
|
{
|
||||||
VOC_INFOp vp; // Pointer to the sound
|
VOC_INFOp vp; // Pointer to the sound
|
||||||
|
@ -175,6 +174,7 @@ struct VOC3Dstruct
|
||||||
SWBOOL deleted; // Has sound been marked for deletion?
|
SWBOOL deleted; // Has sound been marked for deletion?
|
||||||
SWBOOL FX_Ok; // Did this sound play ok?
|
SWBOOL FX_Ok; // Did this sound play ok?
|
||||||
};
|
};
|
||||||
|
#endif
|
||||||
|
|
||||||
extern VOC_INFO voc[];
|
extern VOC_INFO voc[];
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue