mirror of https://github.com/OpenTTD/OpenTTD
Codechange: let NewGRF sounds make use of RandomAccessFile instead of the FIO slot functions
parent
b144e56b2c
commit
c097bc9d7d
|
@ -39,6 +39,14 @@ static bool _do_scan_working_directory = true;
|
||||||
extern std::string _config_file;
|
extern std::string _config_file;
|
||||||
extern std::string _highscore_file;
|
extern std::string _highscore_file;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transition helper to get the RandomAccessFile associated with a given slot.
|
||||||
|
*/
|
||||||
|
RandomAccessFile *FioGetRandomAccessFile(int slot)
|
||||||
|
{
|
||||||
|
return _fio_files[slot];
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get position in the current file.
|
* Get position in the current file.
|
||||||
* @return Position in the file.
|
* @return Position in the file.
|
||||||
|
|
|
@ -26,6 +26,7 @@ void FioCloseAll();
|
||||||
void FioOpenFile(int slot, const std::string &filename, Subdirectory subdir);
|
void FioOpenFile(int slot, const std::string &filename, Subdirectory subdir);
|
||||||
void FioReadBlock(void *ptr, size_t size);
|
void FioReadBlock(void *ptr, size_t size);
|
||||||
void FioSkipBytes(int n);
|
void FioSkipBytes(int n);
|
||||||
|
class RandomAccessFile *FioGetRandomAccessFile(int slot);
|
||||||
|
|
||||||
void FioFCloseFile(FILE *f);
|
void FioFCloseFile(FILE *f);
|
||||||
FILE *FioFOpenFile(const std::string &filename, const char *mode, Subdirectory subdir, size_t *filesize = nullptr);
|
FILE *FioFOpenFile(const std::string &filename, const char *mode, Subdirectory subdir, size_t *filesize = nullptr);
|
||||||
|
|
|
@ -91,8 +91,6 @@ enum FileSlots {
|
||||||
* and thus cannot be used for files, which are continuously accessed during a game.
|
* and thus cannot be used for files, which are continuously accessed during a game.
|
||||||
*/
|
*/
|
||||||
CONFIG_SLOT = 0,
|
CONFIG_SLOT = 0,
|
||||||
/** Slot for the sound. */
|
|
||||||
SOUND_SLOT = 1,
|
|
||||||
/** First slot usable for (New)GRFs used during the game. */
|
/** First slot usable for (New)GRFs used during the game. */
|
||||||
FIRST_GRF_SLOT = 2,
|
FIRST_GRF_SLOT = 2,
|
||||||
/** Maximum number of slots. */
|
/** Maximum number of slots. */
|
||||||
|
|
|
@ -7553,7 +7553,7 @@ static void LoadGRFSound(size_t offs, SoundEntry *sound)
|
||||||
|
|
||||||
if (offs != SIZE_MAX) {
|
if (offs != SIZE_MAX) {
|
||||||
/* Sound is present in the NewGRF. */
|
/* Sound is present in the NewGRF. */
|
||||||
sound->file_slot = _cur.file_index;
|
sound->file = FioGetRandomAccessFile(_cur.file_index);
|
||||||
sound->file_offset = offs;
|
sound->file_offset = offs;
|
||||||
sound->grf_container_ver = _cur.grf_container_ver;
|
sound->grf_container_ver = _cur.grf_container_ver;
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,7 +14,7 @@
|
||||||
#include "newgrf_sound.h"
|
#include "newgrf_sound.h"
|
||||||
#include "vehicle_base.h"
|
#include "vehicle_base.h"
|
||||||
#include "sound_func.h"
|
#include "sound_func.h"
|
||||||
#include "fileio_func.h"
|
#include "random_access_file_type.h"
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
#include "settings_type.h"
|
#include "settings_type.h"
|
||||||
|
|
||||||
|
@ -65,56 +65,57 @@ uint GetNumSounds()
|
||||||
*/
|
*/
|
||||||
bool LoadNewGRFSound(SoundEntry *sound)
|
bool LoadNewGRFSound(SoundEntry *sound)
|
||||||
{
|
{
|
||||||
if (sound->file_offset == SIZE_MAX || sound->file_slot == 0) return false;
|
if (sound->file_offset == SIZE_MAX || sound->file == nullptr) return false;
|
||||||
|
|
||||||
FioSeekToFile(sound->file_slot, sound->file_offset);
|
RandomAccessFile &file = *sound->file;
|
||||||
|
file.SeekTo(sound->file_offset, SEEK_SET);
|
||||||
|
|
||||||
/* Skip ID for container version >= 2 as we only look at the first
|
/* Skip ID for container version >= 2 as we only look at the first
|
||||||
* entry and ignore any further entries with the same ID. */
|
* entry and ignore any further entries with the same ID. */
|
||||||
if (sound->grf_container_ver >= 2) FioReadDword();
|
if (sound->grf_container_ver >= 2) file.ReadDword();
|
||||||
|
|
||||||
/* Format: <num> <FF> <FF> <name_len> <name> '\0' <data> */
|
/* Format: <num> <FF> <FF> <name_len> <name> '\0' <data> */
|
||||||
|
|
||||||
uint32 num = sound->grf_container_ver >= 2 ? FioReadDword() : FioReadWord();
|
uint32 num = sound->grf_container_ver >= 2 ? file.ReadDword() : file.ReadWord();
|
||||||
if (FioReadByte() != 0xFF) return false;
|
if (file.ReadByte() != 0xFF) return false;
|
||||||
if (FioReadByte() != 0xFF) return false;
|
if (file.ReadByte() != 0xFF) return false;
|
||||||
|
|
||||||
uint8 name_len = FioReadByte();
|
uint8 name_len = file.ReadByte();
|
||||||
char *name = AllocaM(char, name_len + 1);
|
char *name = AllocaM(char, name_len + 1);
|
||||||
FioReadBlock(name, name_len + 1);
|
file.ReadBlock(name, name_len + 1);
|
||||||
|
|
||||||
/* Test string termination */
|
/* Test string termination */
|
||||||
if (name[name_len] != 0) {
|
if (name[name_len] != 0) {
|
||||||
DEBUG(grf, 2, "LoadNewGRFSound [%s]: Name not properly terminated", FioGetFilename(sound->file_slot));
|
DEBUG(grf, 2, "LoadNewGRFSound [%s]: Name not properly terminated", file.GetSimplifiedFilename().c_str());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
DEBUG(grf, 2, "LoadNewGRFSound [%s]: Sound name '%s'...", FioGetFilename(sound->file_slot), name);
|
DEBUG(grf, 2, "LoadNewGRFSound [%s]: Sound name '%s'...", file.GetSimplifiedFilename().c_str(), name);
|
||||||
|
|
||||||
if (FioReadDword() != BSWAP32('RIFF')) {
|
if (file.ReadDword() != BSWAP32('RIFF')) {
|
||||||
DEBUG(grf, 1, "LoadNewGRFSound [%s]: Missing RIFF header", FioGetFilename(sound->file_slot));
|
DEBUG(grf, 1, "LoadNewGRFSound [%s]: Missing RIFF header", file.GetSimplifiedFilename().c_str());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32 total_size = FioReadDword();
|
uint32 total_size = file.ReadDword();
|
||||||
uint header_size = 11;
|
uint header_size = 11;
|
||||||
if (sound->grf_container_ver >= 2) header_size++; // The first FF in the sprite is only counted for container version >= 2.
|
if (sound->grf_container_ver >= 2) header_size++; // The first FF in the sprite is only counted for container version >= 2.
|
||||||
if (total_size + name_len + header_size > num) {
|
if (total_size + name_len + header_size > num) {
|
||||||
DEBUG(grf, 1, "LoadNewGRFSound [%s]: RIFF was truncated", FioGetFilename(sound->file_slot));
|
DEBUG(grf, 1, "LoadNewGRFSound [%s]: RIFF was truncated", file.GetSimplifiedFilename().c_str());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (FioReadDword() != BSWAP32('WAVE')) {
|
if (file.ReadDword() != BSWAP32('WAVE')) {
|
||||||
DEBUG(grf, 1, "LoadNewGRFSound [%s]: Invalid RIFF type", FioGetFilename(sound->file_slot));
|
DEBUG(grf, 1, "LoadNewGRFSound [%s]: Invalid RIFF type", file.GetSimplifiedFilename().c_str());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
while (total_size >= 8) {
|
while (total_size >= 8) {
|
||||||
uint32 tag = FioReadDword();
|
uint32 tag = file.ReadDword();
|
||||||
uint32 size = FioReadDword();
|
uint32 size = file.ReadDword();
|
||||||
total_size -= 8;
|
total_size -= 8;
|
||||||
if (total_size < size) {
|
if (total_size < size) {
|
||||||
DEBUG(grf, 1, "LoadNewGRFSound [%s]: Invalid RIFF", FioGetFilename(sound->file_slot));
|
DEBUG(grf, 1, "LoadNewGRFSound [%s]: Invalid RIFF", file.GetSimplifiedFilename().c_str());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
total_size -= size;
|
total_size -= size;
|
||||||
|
@ -122,15 +123,15 @@ bool LoadNewGRFSound(SoundEntry *sound)
|
||||||
switch (tag) {
|
switch (tag) {
|
||||||
case ' tmf': // 'fmt '
|
case ' tmf': // 'fmt '
|
||||||
/* Audio format, must be 1 (PCM) */
|
/* Audio format, must be 1 (PCM) */
|
||||||
if (size < 16 || FioReadWord() != 1) {
|
if (size < 16 || file.ReadWord() != 1) {
|
||||||
DEBUG(grf, 1, "LoadGRFSound [%s]: Invalid audio format", FioGetFilename(sound->file_slot));
|
DEBUG(grf, 1, "LoadGRFSound [%s]: Invalid audio format", file.GetSimplifiedFilename().c_str());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
sound->channels = FioReadWord();
|
sound->channels = file.ReadWord();
|
||||||
sound->rate = FioReadDword();
|
sound->rate = file.ReadDword();
|
||||||
FioReadDword();
|
file.ReadDword();
|
||||||
FioReadWord();
|
file.ReadWord();
|
||||||
sound->bits_per_sample = FioReadWord();
|
sound->bits_per_sample = file.ReadWord();
|
||||||
|
|
||||||
/* The rest will be skipped */
|
/* The rest will be skipped */
|
||||||
size -= 16;
|
size -= 16;
|
||||||
|
@ -138,9 +139,9 @@ bool LoadNewGRFSound(SoundEntry *sound)
|
||||||
|
|
||||||
case 'atad': // 'data'
|
case 'atad': // 'data'
|
||||||
sound->file_size = size;
|
sound->file_size = size;
|
||||||
sound->file_offset = FioGetPos();
|
sound->file_offset = file.GetPos();
|
||||||
|
|
||||||
DEBUG(grf, 2, "LoadNewGRFSound [%s]: channels %u, sample rate %u, bits per sample %u, length %u", FioGetFilename(sound->file_slot), sound->channels, sound->rate, sound->bits_per_sample, size);
|
DEBUG(grf, 2, "LoadNewGRFSound [%s]: channels %u, sample rate %u, bits per sample %u, length %u", file.GetSimplifiedFilename().c_str(), sound->channels, sound->rate, sound->bits_per_sample, size);
|
||||||
return true; // the fmt chunk has to appear before data, so we are finished
|
return true; // the fmt chunk has to appear before data, so we are finished
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
@ -149,10 +150,10 @@ bool LoadNewGRFSound(SoundEntry *sound)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Skip rest of chunk */
|
/* Skip rest of chunk */
|
||||||
if (size > 0) FioSkipBytes(size);
|
if (size > 0) file.SkipBytes(size);
|
||||||
}
|
}
|
||||||
|
|
||||||
DEBUG(grf, 1, "LoadNewGRFSound [%s]: RIFF does not contain any sound data", FioGetFilename(sound->file_slot));
|
DEBUG(grf, 1, "LoadNewGRFSound [%s]: RIFF does not contain any sound data", file.GetSimplifiedFilename().c_str());
|
||||||
|
|
||||||
/* Clear everything that was read */
|
/* Clear everything that was read */
|
||||||
MemSetT(sound, 0);
|
MemSetT(sound, 0);
|
||||||
|
|
|
@ -11,7 +11,7 @@
|
||||||
#include "landscape.h"
|
#include "landscape.h"
|
||||||
#include "mixer.h"
|
#include "mixer.h"
|
||||||
#include "newgrf_sound.h"
|
#include "newgrf_sound.h"
|
||||||
#include "fios.h"
|
#include "random_access_file_type.h"
|
||||||
#include "window_gui.h"
|
#include "window_gui.h"
|
||||||
#include "vehicle_base.h"
|
#include "vehicle_base.h"
|
||||||
|
|
||||||
|
@ -25,14 +25,20 @@ static SoundEntry _original_sounds[ORIGINAL_SAMPLE_COUNT];
|
||||||
|
|
||||||
static void OpenBankFile(const char *filename)
|
static void OpenBankFile(const char *filename)
|
||||||
{
|
{
|
||||||
|
/**
|
||||||
|
* The sound file for the original sounds, i.e. those not defined/overridden by a NewGRF.
|
||||||
|
* Needs to be kept alive during the game as _original_sounds[n].file refers to this.
|
||||||
|
*/
|
||||||
|
static std::unique_ptr<RandomAccessFile> original_sound_file;
|
||||||
|
|
||||||
memset(_original_sounds, 0, sizeof(_original_sounds));
|
memset(_original_sounds, 0, sizeof(_original_sounds));
|
||||||
|
|
||||||
/* If there is no sound file (nosound set), don't load anything */
|
/* If there is no sound file (nosound set), don't load anything */
|
||||||
if (filename == nullptr) return;
|
if (filename == nullptr) return;
|
||||||
|
|
||||||
FioOpenFile(SOUND_SLOT, filename, BASESET_DIR);
|
original_sound_file.reset(new RandomAccessFile(filename, BASESET_DIR));
|
||||||
size_t pos = FioGetPos();
|
size_t pos = original_sound_file->GetPos();
|
||||||
uint count = FioReadDword();
|
uint count = original_sound_file->ReadDword();
|
||||||
|
|
||||||
/* The new format has the highest bit always set */
|
/* The new format has the highest bit always set */
|
||||||
bool new_format = HasBit(count, 31);
|
bool new_format = HasBit(count, 31);
|
||||||
|
@ -48,43 +54,43 @@ static void OpenBankFile(const char *filename)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
FioSeekTo(pos, SEEK_SET);
|
original_sound_file->SeekTo(pos, SEEK_SET);
|
||||||
|
|
||||||
for (uint i = 0; i != ORIGINAL_SAMPLE_COUNT; i++) {
|
for (uint i = 0; i != ORIGINAL_SAMPLE_COUNT; i++) {
|
||||||
_original_sounds[i].file_slot = SOUND_SLOT;
|
_original_sounds[i].file = original_sound_file.get();
|
||||||
_original_sounds[i].file_offset = GB(FioReadDword(), 0, 31) + pos;
|
_original_sounds[i].file_offset = GB(original_sound_file->ReadDword(), 0, 31) + pos;
|
||||||
_original_sounds[i].file_size = FioReadDword();
|
_original_sounds[i].file_size = original_sound_file->ReadDword();
|
||||||
}
|
}
|
||||||
|
|
||||||
for (uint i = 0; i != ORIGINAL_SAMPLE_COUNT; i++) {
|
for (uint i = 0; i != ORIGINAL_SAMPLE_COUNT; i++) {
|
||||||
SoundEntry *sound = &_original_sounds[i];
|
SoundEntry *sound = &_original_sounds[i];
|
||||||
char name[255];
|
char name[255];
|
||||||
|
|
||||||
FioSeekTo(sound->file_offset, SEEK_SET);
|
original_sound_file->SeekTo(sound->file_offset, SEEK_SET);
|
||||||
|
|
||||||
/* Check for special case, see else case */
|
/* Check for special case, see else case */
|
||||||
FioReadBlock(name, FioReadByte()); // Read the name of the sound
|
original_sound_file->ReadBlock(name, original_sound_file->ReadByte()); // Read the name of the sound
|
||||||
if (new_format || strcmp(name, "Corrupt sound") != 0) {
|
if (new_format || strcmp(name, "Corrupt sound") != 0) {
|
||||||
FioSeekTo(12, SEEK_CUR); // Skip past RIFF header
|
original_sound_file->SeekTo(12, SEEK_CUR); // Skip past RIFF header
|
||||||
|
|
||||||
/* Read riff tags */
|
/* Read riff tags */
|
||||||
for (;;) {
|
for (;;) {
|
||||||
uint32 tag = FioReadDword();
|
uint32 tag = original_sound_file->ReadDword();
|
||||||
uint32 size = FioReadDword();
|
uint32 size = original_sound_file->ReadDword();
|
||||||
|
|
||||||
if (tag == ' tmf') {
|
if (tag == ' tmf') {
|
||||||
FioReadWord(); // wFormatTag
|
original_sound_file->ReadWord(); // wFormatTag
|
||||||
sound->channels = FioReadWord(); // wChannels
|
sound->channels = original_sound_file->ReadWord(); // wChannels
|
||||||
sound->rate = FioReadDword(); // samples per second
|
sound->rate = original_sound_file->ReadDword(); // samples per second
|
||||||
if (!new_format) sound->rate = 11025; // seems like all old samples should be played at this rate.
|
if (!new_format) sound->rate = 11025; // seems like all old samples should be played at this rate.
|
||||||
FioReadDword(); // avg bytes per second
|
original_sound_file->ReadDword(); // avg bytes per second
|
||||||
FioReadWord(); // alignment
|
original_sound_file->ReadWord(); // alignment
|
||||||
sound->bits_per_sample = FioReadByte(); // bits per sample
|
sound->bits_per_sample = original_sound_file->ReadByte(); // bits per sample
|
||||||
FioSeekTo(size - (2 + 2 + 4 + 4 + 2 + 1), SEEK_CUR);
|
original_sound_file->SeekTo(size - (2 + 2 + 4 + 4 + 2 + 1), SEEK_CUR);
|
||||||
} else if (tag == 'atad') {
|
} else if (tag == 'atad') {
|
||||||
sound->file_size = size;
|
sound->file_size = size;
|
||||||
sound->file_slot = SOUND_SLOT;
|
sound->file = original_sound_file.get();
|
||||||
sound->file_offset = FioGetPos();
|
sound->file_offset = original_sound_file->GetPos();
|
||||||
break;
|
break;
|
||||||
} else {
|
} else {
|
||||||
sound->file_size = 0;
|
sound->file_size = 0;
|
||||||
|
@ -100,8 +106,8 @@ static void OpenBankFile(const char *filename)
|
||||||
sound->channels = 1;
|
sound->channels = 1;
|
||||||
sound->rate = 11025;
|
sound->rate = 11025;
|
||||||
sound->bits_per_sample = 8;
|
sound->bits_per_sample = 8;
|
||||||
sound->file_slot = SOUND_SLOT;
|
sound->file = original_sound_file.get();
|
||||||
sound->file_offset = FioGetPos();
|
sound->file_offset = original_sound_file->GetPos();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -119,8 +125,9 @@ static bool SetBankSource(MixerChannel *mc, const SoundEntry *sound)
|
||||||
mem[sound->file_size ] = 0;
|
mem[sound->file_size ] = 0;
|
||||||
mem[sound->file_size + 1] = 0;
|
mem[sound->file_size + 1] = 0;
|
||||||
|
|
||||||
FioSeekToFile(sound->file_slot, sound->file_offset);
|
RandomAccessFile *file = sound->file;
|
||||||
FioReadBlock(mem, sound->file_size);
|
file->SeekTo(sound->file_offset, SEEK_SET);
|
||||||
|
file->ReadBlock(mem, sound->file_size);
|
||||||
|
|
||||||
/* 16-bit PCM WAV files should be signed by default */
|
/* 16-bit PCM WAV files should be signed by default */
|
||||||
if (sound->bits_per_sample == 8) {
|
if (sound->bits_per_sample == 8) {
|
||||||
|
@ -163,10 +170,10 @@ static void StartSound(SoundID sound_id, float pan, uint volume)
|
||||||
if (sound == nullptr) return;
|
if (sound == nullptr) return;
|
||||||
|
|
||||||
/* NewGRF sound that wasn't loaded yet? */
|
/* NewGRF sound that wasn't loaded yet? */
|
||||||
if (sound->rate == 0 && sound->file_slot != 0) {
|
if (sound->rate == 0 && sound->file != nullptr) {
|
||||||
if (!LoadNewGRFSound(sound)) {
|
if (!LoadNewGRFSound(sound)) {
|
||||||
/* Mark as invalid. */
|
/* Mark as invalid. */
|
||||||
sound->file_slot = 0;
|
sound->file = nullptr;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,7 +11,7 @@
|
||||||
#define SOUND_TYPE_H
|
#define SOUND_TYPE_H
|
||||||
|
|
||||||
struct SoundEntry {
|
struct SoundEntry {
|
||||||
uint8 file_slot;
|
class RandomAccessFile *file;
|
||||||
size_t file_offset;
|
size_t file_offset;
|
||||||
size_t file_size;
|
size_t file_size;
|
||||||
uint16 rate;
|
uint16 rate;
|
||||||
|
|
Loading…
Reference in New Issue