Dump and load correct volume and pitch values for T6 sound aliases

This commit is contained in:
Jbleezy 2024-10-01 14:29:28 -07:00
parent cdc65e0425
commit 0ea55810ab
3 changed files with 88 additions and 10 deletions

View File

@ -3,12 +3,15 @@
#include "T6.h" #include "T6.h"
#include <cctype> #include <cctype>
#include <limits>
namespace T6 namespace T6
{ {
class Common class Common
{ {
public: public:
static constexpr double AMP_RATIO = 100000.0 / std::numeric_limits<uint16_t>::max();
static constexpr int Com_HashKey(const char* str, const int maxLen) static constexpr int Com_HashKey(const char* str, const int maxLen)
{ {
if (str == nullptr) if (str == nullptr)

View File

@ -8,6 +8,7 @@
#include "Pool/GlobalAssetPool.h" #include "Pool/GlobalAssetPool.h"
#include "Utils/StringUtils.h" #include "Utils/StringUtils.h"
#include <cmath>
#include <cstring> #include <cstring>
#include <fstream> #include <fstream>
#include <iostream> #include <iostream>
@ -113,7 +114,17 @@ unsigned int GetAliasSubListCount(const unsigned int startRow, const ParsedCsv&
return count; return count;
} }
bool LoadSoundAlias(MemoryManager* memory, SndAlias* alias, const ParsedCsvRow& row) double DecibelsToAmp(double decibels)
{
return std::pow(10.0, decibels / 20.0);
}
double CentsToHertz(double cents)
{
return std::numeric_limits<int16_t>::max() * std::pow(2, cents / 1200.0);
}
bool LoadSoundAlias(MemoryManager* memory, SndAlias* alias, const ParsedCsvRow& row, const unsigned int rowNum)
{ {
memset(alias, 0, sizeof(SndAlias)); memset(alias, 0, sizeof(SndAlias));
@ -141,15 +152,55 @@ bool LoadSoundAlias(MemoryManager* memory, SndAlias* alias, const ParsedCsvRow&
alias->duck = Common::SND_HashName(row.GetValue("duck").data()); alias->duck = Common::SND_HashName(row.GetValue("duck").data());
alias->volMin = row.GetValueInt<decltype(alias->volMin)>("vol_min"); const auto volMinDecibels = row.GetValueFloat("vol_min");
alias->volMax = row.GetValueInt<decltype(alias->volMax)>("vol_max");
if (volMinDecibels < 0.0f || volMinDecibels > 100.0f)
{
std::cerr << std::format("Invalid value for row {} col 'vol_min' - {} [0.0, 100.0]\n", rowNum + 1, volMinDecibels);
return false;
}
const auto volMinAmp = static_cast<int>(std::round(DecibelsToAmp(volMinDecibels) / T6::Common::AMP_RATIO));
alias->volMin = volMinAmp;
const auto volMaxDecibels = row.GetValueFloat("vol_max");
if (volMaxDecibels < 0.0f || volMaxDecibels > 100.0f)
{
std::cerr << std::format("Invalid value for row {} col 'vol_max' - {} [0.0, 100.0]\n", rowNum + 1, volMaxDecibels);
return false;
}
const auto volMaxAmp = static_cast<int>(std::round(DecibelsToAmp(volMaxDecibels) / T6::Common::AMP_RATIO));
alias->volMax = volMaxAmp;
const auto pitchMinCents = row.GetValueFloat("pitch_min");
if (pitchMinCents < -2400.0f || pitchMinCents > 1200.0f)
{
std::cerr << std::format("Invalid value for row {} col 'pitch_min' - {} [-2400.0, 1200.0]\n", rowNum + 1, pitchMinCents);
return false;
}
const auto pitchMinHertz = static_cast<int>(std::round(CentsToHertz(pitchMinCents)));
alias->pitchMin = pitchMinHertz;
const auto pitchMaxCents = row.GetValueFloat("pitch_max");
if (pitchMaxCents < -2400.0f || pitchMaxCents > 1200.0f)
{
std::cerr << std::format("Invalid value for row {} col 'pitch_max' - {} [-2400.0, 1200.0]\n", rowNum + 1, pitchMaxCents);
return false;
}
const auto pitchMaxHertz = static_cast<int>(std::round(CentsToHertz(pitchMaxCents)));
alias->pitchMax = pitchMaxHertz;
alias->distMin = row.GetValueInt<decltype(alias->distMin)>("dist_min"); alias->distMin = row.GetValueInt<decltype(alias->distMin)>("dist_min");
alias->distMax = row.GetValueInt<decltype(alias->distMax)>("dist_max"); alias->distMax = row.GetValueInt<decltype(alias->distMax)>("dist_max");
alias->distReverbMax = row.GetValueInt<decltype(alias->distReverbMax)>("dist_reverb_max"); alias->distReverbMax = row.GetValueInt<decltype(alias->distReverbMax)>("dist_reverb_max");
alias->limitCount = row.GetValueInt<decltype(alias->limitCount)>("limit_count"); alias->limitCount = row.GetValueInt<decltype(alias->limitCount)>("limit_count");
alias->entityLimitCount = row.GetValueInt<decltype(alias->entityLimitCount)>("entity_limit_count"); alias->entityLimitCount = row.GetValueInt<decltype(alias->entityLimitCount)>("entity_limit_count");
alias->pitchMin = row.GetValueInt<decltype(alias->pitchMin)>("pitch_min");
alias->pitchMax = row.GetValueInt<decltype(alias->pitchMax)>("pitch_max");
alias->minPriority = row.GetValueInt<decltype(alias->minPriority)>("min_priority"); alias->minPriority = row.GetValueInt<decltype(alias->minPriority)>("min_priority");
alias->maxPriority = row.GetValueInt<decltype(alias->maxPriority)>("max_priority"); alias->maxPriority = row.GetValueInt<decltype(alias->maxPriority)>("max_priority");
alias->minPriorityThreshold = row.GetValueInt<decltype(alias->minPriorityThreshold)>("min_priority_threshold"); alias->minPriorityThreshold = row.GetValueInt<decltype(alias->minPriorityThreshold)>("min_priority_threshold");
@ -292,7 +343,7 @@ bool LoadSoundAliasList(
// list are next to each other in the file // list are next to each other in the file
for (auto i = 0u; i < subListCount; i++) for (auto i = 0u; i < subListCount; i++)
{ {
if (!LoadSoundAlias(memory, &sndBank->alias[listIndex].head[i], aliasCsv[row])) if (!LoadSoundAlias(memory, &sndBank->alias[listIndex].head[i], aliasCsv[row], row))
return false; return false;
// if this asset is loaded instead of stream, increment the loaded count for later // if this asset is loaded instead of stream, increment the loaded count for later

View File

@ -7,6 +7,7 @@
#include "Sound/WavWriter.h" #include "Sound/WavWriter.h"
#include "nlohmann/json.hpp" #include "nlohmann/json.hpp"
#include <cmath>
#include <filesystem> #include <filesystem>
#include <format> #include <format>
#include <fstream> #include <fstream>
@ -225,6 +226,21 @@ class AssetDumperSndBank::Internal
} }
} }
static double AmpToDecibels(double amp)
{
if (amp == 0.0)
{
return 0.0;
}
return 20.0 * std::log10(amp);
}
static double HertzToCents(double hertz)
{
return 1200.0 * std::log2(hertz / std::numeric_limits<int16_t>::max());
}
static void WriteAliasToFile(CsvOutputStream& stream, const SndAlias* alias, const std::optional<snd_asset_format> maybeFormat, const SndBank* bank) static void WriteAliasToFile(CsvOutputStream& stream, const SndAlias* alias, const std::optional<snd_asset_format> maybeFormat, const SndBank* bank)
{ {
// name // name
@ -247,10 +263,14 @@ class AssetDumperSndBank::Internal
stream.WriteColumn(SOUND_GROUPS[alias->flags.volumeGroup]); stream.WriteColumn(SOUND_GROUPS[alias->flags.volumeGroup]);
// vol_min // vol_min
stream.WriteColumn(std::to_string(alias->volMin)); const auto volMinAmp = alias->volMin * T6::Common::AMP_RATIO;
const auto volMinDecibels = static_cast<int>(std::round(AmpToDecibels(volMinAmp)));
stream.WriteColumn(std::to_string(volMinDecibels));
// vol_max // vol_max
stream.WriteColumn(std::to_string(alias->volMax)); const auto volMaxAmp = alias->volMax * T6::Common::AMP_RATIO;
const auto volMaxDecibels = static_cast<int>(std::round(AmpToDecibels(volMaxAmp)));
stream.WriteColumn(std::to_string(volMaxDecibels));
// team_vol_mod // team_vol_mod
stream.WriteColumn(""); stream.WriteColumn("");
@ -289,10 +309,14 @@ class AssetDumperSndBank::Internal
stream.WriteColumn(SOUND_LIMIT_TYPES[alias->flags.entityLimitType]); stream.WriteColumn(SOUND_LIMIT_TYPES[alias->flags.entityLimitType]);
// pitch_min // pitch_min
stream.WriteColumn(std::to_string(alias->pitchMin)); const auto pitchMinHertz = alias->pitchMin;
const auto pitchMinCents = static_cast<int>(std::round(HertzToCents(pitchMinHertz)));
stream.WriteColumn(std::to_string(pitchMinCents));
// pitch_max // pitch_max
stream.WriteColumn(std::to_string(alias->pitchMax)); const auto pitchMaxHertz = alias->pitchMax;
const auto pitchMaxCents = static_cast<int>(std::round(HertzToCents(pitchMaxHertz)));
stream.WriteColumn(std::to_string(pitchMaxCents));
// team_pitch_mod // team_pitch_mod
stream.WriteColumn(""); stream.WriteColumn("");