2019-11-11 21:52:07 +00:00
|
|
|
/*
|
|
|
|
** s_advsound.cpp
|
|
|
|
** Routines for managing SNDINFO lumps and ambient sounds
|
|
|
|
** Thid is a stripped down version only handling the music related settings.
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
** Copyright 1998-2008 Randy Heit
|
|
|
|
** All rights reserved.
|
|
|
|
**
|
|
|
|
** Redistribution and use in source and binary forms, with or without
|
|
|
|
** modification, are permitted provided that the following conditions
|
|
|
|
** are met:
|
|
|
|
**
|
|
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer.
|
|
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
|
|
** documentation and/or other materials provided with the distribution.
|
|
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
|
|
** derived from this software without specific prior written permission.
|
|
|
|
**
|
|
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
|
|
|
|
// HEADER FILES ------------------------------------------------------------
|
|
|
|
|
2021-10-30 08:51:03 +00:00
|
|
|
|
2019-11-11 21:52:07 +00:00
|
|
|
#include "c_dispatch.h"
|
|
|
|
#include "filesystem.h"
|
|
|
|
#include "v_text.h"
|
|
|
|
#include "s_music.h"
|
|
|
|
#include "sc_man.h"
|
2022-11-24 15:49:38 +00:00
|
|
|
#include "s_soundinternal.h"
|
2020-02-09 12:26:51 +00:00
|
|
|
#include <zmusic.h>
|
2019-11-11 21:52:07 +00:00
|
|
|
|
2020-04-12 06:07:48 +00:00
|
|
|
#include "raze_music.h"
|
|
|
|
|
2019-11-11 21:52:07 +00:00
|
|
|
// MACROS ------------------------------------------------------------------
|
|
|
|
|
|
|
|
enum SICommands
|
|
|
|
{
|
|
|
|
SI_MusicVolume,
|
|
|
|
SI_MidiDevice,
|
|
|
|
SI_MusicAlias,
|
2022-11-24 15:49:38 +00:00
|
|
|
SI_ConReserve,
|
2022-11-24 22:19:28 +00:00
|
|
|
SI_Alias,
|
|
|
|
SI_Limit,
|
|
|
|
SI_Singular
|
2019-11-11 21:52:07 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// This specifies whether Timidity or Windows playback is preferred for a certain song (only useful for Windows.)
|
|
|
|
extern MusicAliasMap MusicAliases;
|
|
|
|
extern MidiDeviceMap MidiDevices;
|
|
|
|
extern MusicVolumeMap MusicVolumes;
|
|
|
|
|
|
|
|
// EXTERNAL FUNCTION PROTOTYPES --------------------------------------------
|
|
|
|
|
|
|
|
extern bool IsFloat (const char *str);
|
|
|
|
|
|
|
|
// PUBLIC FUNCTION PROTOTYPES ----------------------------------------------
|
|
|
|
|
|
|
|
// PRIVATE FUNCTION PROTOTYPES ---------------------------------------------
|
|
|
|
|
|
|
|
static void S_AddSNDINFO (int lumpnum);
|
|
|
|
|
|
|
|
|
|
|
|
// PRIVATE DATA DEFINITIONS ------------------------------------------------
|
|
|
|
|
|
|
|
static const char *SICommandStrings[] =
|
|
|
|
{
|
|
|
|
"$musicvolume",
|
|
|
|
"$mididevice",
|
|
|
|
"$musicalias",
|
2022-11-24 15:49:38 +00:00
|
|
|
"$conreserve",
|
|
|
|
"$alias",
|
2022-11-24 22:19:28 +00:00
|
|
|
"$limit",
|
|
|
|
"$singular",
|
2019-11-11 21:52:07 +00:00
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
2022-11-24 22:19:28 +00:00
|
|
|
static const int DEFAULT_LIMIT = 6;
|
2019-11-11 21:52:07 +00:00
|
|
|
// CODE --------------------------------------------------------------------
|
|
|
|
|
2022-11-24 15:49:38 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
2022-11-24 22:19:28 +00:00
|
|
|
// S_AddSound
|
2022-11-24 15:49:38 +00:00
|
|
|
//
|
2022-11-24 22:19:28 +00:00
|
|
|
// If logical name is already in S_sfx, updates it to use the new sound
|
|
|
|
// lump. Otherwise, adds the new mapping by using S_AddSoundLump().
|
2022-11-24 15:49:38 +00:00
|
|
|
//==========================================================================
|
|
|
|
|
2022-11-24 22:19:28 +00:00
|
|
|
static FSoundID S_AddSound(const char* logicalname, int lumpnum, FScanner* sc)
|
2022-11-24 15:49:38 +00:00
|
|
|
{
|
2022-11-24 22:19:28 +00:00
|
|
|
FSoundID sfxid = soundEngine->FindSoundNoHash(logicalname);
|
2022-11-24 15:49:38 +00:00
|
|
|
|
2022-11-24 22:19:28 +00:00
|
|
|
if (sfxid.isvalid())
|
2022-11-24 15:49:38 +00:00
|
|
|
{ // If the sound has already been defined, change the old definition
|
2022-11-24 22:19:28 +00:00
|
|
|
auto sfx = soundEngine->GetWritableSfx(sfxid);
|
2022-11-24 15:49:38 +00:00
|
|
|
|
|
|
|
if (sfx->bRandomHeader)
|
|
|
|
{
|
|
|
|
FRandomSoundList* rnd = soundEngine->ResolveRandomSound(sfx);
|
|
|
|
rnd->Choices.Reset();
|
2022-11-24 20:27:08 +00:00
|
|
|
rnd->Owner = NO_SOUND;
|
2022-11-24 15:49:38 +00:00
|
|
|
}
|
2022-11-24 22:19:28 +00:00
|
|
|
sfx->lumpnum = lumpnum;
|
2022-11-24 15:49:38 +00:00
|
|
|
sfx->bRandomHeader = false;
|
|
|
|
sfx->link = sfxinfo_t::NO_LINK;
|
2022-11-24 22:19:28 +00:00
|
|
|
sfx->bTentative = false;
|
2022-11-24 15:49:38 +00:00
|
|
|
if (sfx->NearLimit == -1)
|
|
|
|
{
|
2022-11-24 22:19:28 +00:00
|
|
|
sfx->NearLimit = 6;
|
2022-11-24 15:49:38 +00:00
|
|
|
sfx->LimitRange = 256 * 256;
|
|
|
|
}
|
|
|
|
//sfx->PitchMask = CurrentPitchMask;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ // Otherwise, create a new definition.
|
2022-11-24 22:19:28 +00:00
|
|
|
sfxid = soundEngine->AddSoundLump(logicalname, lumpnum, 0);
|
2022-11-24 15:49:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return sfxid;
|
|
|
|
}
|
|
|
|
|
2022-11-24 22:19:28 +00:00
|
|
|
FSoundID S_AddSound(const char* logicalname, const char* lumpname, FScanner* sc)
|
2019-11-11 21:52:07 +00:00
|
|
|
{
|
2022-11-24 22:19:28 +00:00
|
|
|
int lump = fileSystem.CheckNumForFullName(lumpname, true, ns_sounds);
|
|
|
|
return S_AddSound(logicalname, lump, sc);
|
2019-11-11 21:52:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// S_AddSNDINFO
|
|
|
|
//
|
|
|
|
// Reads a SNDINFO and does what it says.
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
static void S_AddSNDINFO (int lump)
|
|
|
|
{
|
|
|
|
bool skipToEndIf;
|
|
|
|
TArray<uint32_t> list;
|
2022-11-24 22:19:28 +00:00
|
|
|
int wantassigns = -1;
|
2019-11-11 21:52:07 +00:00
|
|
|
|
|
|
|
FScanner sc(lump);
|
|
|
|
skipToEndIf = false;
|
|
|
|
|
|
|
|
while (sc.GetString ())
|
|
|
|
{
|
|
|
|
if (skipToEndIf)
|
|
|
|
{
|
|
|
|
if (sc.Compare ("$endif"))
|
|
|
|
{
|
|
|
|
skipToEndIf = false;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sc.String[0] == '$')
|
|
|
|
{ // Got a command
|
|
|
|
switch (sc.MatchString (SICommandStrings))
|
|
|
|
{
|
|
|
|
case SI_MusicVolume: {
|
|
|
|
sc.MustGetString();
|
|
|
|
FName musname (sc.String);
|
|
|
|
sc.MustGetFloat();
|
2021-05-11 23:58:42 +00:00
|
|
|
MusicVolumes[musname] = (float)sc.Float;
|
2019-11-11 21:52:07 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SI_MusicAlias: {
|
|
|
|
sc.MustGetString();
|
2021-12-24 08:56:02 +00:00
|
|
|
int mlump = fileSystem.FindFile(sc.String);
|
|
|
|
if (mlump < 0)
|
|
|
|
mlump = fileSystem.FindFile(FStringf("music/%s", sc.String));
|
|
|
|
if (mlump >= 0)
|
2019-11-11 21:52:07 +00:00
|
|
|
{
|
|
|
|
// do not set the alias if a later WAD defines its own music of this name
|
2021-12-24 08:56:02 +00:00
|
|
|
int file = fileSystem.GetFileContainer(mlump);
|
2019-11-11 21:52:07 +00:00
|
|
|
int sndifile = fileSystem.GetFileContainer(sc.LumpNum);
|
|
|
|
if (file > sndifile)
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FName alias = sc.String;
|
|
|
|
sc.MustGetString();
|
|
|
|
FName mapped = sc.String;
|
|
|
|
|
|
|
|
// only set the alias if the lump it maps to exists.
|
|
|
|
if (mapped == NAME_None || fileSystem.FindFile(sc.String) >= 0)
|
|
|
|
{
|
|
|
|
MusicAliases[alias] = mapped;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SI_MidiDevice: {
|
|
|
|
sc.MustGetString();
|
|
|
|
FName nm = sc.String;
|
|
|
|
FScanner::SavedPos save = sc.SavePos();
|
2021-12-30 09:30:21 +00:00
|
|
|
|
2019-11-11 21:52:07 +00:00
|
|
|
sc.SetCMode(true);
|
|
|
|
sc.MustGetString();
|
|
|
|
MidiDeviceSetting devset;
|
2020-02-09 12:26:51 +00:00
|
|
|
// Important: This needs to handle all the devices not present in ZMusic Lite to be able to load configs made for GZDoom.
|
|
|
|
// Also let the library handle the fallback so this can adjust automatically if the feature set gets extended.
|
2019-11-11 21:52:07 +00:00
|
|
|
if (sc.Compare("timidity")) devset.device = MDEV_TIMIDITY;
|
|
|
|
else if (sc.Compare("fmod") || sc.Compare("sndsys")) devset.device = MDEV_SNDSYS;
|
2020-02-09 12:26:51 +00:00
|
|
|
else if (sc.Compare("standard")) devset.device = MDEV_STANDARD;
|
2019-11-11 21:52:07 +00:00
|
|
|
else if (sc.Compare("opl")) devset.device = MDEV_OPL;
|
|
|
|
else if (sc.Compare("default")) devset.device = MDEV_DEFAULT;
|
|
|
|
else if (sc.Compare("fluidsynth")) devset.device = MDEV_FLUIDSYNTH;
|
2020-02-09 12:26:51 +00:00
|
|
|
else if (sc.Compare("gus")) devset.device = MDEV_GUS;
|
|
|
|
else if (sc.Compare("wildmidi")) devset.device = MDEV_WILDMIDI;
|
|
|
|
else if (sc.Compare("adl")) devset.device = MDEV_ADL;
|
|
|
|
else if (sc.Compare("opn")) devset.device = MDEV_OPN;
|
2019-11-11 21:52:07 +00:00
|
|
|
else sc.ScriptError("Unknown MIDI device %s\n", sc.String);
|
|
|
|
if (sc.CheckString(","))
|
|
|
|
{
|
|
|
|
sc.SetCMode(false);
|
|
|
|
sc.MustGetString();
|
|
|
|
devset.args = sc.String;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// This does not really do what one might expect, because the next token has already been parsed and can be a '$'.
|
|
|
|
// So in order to continue parsing without C-Mode, we need to reset and parse the last token again.
|
|
|
|
sc.SetCMode(false);
|
|
|
|
sc.RestorePos(save);
|
|
|
|
sc.MustGetString();
|
|
|
|
}
|
|
|
|
MidiDevices[nm] = devset;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-11-24 22:19:28 +00:00
|
|
|
case SI_Alias: {
|
|
|
|
// $alias <name of alias> <name of real sound>
|
|
|
|
FSoundID sfxfrom;
|
|
|
|
|
|
|
|
sc.MustGetString ();
|
|
|
|
sfxfrom = S_AddSound (sc.String, -1, &sc);
|
|
|
|
sc.MustGetString ();
|
|
|
|
auto sfx = soundEngine->GetWritableSfx(sfxfrom);
|
|
|
|
sfx->link = soundEngine->FindSoundTentative (sc.String);
|
|
|
|
sfx->NearLimit = -1; // Aliases must use the original sound's limit. (Can be changed later)
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SI_Limit: {
|
|
|
|
// $limit <logical name> <max channels> [<distance>]
|
|
|
|
FSoundID sfxfrom;
|
|
|
|
|
|
|
|
sc.MustGetString ();
|
|
|
|
sfxfrom = soundEngine->FindSoundTentative (sc.String);
|
|
|
|
sc.MustGetNumber ();
|
|
|
|
auto sfx = soundEngine->GetWritableSfx(sfxfrom);
|
|
|
|
sfx->NearLimit = min(max(sc.Number, 0), 255);
|
|
|
|
if (sc.CheckFloat())
|
|
|
|
{
|
|
|
|
sfx->LimitRange = float(sc.Float * sc.Float);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SI_Singular: {
|
|
|
|
// $singular <logical name>
|
|
|
|
FSoundID sfx;
|
|
|
|
|
|
|
|
sc.MustGetString ();
|
|
|
|
sfx = soundEngine->FindSoundTentative (sc.String, DEFAULT_LIMIT);
|
|
|
|
auto sfxp = soundEngine->GetWritableSfx(sfx);
|
|
|
|
sfxp->bSingular = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
2022-11-24 15:49:38 +00:00
|
|
|
case SI_ConReserve: {
|
2022-11-24 22:19:28 +00:00
|
|
|
// $conreserve <logical name> <resource id>
|
|
|
|
// Assigns a resource ID to the given sound.
|
2022-11-24 15:49:38 +00:00
|
|
|
sc.MustGetString();
|
2022-11-24 22:19:28 +00:00
|
|
|
FSoundID sfx = soundEngine->FindSoundTentative(sc.String, DEFAULT_LIMIT);
|
|
|
|
auto sfxp = soundEngine->GetWritableSfx(sfx);
|
|
|
|
|
|
|
|
sc.MustGetNumber();
|
|
|
|
sfxp->ResourceId = sc.Number;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
{ // Got a logical sound mapping
|
|
|
|
FString name (sc.String);
|
|
|
|
if (wantassigns == -1)
|
2022-11-24 15:49:38 +00:00
|
|
|
{
|
2022-11-24 22:19:28 +00:00
|
|
|
wantassigns = sc.CheckString("=");
|
2022-11-24 15:49:38 +00:00
|
|
|
}
|
2022-11-24 22:19:28 +00:00
|
|
|
else if (wantassigns)
|
|
|
|
{
|
|
|
|
sc.MustGetStringName("=");
|
|
|
|
}
|
|
|
|
|
|
|
|
sc.MustGetString ();
|
|
|
|
S_AddSound (name, sc.String, &sc);
|
2022-11-24 15:49:38 +00:00
|
|
|
}
|
|
|
|
|
2022-11-24 22:19:28 +00:00
|
|
|
}
|
2022-11-24 15:49:38 +00:00
|
|
|
|
2019-11-11 21:52:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-24 22:19:28 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// S_ParseSndInfo
|
|
|
|
//
|
|
|
|
// Parses all loaded SNDINFO lumps.
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void S_ParseSndInfo()
|
|
|
|
{
|
|
|
|
int lump;
|
|
|
|
|
|
|
|
soundEngine->Clear();
|
|
|
|
MusicAliases.Clear();
|
|
|
|
MidiDevices.Clear();
|
|
|
|
MusicVolumes.Clear();
|
|
|
|
|
|
|
|
S_AddSound("{ no sound }", "DSEMPTY", nullptr); // Sound 0 is no sound at all
|
|
|
|
|
|
|
|
int lastlump = 0;
|
|
|
|
while ((lump = fileSystem.FindLump("SNDINFO", &lastlump, false)) != -1)
|
|
|
|
{
|
|
|
|
S_AddSNDINFO(lump);
|
|
|
|
}
|
|
|
|
soundEngine->HashSounds();
|
|
|
|
}
|
|
|
|
|