5151 lines
140 KiB
C
5151 lines
140 KiB
C
// VGMPlay.c: C Source File of the Main Executable
|
|
//
|
|
|
|
// Line Size: 96 Chars
|
|
// Tab Size: 4 Spaces
|
|
|
|
/*3456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456
|
|
0000000001111111111222222222233333333334444444444555555555566666666667777777777888888888899999*/
|
|
// TODO: Callback "ChangeSampleRate" to fix YM2203's AY8910
|
|
|
|
// Mixer Muting ON:
|
|
// Mixer's FM Volume is set to 0 or Mute -> absolutely muted
|
|
// (sometimes it can take some time to get the Mixer Control under Windows)
|
|
// Mixer Muting OFF:
|
|
// FM Volume is set to 0 through commands -> very very low volume level ~0.4%
|
|
// (faster way)
|
|
//#define MIXER_MUTING
|
|
|
|
// These defines enable additional features.
|
|
// ADDITIONAL_FORMATS enables CMF and DRO support.
|
|
// CONSOLE_MODE switches between VGMPlay and in_vgm mode.
|
|
// in_vgm mode can also be used for custom players.
|
|
//
|
|
//#define ADDITIONAL_FORMATS
|
|
//#define CONSOLE_MODE
|
|
//#define VGM_BIG_ENDIAN
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <wchar.h>
|
|
#include "stdbool.h"
|
|
#include <math.h> // for pow()
|
|
|
|
#ifndef NO_ZLIB
|
|
#include <zlib.h>
|
|
#endif
|
|
|
|
#include "resampler.h"
|
|
|
|
#include "chips/mamedef.h"
|
|
|
|
// integer types for fast integer calculation
|
|
// the bit number is unused (it's an orientation)
|
|
#define FUINT8 unsigned int
|
|
#define FUINT16 unsigned int
|
|
|
|
#include "VGMPlay.h"
|
|
//#include "VGMPlay_Intf.h" // Already included by VGMPlay.h now
|
|
|
|
#include "chips/ChipIncl.h"
|
|
|
|
unsigned char OpenPortTalk(void *);
|
|
void ClosePortTalk(void *);
|
|
|
|
#include "ChipMapper.h"
|
|
|
|
|
|
|
|
// Function Prototypes (prototypes in comments are defined in VGMPlay_Intf.h)
|
|
//void VGMPlay_Init(void);
|
|
//void VGMPlay_Init2(void);
|
|
//void VGMPlay_Deinit(void);
|
|
|
|
INLINE UINT16 ReadLE16(const UINT8* Data);
|
|
INLINE UINT16 ReadBE16(const UINT8* Data);
|
|
INLINE UINT32 ReadLE24(const UINT8* Data);
|
|
INLINE UINT32 ReadLE32(const UINT8* Data);
|
|
INLINE int FILE_getLE16(VGM_FILE* hFile, UINT16* RetValue);
|
|
INLINE int FILE_getLE32(VGM_FILE* hFile, UINT32* RetValue);
|
|
static UINT32 gcd(UINT32 x, UINT32 y);
|
|
//void PlayVGM(void);
|
|
//void StopVGM(void);
|
|
//void RestartVGM(void);
|
|
//void PauseVGM(bool Pause);
|
|
//void SeekVGM(bool Relative, INT32 PlayBkSamples);
|
|
//void RefreshMuting(void);
|
|
//void RefreshPanning(void);
|
|
//void RefreshPlaybackOptions(void);
|
|
|
|
//UINT32 GetGZFileLength(const char* FileName);
|
|
//UINT32 GetGZFileLengthW(const wchar_t* FileName);
|
|
static UINT32 GetGZFileLength_Internal(FILE* hFile);
|
|
//bool OpenVGMFile(const char* FileName);
|
|
static bool OpenVGMFile_Internal(VGM_PLAYER*, VGM_FILE* hFile, UINT32 FileSize);
|
|
static void ReadVGMHeader(VGM_FILE* hFile, VGM_HEADER* RetVGMHead);
|
|
static UINT8 ReadGD3Tag(VGM_FILE* hFile, UINT32 GD3Offset, GD3_TAG* RetGD3Tag);
|
|
static void ReadChipExtraData32(VGM_PLAYER*, UINT32 StartOffset, VGMX_CHP_EXTRA32* ChpExtra);
|
|
static void ReadChipExtraData16(VGM_PLAYER*, UINT32 StartOffset, VGMX_CHP_EXTRA16* ChpExtra);
|
|
//void CloseVGMFile(void);
|
|
//void FreeGD3Tag(GD3_TAG* TagData);
|
|
static wchar_t* MakeEmptyWStr(void);
|
|
static wchar_t* ReadWStrFromFile(VGM_FILE* hFile, UINT32* FilePos, UINT32 EOFPos);
|
|
//UINT32 GetVGMFileInfo(const char* FileName, VGM_HEADER* RetVGMHead, GD3_TAG* RetGD3Tag);
|
|
static UINT32 GetVGMFileInfo_Internal(VGM_FILE* hFile, UINT32 FileSize,
|
|
VGM_HEADER* RetVGMHead, GD3_TAG* RetGD3Tag);
|
|
INLINE UINT32 MulDivRound(UINT64 Number, UINT64 Numerator, UINT64 Denominator);
|
|
//UINT32 CalcSampleMSec(VGM_PLAYER* p, UINT64 Value, UINT8 Mode);
|
|
//UINT32 CalcSampleMSecExt(VGM_PLAYER* p, UINT64 Value, UINT8 Mode, VGM_HEADER* FileHead);
|
|
//const char* GetChipName(UINT8 ChipID);
|
|
//const char* GetAccurateChipName(UINT8 ChipID, UINT8 SubType);
|
|
//UINT32 GetChipClock(void*, UINT8 ChipID, UINT8* RetSubType);
|
|
static UINT16 GetChipVolume(VGM_PLAYER*, UINT8 ChipID, UINT8 ChipNum, UINT8 ChipCnt);
|
|
|
|
static void RestartPlaying(VGM_PLAYER*);
|
|
static void Chips_GeneralActions(VGM_PLAYER*, UINT8 Mode);
|
|
|
|
INLINE INT32 SampleVGM2Pbk_I(VGM_PLAYER*, INT32 SampleVal); // inline functions
|
|
INLINE INT32 SamplePbk2VGM_I(VGM_PLAYER*, INT32 SampleVal);
|
|
//INT32 SampleVGM2Playback(void*, INT32 SampleVal); // non-inline functions
|
|
//INT32 SamplePlayback2VGM(void*, INT32 SampleVal);
|
|
static bool SetMuteControl(VGM_PLAYER*, bool mute);
|
|
|
|
static void InterpretFile(VGM_PLAYER*, UINT32 SampleCount);
|
|
static void AddPCMData(VGM_PLAYER*, UINT8 Type, UINT32 DataSize, const UINT8* Data);
|
|
//INLINE FUINT16 ReadBits(UINT8* Data, UINT32* Pos, FUINT8* BitPos, FUINT8 BitsToRead);
|
|
static bool DecompressDataBlk(VGM_PLAYER* p, VGM_PCM_DATA* Bank, UINT32 DataSize, const UINT8* Data);
|
|
static UINT8 GetDACFromPCMBank(VGM_PLAYER*);
|
|
static UINT8* GetPointerFromPCMBank(VGM_PLAYER*, UINT8 Type, UINT32 DataPos);
|
|
static void ReadPCMTable(VGM_PLAYER*, UINT32 DataSize, const UINT8* Data);
|
|
static void InterpretVGM(VGM_PLAYER*, UINT32 SampleCount);
|
|
#ifdef ADDITIONAL_FORMATS
|
|
extern void InterpretOther(VGM_PLAYER*, UINT32 SampleCount);
|
|
#endif
|
|
|
|
static void GeneralChipLists(VGM_PLAYER*);
|
|
static void SetupResampler(VGM_PLAYER*, CAUD_ATTR* CAA);
|
|
static void ChangeChipSampleRate(void* DataPtr, UINT32 NewSmplRate);
|
|
|
|
INLINE INT16 Limit2Short(INT32 Value);
|
|
static void null_update(void *param, stream_sample_t **outputs, int samples);
|
|
struct dual_opl2_info
|
|
{
|
|
void * chip;
|
|
int ChipID;
|
|
};
|
|
static void dual_opl2_stereo(void *param, stream_sample_t **outputs, int samples);
|
|
static void ResampleChipStream(VGM_PLAYER*, CA_LIST* CLst, WAVE_32BS* RetSample, UINT32 Length);
|
|
static INT32 RecalcFadeVolume(VGM_PLAYER*);
|
|
//UINT32 FillBuffer(void *, WAVE_16BS* Buffer, UINT32 BufferSize)
|
|
|
|
// Options and such moved to VGM_PLAYER structure
|
|
|
|
void * VGMPlay_Init(void)
|
|
{
|
|
UINT8 CurChip;
|
|
UINT8 CurCSet;
|
|
UINT8 CurChn;
|
|
CHIP_OPTS* TempCOpt;
|
|
CAUD_ATTR* TempCAud;
|
|
|
|
VGM_PLAYER* p = (VGM_PLAYER*) calloc(1, sizeof(*p));
|
|
if (!p)
|
|
return NULL;
|
|
|
|
p->SampleRate = 44100;
|
|
p->FadeTime = 5000;
|
|
|
|
p->FadeRAWLog = false;
|
|
p->VolumeLevel = 1.0f;
|
|
//p->FullBufFill = false;
|
|
p->SurroundSound = false;
|
|
p->VGMMaxLoop = 0x02;
|
|
p->VGMPbRate = 0;
|
|
#ifdef ADDITIONAL_FORMATS
|
|
p->CMFMaxLoop = 0x01;
|
|
#endif
|
|
p->ResampleMode = 0x00;
|
|
p->CHIP_SAMPLING_MODE = 0x00;
|
|
p->CHIP_SAMPLE_RATE = 0x00000000;
|
|
p->DoubleSSGVol = false;
|
|
|
|
for (CurCSet = 0x00; CurCSet < 0x02; CurCSet ++)
|
|
{
|
|
TempCAud = (CAUD_ATTR*)&p->ChipAudio[CurCSet];
|
|
for (CurChip = 0x00; CurChip < CHIP_COUNT; CurChip ++, TempCAud ++)
|
|
{
|
|
TempCOpt = (CHIP_OPTS*)&p->ChipOpts[CurCSet] + CurChip;
|
|
|
|
TempCOpt->Disabled = false;
|
|
TempCOpt->EmuCore = 0x00;
|
|
TempCOpt->SpecialFlags = 0x00;
|
|
TempCOpt->ChnCnt = 0x00;
|
|
TempCOpt->ChnMute1 = 0x00;
|
|
TempCOpt->ChnMute2 = 0x00;
|
|
TempCOpt->ChnMute3 = 0x00;
|
|
TempCOpt->Panning = NULL;
|
|
|
|
// Set up some important fields to prevent in_vgm from crashing
|
|
// when clicking on Muting checkboxes after init.
|
|
TempCAud->ChipType = 0xFF;
|
|
TempCAud->ChipID = CurCSet;
|
|
TempCAud->Paired = NULL;
|
|
}
|
|
p->ChipOpts[CurCSet].GameBoy.SpecialFlags = 0x0003;
|
|
// default options, 0x8000 skips the option write and keeps NSFPlay's default values
|
|
p->ChipOpts[CurCSet].NES.SpecialFlags = 0x8000 |
|
|
(0x00 << 12) | (0x3B << 4) | (0x01 << 2) | (0x03 << 0);
|
|
|
|
TempCAud = p->CA_Paired[CurCSet];
|
|
for (CurChip = 0x00; CurChip < 0x03; CurChip ++, TempCAud ++)
|
|
{
|
|
TempCAud->ChipType = 0xFF;
|
|
TempCAud->ChipID = CurCSet;
|
|
TempCAud->Paired = NULL;
|
|
}
|
|
|
|
// currently the only chips with Panning support are
|
|
// SN76496 and YM2413, it should be not a problem that it's hardcoded.
|
|
TempCOpt = (CHIP_OPTS*)&p->ChipOpts[CurCSet].SN76496;
|
|
TempCOpt->ChnCnt = 0x04;
|
|
TempCOpt->Panning = (INT16*)malloc(sizeof(INT16) * TempCOpt->ChnCnt);
|
|
for (CurChn = 0x00; CurChn < TempCOpt->ChnCnt; CurChn ++)
|
|
TempCOpt->Panning[CurChn] = 0x00;
|
|
|
|
TempCOpt = (CHIP_OPTS*)&p->ChipOpts[CurCSet].YM2413;
|
|
TempCOpt->ChnCnt = 0x0E; // 0x09 + 0x05
|
|
TempCOpt->Panning = (INT16*)malloc(sizeof(INT16) * TempCOpt->ChnCnt);
|
|
for (CurChn = 0x00; CurChn < TempCOpt->ChnCnt; CurChn ++)
|
|
TempCOpt->Panning[CurChn] = 0x00;
|
|
}
|
|
|
|
p->FileMode = 0xFF;
|
|
|
|
return p;
|
|
}
|
|
|
|
void VGMPlay_Init2(void *_p)
|
|
{
|
|
VGM_PLAYER* p = (VGM_PLAYER*)_p;
|
|
// has to be called after the configuration is loaded
|
|
|
|
p->StreamBufs[0x00] = (INT32*)malloc(SMPL_BUFSIZE * sizeof(INT32));
|
|
p->StreamBufs[0x01] = (INT32*)malloc(SMPL_BUFSIZE * sizeof(INT32));
|
|
|
|
if (p->CHIP_SAMPLE_RATE <= 0)
|
|
p->CHIP_SAMPLE_RATE = p->SampleRate;
|
|
p->PlayingMode = 0xFF;
|
|
|
|
return;
|
|
}
|
|
|
|
void VGMPlay_Deinit(void *_p)
|
|
{
|
|
UINT8 CurChip;
|
|
UINT8 CurCSet;
|
|
CHIP_OPTS* TempCOpt;
|
|
|
|
VGM_PLAYER* p = (VGM_PLAYER*)_p;
|
|
|
|
free(p->StreamBufs[0x00]); p->StreamBufs[0x00] = NULL;
|
|
free(p->StreamBufs[0x01]); p->StreamBufs[0x01] = NULL;
|
|
|
|
for (CurCSet = 0x00; CurCSet < 0x02; CurCSet ++)
|
|
{
|
|
for (CurChip = 0x00; CurChip < CHIP_COUNT; CurChip ++)
|
|
{
|
|
TempCOpt = (CHIP_OPTS*)&p->ChipOpts[CurCSet] + CurChip;
|
|
|
|
if (TempCOpt->Panning != NULL)
|
|
{
|
|
free(TempCOpt->Panning); TempCOpt->Panning = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
free(p);
|
|
|
|
return;
|
|
}
|
|
|
|
INLINE UINT16 ReadLE16(const UINT8* Data)
|
|
{
|
|
// read 16-Bit Word (Little Endian/Intel Byte Order)
|
|
#ifndef VGM_BIG_ENDIAN
|
|
return *(UINT16*)Data;
|
|
#else
|
|
return (Data[0x01] << 8) | (Data[0x00] << 0);
|
|
#endif
|
|
}
|
|
|
|
INLINE UINT16 ReadBE16(const UINT8* Data)
|
|
{
|
|
// read 16-Bit Word (Big Endian/Motorola Byte Order)
|
|
#ifndef VGM_BIG_ENDIAN
|
|
return (Data[0x00] << 8) | (Data[0x01] << 0);
|
|
#else
|
|
return *(UINT16*)Data;
|
|
#endif
|
|
}
|
|
|
|
INLINE UINT32 ReadLE24(const UINT8* Data)
|
|
{
|
|
// read 24-Bit Word (Little Endian/Intel Byte Order)
|
|
#ifndef VGM_BIG_ENDIAN
|
|
return (*(UINT32*)Data) & 0x00FFFFFF;
|
|
#else
|
|
return (Data[0x02] << 16) | (Data[0x01] << 8) | (Data[0x00] << 0);
|
|
#endif
|
|
}
|
|
|
|
INLINE UINT32 ReadLE32(const UINT8* Data)
|
|
{
|
|
// read 32-Bit Word (Little Endian/Intel Byte Order)
|
|
#ifndef VGM_BIG_ENDIAN
|
|
return *(UINT32*)Data;
|
|
#else
|
|
return (Data[0x03] << 24) | (Data[0x02] << 16) |
|
|
(Data[0x01] << 8) | (Data[0x00] << 0);
|
|
#endif
|
|
}
|
|
|
|
INLINE int FILE_getLE16(VGM_FILE* hFile, UINT16* RetValue)
|
|
{
|
|
#ifndef VGM_BIG_ENDIAN
|
|
return hFile->Read(hFile, RetValue, 0x02);
|
|
#else
|
|
int RetVal;
|
|
UINT8 Data[0x02];
|
|
|
|
RetVal = hFile->Read(hFile, Data, 0x02);
|
|
*RetValue = (Data[0x01] << 8) | (Data[0x00] << 0);
|
|
return RetVal;
|
|
#endif
|
|
}
|
|
|
|
INLINE int FILE_getLE32(VGM_FILE* hFile, UINT32* RetValue)
|
|
{
|
|
#ifndef VGM_BIG_ENDIAN
|
|
return hFile->Read(hFile, RetValue, 0x04);
|
|
#else
|
|
int RetVal;
|
|
UINT8 Data[0x04];
|
|
|
|
RetVal = hFime->Read(hFile, Data, 0x04);
|
|
*RetValue = (Data[0x03] << 24) | (Data[0x02] << 16) |
|
|
(Data[0x01] << 8) | (Data[0x00] << 0);
|
|
return RetVal;
|
|
#endif
|
|
}
|
|
|
|
static UINT32 gcd(UINT32 x, UINT32 y)
|
|
{
|
|
UINT32 shift;
|
|
UINT32 diff;
|
|
|
|
// Thanks to Wikipedia for this algorithm
|
|
// http://en.wikipedia.org/wiki/Binary_GCD_algorithm
|
|
if (! x || ! y)
|
|
return x | y;
|
|
|
|
for (shift = 0; ((x | y) & 1) == 0; shift ++)
|
|
{
|
|
x >>= 1;
|
|
y >>= 1;
|
|
}
|
|
|
|
while((x & 1) == 0)
|
|
x >>= 1;
|
|
|
|
do
|
|
{
|
|
while((y & 1) == 0)
|
|
y >>= 1;
|
|
|
|
if (x < y)
|
|
{
|
|
y -= x;
|
|
}
|
|
else
|
|
{
|
|
diff = x - y;
|
|
x = y;
|
|
y = diff;
|
|
}
|
|
y >>= 1;
|
|
} while(y);
|
|
|
|
return x << shift;
|
|
}
|
|
|
|
void PlayVGM(void *_p)
|
|
{
|
|
UINT8 CurChip;
|
|
UINT8 FMVal;
|
|
INT32 TempSLng;
|
|
|
|
VGM_PLAYER* p = (VGM_PLAYER*)_p;
|
|
|
|
if (p->PlayingMode != 0xFF)
|
|
return;
|
|
|
|
p->FadePlay = false;
|
|
p->MasterVol = 1.0f;
|
|
p->ForceVGMExec = false;
|
|
p->FadeStart = 0;
|
|
p->ForceVGMExec = true;
|
|
|
|
p->PlayingMode = 0x00; // Normal Mode
|
|
|
|
if (p->VGMHead.bytVolumeModifier <= VOLUME_MODIF_WRAP)
|
|
TempSLng = p->VGMHead.bytVolumeModifier;
|
|
else if (p->VGMHead.bytVolumeModifier == (VOLUME_MODIF_WRAP + 0x01))
|
|
TempSLng = VOLUME_MODIF_WRAP - 0x100;
|
|
else
|
|
TempSLng = p->VGMHead.bytVolumeModifier - 0x100;
|
|
p->VolumeLevelM = (float)(p->VolumeLevel * pow(2.0, TempSLng / (double)0x20));
|
|
p->FinalVol = p->VolumeLevelM;
|
|
|
|
if (! p->VGMMaxLoop)
|
|
{
|
|
p->VGMMaxLoopM = 0x00;
|
|
}
|
|
else
|
|
{
|
|
TempSLng = (p->VGMMaxLoop * p->VGMHead.bytLoopModifier + 0x08) / 0x10 - p->VGMHead.bytLoopBase;
|
|
p->VGMMaxLoopM = (TempSLng >= 0x01) ? TempSLng : 0x01;
|
|
}
|
|
|
|
if (! p->VGMPbRate || ! p->VGMHead.lngRate)
|
|
{
|
|
p->VGMPbRateMul = 1;
|
|
p->VGMPbRateDiv = 1;
|
|
}
|
|
else
|
|
{
|
|
// I prefer small Multiplers and Dividers, as they're used very often
|
|
TempSLng = gcd(p->VGMHead.lngRate, p->VGMPbRate);
|
|
p->VGMPbRateMul = p->VGMHead.lngRate / TempSLng;
|
|
p->VGMPbRateDiv = p->VGMPbRate / TempSLng;
|
|
}
|
|
p->VGMSmplRateMul = p->SampleRate * p->VGMPbRateMul;
|
|
p->VGMSmplRateDiv = p->VGMSampleRate * p->VGMPbRateDiv;
|
|
// same as above - to speed up the VGM <-> Playback calculation
|
|
TempSLng = gcd(p->VGMSmplRateMul, p->VGMSmplRateDiv);
|
|
p->VGMSmplRateMul /= TempSLng;
|
|
p->VGMSmplRateDiv /= TempSLng;
|
|
|
|
p->PlayingTime = 0;
|
|
p->EndPlay = false;
|
|
|
|
p->VGMPos = p->VGMHead.lngDataOffset;
|
|
p->VGMSmplPos = 0;
|
|
p->VGMSmplPlayed = 0;
|
|
p->VGMEnd = false;
|
|
p->VGMCurLoop = 0x00;
|
|
if (p->VGMPos >= p->VGMHead.lngEOFOffset)
|
|
p->VGMEnd = true;
|
|
|
|
Chips_GeneralActions(p, 0x00); // Start chips
|
|
// also does Reset (0x01), Muting Mask (0x10) and Panning (0x20)
|
|
|
|
p->Last95Drum = 0xFFFF;
|
|
p->Last95Freq = 0;
|
|
p->Last95Max = 0xFFFF;
|
|
p->IsVGMInit = true;
|
|
p->ErrorHappened = false;
|
|
InterpretFile(p, 0);
|
|
p->IsVGMInit = false;
|
|
|
|
p->ForceVGMExec = false;
|
|
|
|
return;
|
|
}
|
|
|
|
void StopVGM(void *_p)
|
|
{
|
|
VGM_PLAYER* p = (VGM_PLAYER*)_p;
|
|
if (p->PlayingMode == 0xFF)
|
|
return;
|
|
|
|
Chips_GeneralActions(p, 0x02); // Stop chips
|
|
p->PlayingMode = 0xFF;
|
|
|
|
return;
|
|
}
|
|
|
|
void RestartVGM(void *_p)
|
|
{
|
|
VGM_PLAYER* p = (VGM_PLAYER*)_p;
|
|
|
|
if (p->PlayingMode == 0xFF || ! p->VGMSmplPlayed)
|
|
return;
|
|
|
|
RestartPlaying(p);
|
|
|
|
return;
|
|
}
|
|
|
|
void SeekVGM(void *_p, bool Relative, INT32 PlayBkSamples)
|
|
{
|
|
INT32 Samples;
|
|
UINT32 LoopSmpls;
|
|
|
|
VGM_PLAYER* p = (VGM_PLAYER*)_p;
|
|
|
|
if (p->PlayingMode == 0xFF || (Relative && ! PlayBkSamples))
|
|
return;
|
|
|
|
LoopSmpls = p->VGMCurLoop * SampleVGM2Pbk_I(p, p->VGMHead.lngLoopSamples);
|
|
if (! Relative)
|
|
Samples = PlayBkSamples - (LoopSmpls + p->VGMSmplPlayed);
|
|
else
|
|
Samples = PlayBkSamples;
|
|
|
|
if (Samples < 0)
|
|
{
|
|
Samples = LoopSmpls + p->VGMSmplPlayed + Samples;
|
|
if (Samples < 0)
|
|
Samples = 0;
|
|
RestartPlaying(p);
|
|
}
|
|
|
|
p->ForceVGMExec = true;
|
|
InterpretFile(p, Samples);
|
|
p->ForceVGMExec = false;
|
|
|
|
return;
|
|
}
|
|
|
|
void RefreshMuting(void *_p)
|
|
{
|
|
VGM_PLAYER* p = (VGM_PLAYER*)_p;
|
|
Chips_GeneralActions(p, 0x10); // set muting mask
|
|
|
|
return;
|
|
}
|
|
|
|
void RefreshPanning(void *_p)
|
|
{
|
|
VGM_PLAYER* p = (VGM_PLAYER*)_p;
|
|
Chips_GeneralActions(p, 0x20); // set panning
|
|
|
|
return;
|
|
}
|
|
|
|
void RefreshPlaybackOptions(void *_p)
|
|
{
|
|
INT32 TempVol;
|
|
UINT8 CurChip;
|
|
CHIP_OPTS* TempCOpt1;
|
|
CHIP_OPTS* TempCOpt2;
|
|
|
|
VGM_PLAYER* p = (VGM_PLAYER*)_p;
|
|
|
|
if (p->VGMHead.bytVolumeModifier <= VOLUME_MODIF_WRAP)
|
|
TempVol = p->VGMHead.bytVolumeModifier;
|
|
else if (p->VGMHead.bytVolumeModifier == (VOLUME_MODIF_WRAP + 0x01))
|
|
TempVol = VOLUME_MODIF_WRAP - 0x100;
|
|
else
|
|
TempVol = p->VGMHead.bytVolumeModifier - 0x100;
|
|
p->VolumeLevelM = (float)(p->VolumeLevel * pow(2.0, TempVol / (double)0x20));
|
|
p->FinalVol = p->VolumeLevelM * p->MasterVol * p->MasterVol;
|
|
|
|
if (p->PlayingMode == 0xFF)
|
|
{
|
|
TempCOpt1 = (CHIP_OPTS*)&p->ChipOpts[0x00];
|
|
TempCOpt2 = (CHIP_OPTS*)&p->ChipOpts[0x01];
|
|
for (CurChip = 0x00; CurChip < CHIP_COUNT; CurChip ++, TempCOpt1 ++, TempCOpt2 ++)
|
|
{
|
|
TempCOpt2->EmuCore = TempCOpt1->EmuCore;
|
|
TempCOpt2->SpecialFlags = TempCOpt1->SpecialFlags;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
UINT32 GetGZFileLength(const char* FileName)
|
|
{
|
|
FILE* hFile;
|
|
UINT32 FileSize;
|
|
|
|
hFile = fopen(FileName, "rb");
|
|
if (hFile == NULL)
|
|
return 0xFFFFFFFF;
|
|
|
|
FileSize = GetGZFileLength_Internal(hFile);
|
|
|
|
fclose(hFile);
|
|
return FileSize;
|
|
}
|
|
|
|
#ifndef NO_WCHAR_FILENAMES
|
|
UINT32 GetGZFileLengthW(const wchar_t* FileName)
|
|
{
|
|
FILE* hFile;
|
|
UINT32 FileSize;
|
|
|
|
hFile = _wfopen(FileName, L"rb");
|
|
if (hFile == NULL)
|
|
return 0xFFFFFFFF;
|
|
|
|
FileSize = GetGZFileLength_Internal(hFile);
|
|
|
|
fclose(hFile);
|
|
return FileSize;
|
|
}
|
|
#endif
|
|
|
|
static UINT32 GetGZFileLength_Internal(FILE* hFile)
|
|
{
|
|
UINT32 FileSize;
|
|
UINT16 gzHead;
|
|
size_t RetVal;
|
|
|
|
RetVal = fread(&gzHead, 0x02, 0x01, hFile);
|
|
if (RetVal >= 1)
|
|
{
|
|
gzHead = ReadBE16((UINT8*)&gzHead);
|
|
if (gzHead != 0x1F8B)
|
|
{
|
|
RetVal = 0; // no .gz signature - treat as normal file
|
|
}
|
|
else
|
|
{
|
|
// .gz File
|
|
fseek(hFile, -4, SEEK_END);
|
|
// Note: In the error case it falls back to fseek/ftell.
|
|
RetVal = fread(&FileSize, 0x04, 0x01, hFile);
|
|
#ifdef VGM_BIG_ENDIAN
|
|
FileSize = ReadLE32((UINT8*)&FileSize);
|
|
#endif
|
|
}
|
|
}
|
|
if (RetVal <= 0)
|
|
{
|
|
// normal file
|
|
fseek(hFile, 0x00, SEEK_END);
|
|
FileSize = ftell(hFile);
|
|
}
|
|
|
|
return FileSize;
|
|
}
|
|
|
|
#ifndef NO_ZLIB
|
|
typedef struct vgm_file_gz
|
|
{
|
|
VGM_FILE vf;
|
|
gzFile hFile;
|
|
UINT32 Size;
|
|
} VGM_FILE_gz;
|
|
|
|
static int VGMF_gzread(VGM_FILE* hFile, void* ptr, UINT32 count)
|
|
{
|
|
VGM_FILE_gz* File = (VGM_FILE_gz *)hFile;
|
|
return gzread(File->hFile, ptr, count);
|
|
}
|
|
|
|
static int VGMF_gzseek(VGM_FILE* hFile, UINT32 offset)
|
|
{
|
|
VGM_FILE_gz* File = (VGM_FILE_gz *)hFile;
|
|
return gzseek(File->hFile, offset, SEEK_SET);
|
|
}
|
|
|
|
static UINT32 VGMF_gzgetsize(VGM_FILE* hFile)
|
|
{
|
|
VGM_FILE_gz* File = (VGM_FILE_gz *)hFile;
|
|
return File->Size;
|
|
}
|
|
#endif
|
|
|
|
bool OpenVGMFile(void *_p, const char* FileName)
|
|
{
|
|
#ifdef NO_ZLIB
|
|
return false;
|
|
#else
|
|
gzFile hFile;
|
|
UINT32 FileSize;
|
|
bool RetVal;
|
|
|
|
VGM_PLAYER* p = (VGM_PLAYER*)_p;
|
|
|
|
FileSize = GetGZFileLength(FileName);
|
|
|
|
hFile = gzopen(FileName, "rb");
|
|
if (hFile == NULL)
|
|
return false;
|
|
|
|
VGM_FILE_gz vgmFile;
|
|
|
|
vgmFile.vf.Read = VGMF_gzread;
|
|
vgmFile.vf.Seek = VGMF_gzseek;
|
|
vgmFile.vf.GetSize = VGMF_gzgetsize;
|
|
vgmFile.hFile = hFile;
|
|
vgmFile.Size = FileSize;
|
|
|
|
RetVal = OpenVGMFile_Internal(p, (VGM_FILE *)&vgmFile, FileSize);
|
|
|
|
gzclose(hFile);
|
|
return RetVal;
|
|
#endif
|
|
}
|
|
|
|
#ifndef NO_WCHAR_FILENAMES
|
|
bool OpenVGMFileW(void *_p, const wchar_t* FileName)
|
|
{
|
|
#ifdef NO_ZLIB
|
|
return false;
|
|
#else
|
|
gzFile hFile;
|
|
UINT32 FileSize;
|
|
bool RetVal;
|
|
|
|
VGM_PLAYER* p = (VGM_PLAYER*)_p;
|
|
|
|
#if ZLIB_VERNUM < 0x1270
|
|
int fDesc;
|
|
|
|
FileSize = GetGZFileLengthW(FileName);
|
|
|
|
fDesc = _wopen(FileName, _O_RDONLY | _O_BINARY);
|
|
hFile = gzdopen(fDesc, "rb");
|
|
if (hFile == NULL)
|
|
{
|
|
_close(fDesc);
|
|
return false;
|
|
}
|
|
#else
|
|
FileSize = GetGZFileLengthW(FileName);
|
|
|
|
hFile = gzopen_w(FileName, "rb");
|
|
if (hFile == NULL)
|
|
return false;
|
|
#endif
|
|
VGM_FILE_gz vgmFile;
|
|
|
|
vgmFile.vf.Read = VGMF_gzread;
|
|
vgmFile.vf.Seek = VGMF_gzseek;
|
|
vgmFile.vf.GetSize = VGMF_gzgetsize;
|
|
vgmFile.hFile = hFile;
|
|
vgmFile.Size = FileSize;
|
|
|
|
RetVal = OpenVGMFile_Internal(p, (VGM_FILE *)&vgmFile, FileSize);
|
|
|
|
gzclose(hFile);
|
|
return RetVal;
|
|
#endif
|
|
}
|
|
#endif
|
|
|
|
bool OpenVGMFile_Handle(void* _p, VGM_FILE* hFile)
|
|
{
|
|
UINT32 FileSize = hFile->GetSize(hFile);
|
|
return OpenVGMFile_Internal((VGM_PLAYER*)_p, hFile, FileSize);
|
|
}
|
|
|
|
static bool OpenVGMFile_Internal(VGM_PLAYER* p, VGM_FILE* hFile, UINT32 FileSize)
|
|
{
|
|
UINT32 fccHeader;
|
|
UINT32 CurPos;
|
|
UINT32 HdrLimit;
|
|
|
|
hFile->Seek(hFile, 0x00);
|
|
FILE_getLE32(hFile, &fccHeader);
|
|
if (fccHeader != FCC_VGM)
|
|
return false;
|
|
|
|
if (p->FileMode != 0xFF)
|
|
CloseVGMFile(p);
|
|
|
|
p->FileMode = 0x00;
|
|
p->VGMDataLen = FileSize;
|
|
|
|
hFile->Seek(hFile, 0x00);
|
|
ReadVGMHeader(hFile, &p->VGMHead);
|
|
|
|
p->VGMSampleRate = 44100;
|
|
if (! p->VGMDataLen)
|
|
p->VGMDataLen = p->VGMHead.lngEOFOffset;
|
|
if (! p->VGMHead.lngEOFOffset || p->VGMHead.lngEOFOffset > p->VGMDataLen)
|
|
{
|
|
p->VGMHead.lngEOFOffset = p->VGMDataLen;
|
|
}
|
|
if (p->VGMHead.lngLoopOffset && ! p->VGMHead.lngLoopSamples)
|
|
{
|
|
// 0-Sample-Loops causes the program to hangs in the playback routine
|
|
p->VGMHead.lngLoopOffset = 0x00000000;
|
|
}
|
|
if (p->VGMHead.lngDataOffset < 0x00000040)
|
|
{
|
|
p->VGMHead.lngDataOffset = 0x00000040;
|
|
}
|
|
|
|
memset(&p->VGMHeadX, 0x00, sizeof(VGM_HDR_EXTRA));
|
|
memset(&p->VGMH_Extra, 0x00, sizeof(VGM_EXTRA));
|
|
|
|
// Read Data
|
|
p->VGMDataLen = p->VGMHead.lngEOFOffset;
|
|
p->VGMData = (UINT8*)malloc(p->VGMDataLen);
|
|
if (p->VGMData == NULL)
|
|
return false;
|
|
hFile->Seek(hFile, 0x00);
|
|
hFile->Read(hFile, p->VGMData, p->VGMDataLen);
|
|
|
|
// Read Extra Header Data
|
|
if (p->VGMHead.lngExtraOffset)
|
|
{
|
|
UINT32* TempPtr;
|
|
|
|
CurPos = p->VGMHead.lngExtraOffset;
|
|
TempPtr = (UINT32*)&p->VGMHeadX;
|
|
// Read Header Size
|
|
p->VGMHeadX.DataSize = ReadLE32(&p->VGMData[CurPos]);
|
|
if (p->VGMHeadX.DataSize > sizeof(VGM_HDR_EXTRA))
|
|
p->VGMHeadX.DataSize = sizeof(VGM_HDR_EXTRA);
|
|
HdrLimit = CurPos + p->VGMHeadX.DataSize;
|
|
CurPos += 0x04;
|
|
TempPtr ++;
|
|
|
|
// Read all relative offsets of this header and make them absolute.
|
|
for (; CurPos < HdrLimit; CurPos += 0x04, TempPtr ++)
|
|
{
|
|
*TempPtr = ReadLE32(&p->VGMData[CurPos]);
|
|
if (*TempPtr)
|
|
*TempPtr += CurPos;
|
|
}
|
|
|
|
ReadChipExtraData32(p, p->VGMHeadX.Chp2ClkOffset, &p->VGMH_Extra.Clocks);
|
|
ReadChipExtraData16(p, p->VGMHeadX.ChpVolOffset, &p->VGMH_Extra.Volumes);
|
|
}
|
|
|
|
// Read GD3 Tag
|
|
HdrLimit = ReadGD3Tag(hFile, p->VGMHead.lngGD3Offset, &p->VGMTag);
|
|
if (HdrLimit == 0x10)
|
|
{
|
|
p->VGMHead.lngGD3Offset = 0x00000000;
|
|
//return false;
|
|
}
|
|
if (! p->VGMHead.lngGD3Offset)
|
|
{
|
|
// replace all NULL pointers with empty strings
|
|
p->VGMTag.strTrackNameE = MakeEmptyWStr();
|
|
p->VGMTag.strTrackNameJ = MakeEmptyWStr();
|
|
p->VGMTag.strGameNameE = MakeEmptyWStr();
|
|
p->VGMTag.strGameNameJ = MakeEmptyWStr();
|
|
p->VGMTag.strSystemNameE = MakeEmptyWStr();
|
|
p->VGMTag.strSystemNameJ = MakeEmptyWStr();
|
|
p->VGMTag.strAuthorNameE = MakeEmptyWStr();
|
|
p->VGMTag.strAuthorNameJ = MakeEmptyWStr();
|
|
p->VGMTag.strReleaseDate = MakeEmptyWStr();
|
|
p->VGMTag.strCreator = MakeEmptyWStr();
|
|
p->VGMTag.strNotes = MakeEmptyWStr();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static void ReadVGMHeader(VGM_FILE* hFile, VGM_HEADER* RetVGMHead)
|
|
{
|
|
VGM_HEADER CurHead;
|
|
UINT32 CurPos;
|
|
UINT32 HdrLimit;
|
|
|
|
hFile->Read(hFile, &CurHead, sizeof(VGM_HEADER));
|
|
#ifdef VGM_BIG_ENDIAN
|
|
{
|
|
UINT8* TempPtr;
|
|
|
|
// Warning: Lots of pointer casting ahead!
|
|
for (CurPos = 0x00; CurPos < sizeof(VGM_HEADER); CurPos += 0x04)
|
|
{
|
|
TempPtr = (UINT8*)&CurHead + CurPos;
|
|
switch(CurPos)
|
|
{
|
|
case 0x28:
|
|
// 0x28 [16-bit] SN76496 Feedback Mask
|
|
// 0x2A [ 8-bit] SN76496 Shift Register Width
|
|
// 0x2B [ 8-bit] SN76496 Flags
|
|
*(UINT16*)TempPtr = ReadLE16(TempPtr);
|
|
break;
|
|
case 0x78: // 78-7B [8-bit] AY8910 Type/Flags
|
|
case 0x7C: // 7C-7F [8-bit] Volume/Loop Modifiers
|
|
case 0x94: // 94-97 [8-bit] various flags
|
|
break;
|
|
default:
|
|
// everything else is 32-bit
|
|
*(UINT32*)TempPtr = ReadLE32(TempPtr);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// Header preperations
|
|
if (CurHead.lngVersion < 0x00000101)
|
|
{
|
|
CurHead.lngRate = 0;
|
|
}
|
|
if (CurHead.lngVersion < 0x00000110)
|
|
{
|
|
CurHead.shtPSG_Feedback = 0x0000;
|
|
CurHead.bytPSG_SRWidth = 0x00;
|
|
CurHead.lngHzYM2612 = CurHead.lngHzYM2413;
|
|
CurHead.lngHzYM2151 = CurHead.lngHzYM2413;
|
|
}
|
|
if (CurHead.lngVersion < 0x00000150)
|
|
{
|
|
CurHead.lngDataOffset = 0x00000000;
|
|
// If I would aim to be very strict, I would uncomment these few lines,
|
|
// but I sometimes use v1.51 Flags with v1.50 for better compatibility.
|
|
// (Some hyper-strict players refuse to play v1.51 files, even if there's
|
|
// no new chip used.)
|
|
//}
|
|
//if (CurHead.lngVersion < 0x00000151)
|
|
//{
|
|
CurHead.bytPSG_Flags = 0x00;
|
|
CurHead.lngHzSPCM = 0x0000;
|
|
CurHead.lngSPCMIntf = 0x00000000;
|
|
// all others are zeroed by memset
|
|
}
|
|
|
|
if (CurHead.lngHzPSG)
|
|
{
|
|
if (! CurHead.shtPSG_Feedback)
|
|
CurHead.shtPSG_Feedback = 0x0009;
|
|
if (! CurHead.bytPSG_SRWidth)
|
|
CurHead.bytPSG_SRWidth = 0x10;
|
|
}
|
|
|
|
// relative -> absolute addresses
|
|
if (CurHead.lngEOFOffset)
|
|
CurHead.lngEOFOffset += 0x00000004;
|
|
if (CurHead.lngGD3Offset)
|
|
CurHead.lngGD3Offset += 0x00000014;
|
|
if (CurHead.lngLoopOffset)
|
|
CurHead.lngLoopOffset += 0x0000001C;
|
|
|
|
if (CurHead.lngVersion < 0x00000150)
|
|
CurHead.lngDataOffset = 0x0000000C;
|
|
//if (CurHead.lngDataOffset < 0x0000000C)
|
|
// CurHead.lngDataOffset = 0x0000000C;
|
|
if (CurHead.lngDataOffset)
|
|
CurHead.lngDataOffset += 0x00000034;
|
|
|
|
CurPos = CurHead.lngDataOffset;
|
|
// should actually check v1.51 (first real usage of DataOffset)
|
|
// v1.50 is checked to support things like the Volume Modifiers in v1.50 files
|
|
if (CurHead.lngVersion < 0x00000150 /*0x00000151*/)
|
|
CurPos = 0x40;
|
|
if (! CurPos)
|
|
CurPos = 0x40;
|
|
HdrLimit = sizeof(VGM_HEADER);
|
|
if (HdrLimit > CurPos)
|
|
memset((UINT8*)&CurHead + CurPos, 0x00, HdrLimit - CurPos);
|
|
|
|
if (! CurHead.bytLoopModifier)
|
|
CurHead.bytLoopModifier = 0x10;
|
|
|
|
if (CurHead.lngExtraOffset)
|
|
{
|
|
CurHead.lngExtraOffset += 0xBC;
|
|
|
|
CurPos = CurHead.lngExtraOffset;
|
|
if (CurPos < HdrLimit)
|
|
memset((UINT8*)&CurHead + CurPos, 0x00, HdrLimit - CurPos);
|
|
}
|
|
|
|
if (CurHead.lngGD3Offset >= CurHead.lngEOFOffset)
|
|
CurHead.lngGD3Offset = 0x00;
|
|
if (CurHead.lngLoopOffset >= CurHead.lngEOFOffset)
|
|
CurHead.lngLoopOffset = 0x00;
|
|
if (CurHead.lngDataOffset >= CurHead.lngEOFOffset)
|
|
CurHead.lngDataOffset = 0x40;
|
|
if (CurHead.lngExtraOffset >= CurHead.lngEOFOffset)
|
|
CurHead.lngExtraOffset = 0x00;
|
|
|
|
*RetVGMHead = CurHead;
|
|
return;
|
|
}
|
|
|
|
static UINT8 ReadGD3Tag(VGM_FILE* hFile, UINT32 GD3Offset, GD3_TAG* RetGD3Tag)
|
|
{
|
|
UINT32 CurPos;
|
|
UINT32 TempLng;
|
|
UINT8 ResVal;
|
|
|
|
ResVal = 0x00;
|
|
|
|
// Read GD3 Tag
|
|
if (GD3Offset)
|
|
{
|
|
hFile->Seek(hFile, GD3Offset);
|
|
FILE_getLE32(hFile, &TempLng);
|
|
if (TempLng != FCC_GD3)
|
|
{
|
|
GD3Offset = 0x00000000;
|
|
ResVal = 0x10; // invalid GD3 offset
|
|
}
|
|
}
|
|
|
|
if (RetGD3Tag == NULL)
|
|
return ResVal;
|
|
|
|
if (! GD3Offset)
|
|
{
|
|
RetGD3Tag->fccGD3 = 0x00000000;
|
|
RetGD3Tag->lngVersion = 0x00000000;
|
|
RetGD3Tag->lngTagLength = 0x00000000;
|
|
RetGD3Tag->strTrackNameE = NULL;
|
|
RetGD3Tag->strTrackNameJ = NULL;
|
|
RetGD3Tag->strGameNameE = NULL;
|
|
RetGD3Tag->strGameNameJ = NULL;
|
|
RetGD3Tag->strSystemNameE = NULL;
|
|
RetGD3Tag->strSystemNameJ = NULL;
|
|
RetGD3Tag->strAuthorNameE = NULL;
|
|
RetGD3Tag->strAuthorNameJ = NULL;
|
|
RetGD3Tag->strReleaseDate = NULL;
|
|
RetGD3Tag->strCreator = NULL;
|
|
RetGD3Tag->strNotes = NULL;
|
|
}
|
|
else
|
|
{
|
|
//CurPos = GD3Offset;
|
|
//hFile->Seek(hFile, CurPos, SEEK_SET);
|
|
//CurPos += FILE_getLE32(hFile, &RetGD3Tag->fccGD3);
|
|
|
|
CurPos = GD3Offset + 0x04; // Save some back seeking, yay!
|
|
RetGD3Tag->fccGD3 = TempLng; // (That costs lots of CPU in .gz files.)
|
|
CurPos += FILE_getLE32(hFile, &RetGD3Tag->lngVersion);
|
|
CurPos += FILE_getLE32(hFile, &RetGD3Tag->lngTagLength);
|
|
|
|
TempLng = CurPos + RetGD3Tag->lngTagLength;
|
|
RetGD3Tag->strTrackNameE = ReadWStrFromFile(hFile, &CurPos, TempLng);
|
|
RetGD3Tag->strTrackNameJ = ReadWStrFromFile(hFile, &CurPos, TempLng);
|
|
RetGD3Tag->strGameNameE = ReadWStrFromFile(hFile, &CurPos, TempLng);
|
|
RetGD3Tag->strGameNameJ = ReadWStrFromFile(hFile, &CurPos, TempLng);
|
|
RetGD3Tag->strSystemNameE = ReadWStrFromFile(hFile, &CurPos, TempLng);
|
|
RetGD3Tag->strSystemNameJ = ReadWStrFromFile(hFile, &CurPos, TempLng);
|
|
RetGD3Tag->strAuthorNameE = ReadWStrFromFile(hFile, &CurPos, TempLng);
|
|
RetGD3Tag->strAuthorNameJ = ReadWStrFromFile(hFile, &CurPos, TempLng);
|
|
RetGD3Tag->strReleaseDate = ReadWStrFromFile(hFile, &CurPos, TempLng);
|
|
RetGD3Tag->strCreator = ReadWStrFromFile(hFile, &CurPos, TempLng);
|
|
RetGD3Tag->strNotes = ReadWStrFromFile(hFile, &CurPos, TempLng);
|
|
}
|
|
|
|
return ResVal;
|
|
}
|
|
|
|
static void ReadChipExtraData32(VGM_PLAYER* p, UINT32 StartOffset, VGMX_CHP_EXTRA32* ChpExtra)
|
|
{
|
|
UINT32 CurPos;
|
|
UINT8 CurChp;
|
|
VGMX_CHIP_DATA32* TempCD;
|
|
|
|
if (! StartOffset || StartOffset >= p->VGMDataLen)
|
|
{
|
|
ChpExtra->ChipCnt = 0x00;
|
|
ChpExtra->CCData = NULL;
|
|
return;
|
|
}
|
|
|
|
CurPos = StartOffset;
|
|
ChpExtra->ChipCnt = p->VGMData[CurPos];
|
|
if (ChpExtra->ChipCnt)
|
|
ChpExtra->CCData = (VGMX_CHIP_DATA32*)malloc(sizeof(VGMX_CHIP_DATA32) *
|
|
ChpExtra->ChipCnt);
|
|
else
|
|
ChpExtra->CCData = NULL;
|
|
CurPos ++;
|
|
|
|
for (CurChp = 0x00; CurChp < ChpExtra->ChipCnt; CurChp ++)
|
|
{
|
|
TempCD = &ChpExtra->CCData[CurChp];
|
|
TempCD->Type = p->VGMData[CurPos + 0x00];
|
|
TempCD->Data = ReadLE32(&p->VGMData[CurPos + 0x01]);
|
|
CurPos += 0x05;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
static void ReadChipExtraData16(VGM_PLAYER* p, UINT32 StartOffset, VGMX_CHP_EXTRA16* ChpExtra)
|
|
{
|
|
UINT32 CurPos;
|
|
UINT8 CurChp;
|
|
VGMX_CHIP_DATA16* TempCD;
|
|
|
|
if (! StartOffset || StartOffset >= p->VGMDataLen)
|
|
{
|
|
ChpExtra->ChipCnt = 0x00;
|
|
ChpExtra->CCData = NULL;
|
|
return;
|
|
}
|
|
|
|
CurPos = StartOffset;
|
|
ChpExtra->ChipCnt = p->VGMData[CurPos];
|
|
if (ChpExtra->ChipCnt)
|
|
ChpExtra->CCData = (VGMX_CHIP_DATA16*)malloc(sizeof(VGMX_CHIP_DATA16) *
|
|
ChpExtra->ChipCnt);
|
|
else
|
|
ChpExtra->CCData = NULL;
|
|
CurPos ++;
|
|
|
|
for (CurChp = 0x00; CurChp < ChpExtra->ChipCnt; CurChp ++)
|
|
{
|
|
TempCD = &ChpExtra->CCData[CurChp];
|
|
TempCD->Type = p->VGMData[CurPos + 0x00];
|
|
TempCD->Flags = p->VGMData[CurPos + 0x01];
|
|
TempCD->Data = ReadLE16(&p->VGMData[CurPos + 0x02]);
|
|
CurPos += 0x04;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
void CloseVGMFile(void *_p)
|
|
{
|
|
VGM_PLAYER* p = (VGM_PLAYER*)_p;
|
|
|
|
if (p->FileMode == 0xFF)
|
|
return;
|
|
|
|
p->VGMHead.fccVGM = 0x00;
|
|
free(p->VGMH_Extra.Clocks.CCData); p->VGMH_Extra.Clocks.CCData = NULL;
|
|
free(p->VGMH_Extra.Volumes.CCData); p->VGMH_Extra.Volumes.CCData = NULL;
|
|
free(p->VGMData); p->VGMData = NULL;
|
|
|
|
if (p->FileMode == 0x00)
|
|
FreeGD3Tag(&p->VGMTag);
|
|
|
|
p->FileMode = 0xFF;
|
|
|
|
return;
|
|
}
|
|
|
|
void FreeGD3Tag(GD3_TAG* TagData)
|
|
{
|
|
if (TagData == NULL)
|
|
return;
|
|
|
|
TagData->fccGD3 = 0x00;
|
|
free(TagData->strTrackNameE); TagData->strTrackNameE = NULL;
|
|
free(TagData->strTrackNameJ); TagData->strTrackNameJ = NULL;
|
|
free(TagData->strGameNameE); TagData->strGameNameE = NULL;
|
|
free(TagData->strGameNameJ); TagData->strGameNameJ = NULL;
|
|
free(TagData->strSystemNameE); TagData->strSystemNameE = NULL;
|
|
free(TagData->strSystemNameJ); TagData->strSystemNameJ = NULL;
|
|
free(TagData->strAuthorNameE); TagData->strAuthorNameE = NULL;
|
|
free(TagData->strAuthorNameJ); TagData->strAuthorNameJ = NULL;
|
|
free(TagData->strReleaseDate); TagData->strReleaseDate = NULL;
|
|
free(TagData->strCreator); TagData->strCreator = NULL;
|
|
free(TagData->strNotes); TagData->strNotes = NULL;
|
|
|
|
return;
|
|
}
|
|
|
|
static wchar_t* MakeEmptyWStr(void)
|
|
{
|
|
wchar_t* Str;
|
|
|
|
Str = (wchar_t*)malloc(0x01 * sizeof(wchar_t));
|
|
Str[0x00] = L'\0';
|
|
|
|
return Str;
|
|
}
|
|
|
|
static wchar_t* ReadWStrFromFile(VGM_FILE* hFile, UINT32* FilePos, UINT32 EOFPos)
|
|
{
|
|
// Note: Works with Windows (16-bit wchar_t) as well as Linux (32-bit wchar_t)
|
|
UINT32 CurPos;
|
|
wchar_t* TextStr;
|
|
wchar_t* TempStr;
|
|
UINT32 StrLen;
|
|
UINT16 UnicodeChr;
|
|
|
|
CurPos = *FilePos;
|
|
if (CurPos >= EOFPos)
|
|
return NULL;
|
|
TextStr = (wchar_t*)malloc((EOFPos - CurPos) / 0x02 * sizeof(wchar_t));
|
|
if (TextStr == NULL)
|
|
return NULL;
|
|
|
|
hFile->Seek(hFile, CurPos);
|
|
TempStr = TextStr - 1;
|
|
StrLen = 0x00;
|
|
do
|
|
{
|
|
TempStr ++;
|
|
FILE_getLE16(hFile, &UnicodeChr);
|
|
*TempStr = (wchar_t)UnicodeChr;
|
|
CurPos += 0x02;
|
|
StrLen ++;
|
|
if (CurPos >= EOFPos)
|
|
{
|
|
*TempStr = L'\0';
|
|
break;
|
|
}
|
|
} while(*TempStr != L'\0');
|
|
|
|
TextStr = (wchar_t*)realloc(TextStr, StrLen * sizeof(wchar_t));
|
|
*FilePos = CurPos;
|
|
|
|
return TextStr;
|
|
}
|
|
|
|
UINT32 GetVGMFileInfo(const char* FileName, VGM_HEADER* RetVGMHead, GD3_TAG* RetGD3Tag)
|
|
{
|
|
#ifdef NO_ZLIB
|
|
return 0;
|
|
#else
|
|
gzFile hFile;
|
|
UINT32 FileSize;
|
|
UINT32 RetVal;
|
|
|
|
FileSize = GetGZFileLength(FileName);
|
|
|
|
hFile = gzopen(FileName, "rb");
|
|
if (hFile == NULL)
|
|
return 0x00;
|
|
|
|
VGM_FILE_gz vgmFile;
|
|
|
|
vgmFile.vf.Read = VGMF_gzread;
|
|
vgmFile.vf.Seek = VGMF_gzseek;
|
|
vgmFile.vf.GetSize = VGMF_gzgetsize;
|
|
vgmFile.hFile = hFile;
|
|
vgmFile.Size = FileSize;
|
|
|
|
RetVal = GetVGMFileInfo_Internal((VGM_FILE *)&vgmFile, FileSize, RetVGMHead, RetGD3Tag);
|
|
|
|
gzclose(hFile);
|
|
return RetVal;
|
|
#endif
|
|
}
|
|
|
|
#ifndef NO_WCHAR_FILENAMES
|
|
UINT32 GetVGMFileInfoW(const wchar_t* FileName, VGM_HEADER* RetVGMHead, GD3_TAG* RetGD3Tag)
|
|
{
|
|
#ifdef NO_ZLIB
|
|
return 0;
|
|
#else
|
|
gzFile hFile;
|
|
UINT32 FileSize;
|
|
UINT32 RetVal;
|
|
#if ZLIB_VERNUM < 0x1270
|
|
int fDesc;
|
|
|
|
FileSize = GetGZFileLengthW(FileName);
|
|
|
|
fDesc = _wopen(FileName, _O_RDONLY | _O_BINARY);
|
|
hFile = gzdopen(fDesc, "rb");
|
|
if (hFile == NULL)
|
|
{
|
|
_close(fDesc);
|
|
return 0x00;
|
|
}
|
|
#else
|
|
FileSize = GetGZFileLengthW(FileName);
|
|
|
|
hFile = gzopen_w(FileName, "rb");
|
|
if (hFile == NULL)
|
|
return 0x00;
|
|
#endif
|
|
|
|
VGM_FILE_gz vgmFile;
|
|
|
|
vgmFile.vf.Read = VGMF_gzread;
|
|
vgmFile.vf.Seek = VGMF_gzseek;
|
|
vgmFile.vf.GetSize = VGMF_gzgetsize;
|
|
vgmFile.hFile = hFile;
|
|
vgmFile.Size = FileSize;
|
|
|
|
RetVal = GetVGMFileInfo_Internal((VGM_FILE *)&vgmFile, FileSize, RetVGMHead, RetGD3Tag);
|
|
|
|
gzclose(hFile);
|
|
return RetVal;
|
|
#endif
|
|
}
|
|
#endif
|
|
|
|
UINT32 GetVGMFileInfo_Handle(VGM_FILE* hFile, VGM_HEADER* RetVGMHead, GD3_TAG* RetGD3Tag)
|
|
{
|
|
UINT32 FileSize = hFile->GetSize(hFile);
|
|
return GetVGMFileInfo_Internal(hFile, FileSize, RetVGMHead, RetGD3Tag);
|
|
}
|
|
|
|
static UINT32 GetVGMFileInfo_Internal(VGM_FILE* hFile, UINT32 FileSize,
|
|
VGM_HEADER* RetVGMHead, GD3_TAG* RetGD3Tag)
|
|
{
|
|
// this is a copy-and-paste from OpenVGM, just a little stripped
|
|
UINT32 fccHeader;
|
|
UINT32 TempLng;
|
|
VGM_HEADER TempHead;
|
|
|
|
hFile->Seek(hFile, 0x00);
|
|
FILE_getLE32(hFile, &fccHeader);
|
|
if (fccHeader != FCC_VGM)
|
|
return 0x00;
|
|
|
|
if (RetVGMHead == NULL && RetGD3Tag == NULL)
|
|
return FileSize;
|
|
|
|
hFile->Seek(hFile, 0x00);
|
|
ReadVGMHeader(hFile, &TempHead);
|
|
|
|
if (! TempHead.lngEOFOffset || TempHead.lngEOFOffset > FileSize)
|
|
TempHead.lngEOFOffset = FileSize;
|
|
if (TempHead.lngDataOffset < 0x00000040)
|
|
TempHead.lngDataOffset = 0x00000040;
|
|
|
|
/*if (TempHead.lngGD3Offset)
|
|
{
|
|
gzseek(hFile, TempHead.lngGD3Offset, SEEK_SET);
|
|
gzgetLE32(hFile, &fccHeader);
|
|
if (fccHeader != FCC_GD3)
|
|
TempHead.lngGD3Offset = 0x00000000;
|
|
//return 0x00;
|
|
}*/
|
|
|
|
if (RetVGMHead != NULL)
|
|
*RetVGMHead = TempHead;
|
|
|
|
// Read GD3 Tag
|
|
if (RetGD3Tag != NULL)
|
|
TempLng = ReadGD3Tag(hFile, TempHead.lngGD3Offset, RetGD3Tag);
|
|
|
|
return FileSize;
|
|
}
|
|
|
|
INLINE UINT32 MulDivRound(UINT64 Number, UINT64 Numerator, UINT64 Denominator)
|
|
{
|
|
return (UINT32)((Number * Numerator + Denominator / 2) / Denominator);
|
|
}
|
|
|
|
UINT32 CalcSampleMSec(void* _p, UINT64 Value, UINT8 Mode)
|
|
{
|
|
// Mode:
|
|
// Bit 0 (01): Calculation Mode
|
|
// 0 - Sample2MSec
|
|
// 1 - MSec2Sample
|
|
// Bit 1 (02): Calculation Samlpe Rate
|
|
// 0 - current playback rate
|
|
// 1 - 44.1 KHz (VGM native)
|
|
UINT32 SmplRate;
|
|
UINT32 PbMul;
|
|
UINT32 PbDiv;
|
|
UINT32 RetVal;
|
|
|
|
VGM_PLAYER* p = (VGM_PLAYER *)_p;
|
|
|
|
if (! (Mode & 0x02))
|
|
{
|
|
SmplRate = p->SampleRate;
|
|
PbMul = 1;
|
|
PbDiv = 1;
|
|
}
|
|
else
|
|
{
|
|
SmplRate = p->VGMSampleRate;
|
|
PbMul = p->VGMPbRateMul;
|
|
PbDiv = p->VGMPbRateDiv;
|
|
}
|
|
|
|
switch(Mode & 0x01)
|
|
{
|
|
case 0x00:
|
|
RetVal = MulDivRound(Value, (UINT64)1000 * PbMul, (UINT64)SmplRate * PbDiv);
|
|
break;
|
|
case 0x01:
|
|
RetVal = MulDivRound(Value, (UINT64)SmplRate * PbDiv, (UINT64)1000 * PbMul);
|
|
break;
|
|
}
|
|
|
|
return RetVal;
|
|
}
|
|
|
|
UINT32 CalcSampleMSecExt(void *_p, UINT64 Value, UINT8 Mode, VGM_HEADER* FileHead)
|
|
{
|
|
// Note: This function was NOT tested with non-VGM formats!
|
|
|
|
// Mode: see function above
|
|
UINT32 SmplRate;
|
|
UINT32 PbMul;
|
|
UINT32 PbDiv;
|
|
UINT32 RetVal;
|
|
|
|
VGM_PLAYER* p = (VGM_PLAYER *)_p;
|
|
|
|
if (! (Mode & 0x02))
|
|
{
|
|
SmplRate = p->SampleRate;
|
|
PbMul = 1;
|
|
PbDiv = 1;
|
|
}
|
|
else
|
|
{
|
|
// TODO: make it work for non-VGM formats
|
|
// (i.e. get VGMSampleRate information from FileHead)
|
|
//
|
|
// But currently GetVGMFileInfo doesn't support them, it doesn't matter either way
|
|
SmplRate = 44100;
|
|
if (! p->VGMPbRate || ! FileHead->lngRate)
|
|
{
|
|
PbMul = 1;
|
|
PbDiv = 1;
|
|
}
|
|
else
|
|
{
|
|
PbMul = FileHead->lngRate;
|
|
PbDiv = p->VGMPbRate;
|
|
}
|
|
}
|
|
|
|
switch(Mode & 0x01)
|
|
{
|
|
case 0x00:
|
|
RetVal = MulDivRound(Value, 1000 * PbMul, SmplRate * PbDiv);
|
|
break;
|
|
case 0x01:
|
|
RetVal = MulDivRound(Value, SmplRate * PbDiv, 1000 * PbMul);
|
|
break;
|
|
}
|
|
|
|
return RetVal;
|
|
}
|
|
|
|
static UINT32 EncryptChipName(void* DstBuf, const void* SrcBuf, UINT32 Length)
|
|
{
|
|
// using nineko's awesome encryption algorithm
|
|
// http://forums.sonicretro.org/index.php?showtopic=25300
|
|
// based on C code by sasuke
|
|
const UINT8* SrcPos;
|
|
UINT8* DstPos;
|
|
UINT32 CurPos;
|
|
UINT8 CryptShift; // Src Bit/Dst Byte
|
|
UINT8 PlainShift; // Src Byte/Dst Bit
|
|
|
|
if (Length & 0x07)
|
|
return 0x00; // Length MUST be a multiple of 8
|
|
|
|
SrcPos = (const UINT8*)SrcBuf;
|
|
DstPos = (UINT8*)DstBuf;
|
|
for (CurPos = 0; CurPos < Length; CurPos += 8, SrcPos += 8, DstPos += 8)
|
|
{
|
|
for (CryptShift = 0; CryptShift < 8; CryptShift ++)
|
|
{
|
|
DstPos[CryptShift] = 0x00;
|
|
for (PlainShift = 0; PlainShift < 8; PlainShift ++)
|
|
{
|
|
if (SrcPos[PlainShift] & (1 << CryptShift))
|
|
DstPos[CryptShift] |= (1 << PlainShift);
|
|
}
|
|
}
|
|
}
|
|
|
|
return Length;
|
|
}
|
|
|
|
const char* GetChipName(UINT8 ChipID)
|
|
{
|
|
const char* CHIP_STRS[CHIP_COUNT] =
|
|
{ "SN76496", "YM2413", "YM2612", "YM2151", "SegaPCM", "RF5C68", "YM2203", "YM2608",
|
|
"YM2610", "YM3812", "YM3526", "Y8950", "YMF262", "YMF278B", "YMF271", "YMZ280B",
|
|
"RF5C164", "PWM", "AY8910", "GameBoy", "NES APU", "MultiPCM", "uPD7759", "OKIM6258",
|
|
"OKIM6295", "K051649", "K054539", "HuC6280", "C140", "K053260", "Pokey", "QSound",
|
|
"SCSP", "WSwan", "VSU", "SAA1099", "ES5503", "ES5506", "X1-010", "C352",
|
|
"GA20"};
|
|
|
|
/*if (ChipID == 0x21)
|
|
{
|
|
static char TempStr[0x08];
|
|
UINT32 TempData[2];
|
|
|
|
//EncryptChipName(TempData, "WSwan", 0x08);
|
|
TempData[0] = 0x1015170F;
|
|
TempData[1] = 0x001F1C07;
|
|
EncryptChipName(TempStr, TempData, 0x08);
|
|
return TempStr; // "WSwan"
|
|
}*/
|
|
|
|
if (ChipID < CHIP_COUNT)
|
|
return CHIP_STRS[ChipID];
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
const char* GetAccurateChipName(UINT8 ChipID, UINT8 SubType)
|
|
{
|
|
const char* RetStr;
|
|
static char TempStr[0x10];
|
|
|
|
if ((ChipID & 0x7F) >= CHIP_COUNT)
|
|
return NULL;
|
|
|
|
RetStr = NULL;
|
|
switch(ChipID & 0x7F)
|
|
{
|
|
case 0x00:
|
|
if (! (ChipID & 0x80))
|
|
{
|
|
switch(SubType)
|
|
{
|
|
case 0x01:
|
|
RetStr = "SN76489";
|
|
break;
|
|
case 0x02:
|
|
RetStr = "SN76489A";
|
|
break;
|
|
case 0x03:
|
|
RetStr = "SN76494";
|
|
break;
|
|
case 0x04:
|
|
RetStr = "SN76496";
|
|
break;
|
|
case 0x05:
|
|
RetStr = "SN94624";
|
|
break;
|
|
case 0x06:
|
|
RetStr = "NCR7496";
|
|
break;
|
|
case 0x07:
|
|
RetStr = "SEGA PSG";
|
|
break;
|
|
default:
|
|
RetStr = "SN76496";
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
RetStr = "T6W28";
|
|
}
|
|
break;
|
|
case 0x01:
|
|
if (ChipID & 0x80)
|
|
RetStr = "VRC7";
|
|
break;
|
|
case 0x04:
|
|
RetStr = "Sega PCM";
|
|
break;
|
|
case 0x08:
|
|
if (! (ChipID & 0x80))
|
|
RetStr = "YM2610";
|
|
else
|
|
RetStr = "YM2610B";
|
|
break;
|
|
case 0x12: // AY8910
|
|
switch(SubType)
|
|
{
|
|
case 0x00:
|
|
RetStr = "AY-3-8910";
|
|
break;
|
|
case 0x01:
|
|
RetStr = "AY-3-8912";
|
|
break;
|
|
case 0x02:
|
|
RetStr = "AY-3-8913";
|
|
break;
|
|
case 0x03:
|
|
RetStr = "AY8930";
|
|
break;
|
|
case 0x04:
|
|
RetStr = "AY-3-8914";
|
|
break;
|
|
case 0x10:
|
|
RetStr = "YM2149";
|
|
break;
|
|
case 0x11:
|
|
RetStr = "YM3439";
|
|
break;
|
|
case 0x12:
|
|
RetStr = "YMZ284";
|
|
break;
|
|
case 0x13:
|
|
RetStr = "YMZ294";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x13:
|
|
RetStr = "GB DMG";
|
|
break;
|
|
case 0x14:
|
|
if (! (ChipID & 0x80))
|
|
RetStr = "NES APU";
|
|
else
|
|
RetStr = "NES APU + FDS";
|
|
break;
|
|
case 0x1C:
|
|
switch(SubType)
|
|
{
|
|
case 0x00:
|
|
case 0x01:
|
|
RetStr = "C140";
|
|
break;
|
|
case 0x02:
|
|
RetStr = "C140 (219)";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x21:
|
|
RetStr = "WonderSwan";
|
|
break;
|
|
case 0x22:
|
|
RetStr = "VSU-VUE";
|
|
break;
|
|
case 0x25:
|
|
if (! (ChipID & 0x80))
|
|
RetStr = "ES5505";
|
|
else
|
|
RetStr = "ES5506";
|
|
break;
|
|
case 0x28:
|
|
RetStr = "Irem GA20";
|
|
break;
|
|
}
|
|
// catch all default-cases
|
|
if (RetStr == NULL)
|
|
RetStr = GetChipName(ChipID & 0x7F);
|
|
|
|
return RetStr;
|
|
}
|
|
|
|
UINT32 GetChipClock(void* _p, UINT8 ChipID, UINT8* RetSubType)
|
|
{
|
|
UINT32 Clock;
|
|
UINT8 SubType;
|
|
UINT8 CurChp;
|
|
bool AllowBit31;
|
|
|
|
VGM_PLAYER* p = (VGM_PLAYER *)_p;
|
|
|
|
VGM_HEADER* FileHead = &p->VGMHead;
|
|
|
|
SubType = 0x00;
|
|
AllowBit31 = 0x00;
|
|
switch(ChipID & 0x7F)
|
|
{
|
|
case 0x00:
|
|
Clock = FileHead->lngHzPSG;
|
|
AllowBit31 = 0x01; // T6W28 Mode
|
|
if (RetSubType != NULL && ! (Clock & 0x80000000)) // The T6W28 is handles differently.
|
|
{
|
|
switch(FileHead->bytPSG_SRWidth)
|
|
{
|
|
case 0x0F: // 0x4000
|
|
if (FileHead->bytPSG_Flags & 0x08) // Clock Divider == 1?
|
|
SubType = 0x05; // SN94624
|
|
else
|
|
SubType = 0x01; // SN76489
|
|
break;
|
|
case 0x10: // 0x8000
|
|
if (FileHead->shtPSG_Feedback == 0x0009)
|
|
SubType = 0x07; // SEGA PSG
|
|
else if (FileHead->shtPSG_Feedback == 0x0022)
|
|
SubType = 0x06; // NCR7496
|
|
break;
|
|
case 0x11: // 0x10000
|
|
if (FileHead->bytPSG_Flags & 0x08) // Clock Divider == 1?
|
|
SubType = 0x03; // SN76494
|
|
else
|
|
SubType = 0x02; // SN76489A/SN76496
|
|
break;
|
|
}
|
|
/*
|
|
FbMask Noise Taps Negate Stereo Dv Freq0 Fb SR Flags
|
|
01 SN76489 0x4000, 0x01, 0x02, TRUE, FALSE, 8, TRUE 03 0F 07 (02|04|00|01) [unverified]
|
|
02 SN76489A 0x10000, 0x04, 0x08, FALSE, FALSE, 8, TRUE 0C 11 05 (00|04|00|01)
|
|
03 SN76494 0x10000, 0x04, 0x08, FALSE, FALSE, 1, TRUE 0C 11 0D (00|04|08|01)
|
|
04 SN76496 0x10000, 0x04, 0x08, FALSE, FALSE, 8, TRUE 0C 11 05 (00|04|00|01) [same as SN76489A]
|
|
05 SN94624 0x4000, 0x01, 0x02, TRUE, FALSE, 1, TRUE 03 0F 0F (02|04|08|01) [unverified, SN76489A without /8]
|
|
06 NCR7496 0x8000, 0x02, 0x20, FALSE, FALSE, 8, TRUE 22 10 05 (00|04|00|01) [unverified]
|
|
07 GameGear PSG 0x8000, 0x01, 0x08, TRUE, TRUE, 8, FALSE 09 10 02 (02|00|00|00)
|
|
07 SEGA VDP PSG 0x8000, 0x01, 0x08, TRUE, FALSE, 8, FALSE 09 10 06 (02|04|00|00)
|
|
01 U8106 0x4000, 0x01, 0x02, TRUE, FALSE, 8, TRUE 03 0F 07 (02|04|00|01) [unverified, same as SN76489]
|
|
02 Y2404 0x10000, 0x04, 0x08, FALSE, FALSE; 8, TRUE 0C 11 05 (00|04|00|01) [unverified, same as SN76489A]
|
|
-- T6W28 0x4000, 0x01, 0x04, ????, FALSE, 8, ???? 05 0F ?? (??|??|00|01) [It IS stereo, but not in GameGear way].
|
|
*/
|
|
}
|
|
break;
|
|
case 0x01:
|
|
Clock = FileHead->lngHzYM2413;
|
|
AllowBit31 = 0x01; // VRC7 Mode
|
|
break;
|
|
case 0x02:
|
|
Clock = FileHead->lngHzYM2612;
|
|
break;
|
|
case 0x03:
|
|
Clock = FileHead->lngHzYM2151;
|
|
break;
|
|
case 0x04:
|
|
Clock = FileHead->lngHzSPCM;
|
|
break;
|
|
case 0x05:
|
|
if (ChipID & 0x80)
|
|
return 0;
|
|
Clock = FileHead->lngHzRF5C68;
|
|
break;
|
|
case 0x06:
|
|
Clock = FileHead->lngHzYM2203;
|
|
break;
|
|
case 0x07:
|
|
Clock = FileHead->lngHzYM2608;
|
|
break;
|
|
case 0x08:
|
|
Clock = FileHead->lngHzYM2610;
|
|
AllowBit31 = 0x01; // YM2610B Mode
|
|
break;
|
|
case 0x09:
|
|
Clock = FileHead->lngHzYM3812;
|
|
AllowBit31 = 0x01; // Dual OPL2, panned to the L/R speakers
|
|
break;
|
|
case 0x0A:
|
|
Clock = FileHead->lngHzYM3526;
|
|
break;
|
|
case 0x0B:
|
|
Clock = FileHead->lngHzY8950;
|
|
break;
|
|
case 0x0C:
|
|
Clock = FileHead->lngHzYMF262;
|
|
break;
|
|
case 0x0D:
|
|
Clock = FileHead->lngHzYMF278B;
|
|
break;
|
|
case 0x0E:
|
|
Clock = FileHead->lngHzYMF271;
|
|
break;
|
|
case 0x0F:
|
|
Clock = FileHead->lngHzYMZ280B;
|
|
break;
|
|
case 0x10:
|
|
if (ChipID & 0x80)
|
|
return 0;
|
|
Clock = FileHead->lngHzRF5C164;
|
|
AllowBit31 = 0x01; // hack for Cosmic Fantasy Stories
|
|
break;
|
|
case 0x11:
|
|
if (ChipID & 0x80)
|
|
return 0;
|
|
Clock = FileHead->lngHzPWM;
|
|
break;
|
|
case 0x12:
|
|
Clock = FileHead->lngHzAY8910;
|
|
SubType = FileHead->bytAYType;
|
|
break;
|
|
case 0x13:
|
|
Clock = FileHead->lngHzGBDMG;
|
|
break;
|
|
case 0x14:
|
|
Clock = FileHead->lngHzNESAPU;
|
|
AllowBit31 = 0x01; // FDS Enable
|
|
break;
|
|
case 0x15:
|
|
Clock = FileHead->lngHzMultiPCM;
|
|
break;
|
|
case 0x16:
|
|
Clock = FileHead->lngHzUPD7759;
|
|
AllowBit31 = 0x01; // Master/Slave Bit
|
|
break;
|
|
case 0x17:
|
|
Clock = FileHead->lngHzOKIM6258;
|
|
break;
|
|
case 0x18:
|
|
Clock = FileHead->lngHzOKIM6295;
|
|
AllowBit31 = 0x01; // Pin 7 State
|
|
break;
|
|
case 0x19:
|
|
Clock = FileHead->lngHzK051649;
|
|
break;
|
|
case 0x1A:
|
|
Clock = FileHead->lngHzK054539;
|
|
break;
|
|
case 0x1B:
|
|
Clock = FileHead->lngHzHuC6280;
|
|
break;
|
|
case 0x1C:
|
|
Clock = FileHead->lngHzC140;
|
|
SubType = FileHead->bytC140Type;
|
|
break;
|
|
case 0x1D:
|
|
Clock = FileHead->lngHzK053260;
|
|
break;
|
|
case 0x1E:
|
|
Clock = FileHead->lngHzPokey;
|
|
break;
|
|
case 0x1F:
|
|
if (ChipID & 0x80)
|
|
return 0;
|
|
Clock = FileHead->lngHzQSound;
|
|
break;
|
|
case 0x20:
|
|
Clock = FileHead->lngHzSCSP;
|
|
break;
|
|
case 0x21:
|
|
Clock = FileHead->lngHzWSwan;
|
|
break;
|
|
case 0x22:
|
|
Clock = FileHead->lngHzVSU;
|
|
break;
|
|
case 0x23:
|
|
Clock = FileHead->lngHzSAA1099;
|
|
break;
|
|
case 0x24:
|
|
Clock = FileHead->lngHzES5503;
|
|
break;
|
|
case 0x25:
|
|
Clock = FileHead->lngHzES5506;
|
|
AllowBit31 = 0x01; // ES5505/5506 switch
|
|
break;
|
|
case 0x26:
|
|
Clock = FileHead->lngHzX1_010;
|
|
break;
|
|
case 0x27:
|
|
Clock = FileHead->lngHzC352;
|
|
break;
|
|
case 0x28:
|
|
Clock = FileHead->lngHzGA20;
|
|
break;
|
|
default:
|
|
return 0;
|
|
}
|
|
if (ChipID & 0x80)
|
|
{
|
|
VGMX_CHP_EXTRA32* TempCX;
|
|
|
|
if (! (Clock & 0x40000000))
|
|
return 0;
|
|
|
|
ChipID &= 0x7F;
|
|
TempCX = &p->VGMH_Extra.Clocks;
|
|
for (CurChp = 0x00; CurChp < TempCX->ChipCnt; CurChp ++)
|
|
{
|
|
if (TempCX->CCData[CurChp].Type == ChipID)
|
|
{
|
|
if (TempCX->CCData[CurChp].Data)
|
|
Clock = TempCX->CCData[CurChp].Data;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (RetSubType != NULL)
|
|
*RetSubType = SubType;
|
|
if (AllowBit31)
|
|
return Clock & 0xBFFFFFFF;
|
|
else
|
|
return Clock & 0x3FFFFFFF;
|
|
}
|
|
|
|
static UINT16 GetChipVolume(VGM_PLAYER* p, UINT8 ChipID, UINT8 ChipNum, UINT8 ChipCnt)
|
|
{
|
|
// ChipID: ID of Chip
|
|
// Bit 7 - Is Paired Chip
|
|
// ChipNum: chip number (0 - first chip, 1 - second chip)
|
|
// ChipCnt: chip volume divider (number of used chips)
|
|
const UINT16 CHIP_VOLS[CHIP_COUNT] =
|
|
{ 0x80, 0x200/*0x155*/, 0x100, 0x100, 0x180, 0xB0, 0x100, 0x80, // 00-07
|
|
0x80, 0x100, 0x100, 0x100, 0x100, 0x100, 0x100, 0x98, // 08-0F
|
|
0x80, 0xE0/*0xCD*/, 0x100, 0xC0, 0x100, 0x40, 0x11E, 0x1C0, // 10-17
|
|
0x100/*110*/, 0xA0, 0x100, 0x100, 0x100, 0xB3, 0x100, 0x100, // 18-1F
|
|
0x100, 0x100, 0x100, 0x100, 0x40, 0x20, 0x100, 0x40, // 20-27
|
|
0x280};
|
|
UINT16 Volume;
|
|
UINT8 CurChp;
|
|
VGMX_CHP_EXTRA16* TempCX;
|
|
VGMX_CHIP_DATA16* TempCD;
|
|
|
|
VGM_HEADER* FileHead = &p->VGMHead;
|
|
|
|
Volume = CHIP_VOLS[ChipID & 0x7F];
|
|
switch(ChipID)
|
|
{
|
|
case 0x00: // SN76496
|
|
// if T6W28, set Volume Divider to 01
|
|
if (GetChipClock(p, (ChipID << 7) | ChipID, NULL) & 0x80000000)
|
|
{
|
|
// The T6W28 consists of 2 "half" chips.
|
|
ChipNum = 0x01;
|
|
ChipCnt = 0x01;
|
|
}
|
|
break;
|
|
case 0x18: // OKIM6295
|
|
// CP System 1 patch
|
|
if (p->VGMTag.strSystemNameE != NULL && ! wcsncmp(p->VGMTag.strSystemNameE, L"CP", 0x02))
|
|
Volume = 110;
|
|
break;
|
|
case 0x86: // YM2203's AY
|
|
Volume /= 2;
|
|
break;
|
|
case 0x87: // YM2608's AY
|
|
// The YM2608 outputs twice as loud as the YM2203 here.
|
|
//Volume *= 1;
|
|
break;
|
|
case 0x88: // YM2610's AY
|
|
//Volume *= 1;
|
|
break;
|
|
}
|
|
if (ChipCnt > 1)
|
|
Volume /= ChipCnt;
|
|
|
|
TempCX = &p->VGMH_Extra.Volumes;
|
|
TempCD = TempCX->CCData;
|
|
for (CurChp = 0x00; CurChp < TempCX->ChipCnt; CurChp ++, TempCD ++)
|
|
{
|
|
if (TempCD->Type == ChipID && (TempCD->Flags & 0x01) == ChipNum)
|
|
{
|
|
// Bit 15 - absolute/relative volume
|
|
// 0 - absolute
|
|
// 1 - relative (0x0100 = 1.0, 0x80 = 0.5, etc.)
|
|
if (TempCD->Data & 0x8000)
|
|
Volume = (Volume * (TempCD->Data & 0x7FFF) + 0x80) >> 8;
|
|
else
|
|
{
|
|
Volume = TempCD->Data;
|
|
if ((ChipID & 0x80) && p->DoubleSSGVol)
|
|
Volume *= 2;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
return Volume;
|
|
}
|
|
|
|
|
|
static void RestartPlaying(VGM_PLAYER* p)
|
|
{
|
|
p->VGMPos = p->VGMHead.lngDataOffset;
|
|
p->VGMSmplPos = 0;
|
|
p->VGMSmplPlayed = 0;
|
|
p->VGMEnd = false;
|
|
p->EndPlay = false;
|
|
p->VGMCurLoop = 0x00;
|
|
|
|
Chips_GeneralActions(p, 0x01); // Reset Chips
|
|
// also does Muting Mask (0x10) and Panning (0x20)
|
|
|
|
p->Last95Drum = 0xFFFF;
|
|
p->Last95Freq = 0;
|
|
p->ForceVGMExec = true;
|
|
p->IsVGMInit = true;
|
|
InterpretFile(p, 0);
|
|
p->IsVGMInit = false;
|
|
p->ForceVGMExec = false;
|
|
|
|
return;
|
|
}
|
|
|
|
static void Chips_GeneralActions(VGM_PLAYER* p, UINT8 Mode)
|
|
{
|
|
UINT32 AbsVol;
|
|
//UINT16 ChipVol;
|
|
CAUD_ATTR* CAA;
|
|
CHIP_OPTS* COpt;
|
|
UINT8 ChipCnt;
|
|
UINT8 CurChip;
|
|
UINT8 CurCSet; // Chip Set
|
|
UINT32 MaskVal;
|
|
UINT32 ChipClk;
|
|
|
|
switch(Mode)
|
|
{
|
|
case 0x00: // Start Chips
|
|
for (CurCSet = 0x00; CurCSet < 0x02; CurCSet ++)
|
|
{
|
|
CAA = (CAUD_ATTR*)&p->ChipAudio[CurCSet];
|
|
for (CurChip = 0x00; CurChip < CHIP_COUNT; CurChip ++, CAA ++)
|
|
{
|
|
CAA->SmpRate = 0x00;
|
|
CAA->Volume = 0x00;
|
|
CAA->ChipType = 0xFF;
|
|
CAA->ChipID = CurCSet;
|
|
CAA->Resampler = 0x00;
|
|
CAA->StreamUpdate = &null_update;
|
|
CAA->StreamUpdateParam = NULL;
|
|
CAA->Paired = NULL;
|
|
}
|
|
CAA = p->CA_Paired[CurCSet];
|
|
for (CurChip = 0x00; CurChip < 0x03; CurChip ++, CAA ++)
|
|
{
|
|
CAA->SmpRate = 0x00;
|
|
CAA->Volume = 0x00;
|
|
CAA->ChipType = 0xFF;
|
|
CAA->ChipID = CurCSet;
|
|
CAA->Resampler = 0x00;
|
|
CAA->StreamUpdate = &null_update;
|
|
CAA->StreamUpdateParam = NULL;
|
|
CAA->Paired = NULL;
|
|
}
|
|
}
|
|
|
|
// Initialize Sound Chips
|
|
AbsVol = 0x00;
|
|
if (p->VGMHead.lngHzPSG)
|
|
{
|
|
//ChipVol = UseFM ? 0x00 : 0x80;
|
|
p->ChipOpts[0x01].SN76496.EmuCore = p->ChipOpts[0x00].SN76496.EmuCore;
|
|
|
|
ChipCnt = (p->VGMHead.lngHzPSG & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].SN76496;
|
|
CAA->ChipType = 0x00;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
ChipClk &= ~0x80000000;
|
|
ChipClk |= p->VGMHead.lngHzPSG & ((CurChip & 0x01) << 31);
|
|
CAA->SmpRate = device_start_sn764xx(&p->sn764xx[CurChip],
|
|
p->ChipOpts[CurChip].SN76496.EmuCore, ChipClk, p->SampleRate,
|
|
p->VGMHead.bytPSG_SRWidth,
|
|
p->VGMHead.shtPSG_Feedback,
|
|
(p->VGMHead.bytPSG_Flags & 0x02) >> 1,
|
|
(p->VGMHead.bytPSG_Flags & 0x04) >> 2,
|
|
(p->VGMHead.bytPSG_Flags & 0x08) >> 3,
|
|
(p->VGMHead.bytPSG_Flags & 0x01) >> 0);
|
|
CAA->StreamUpdate = &sn764xx_stream_update;
|
|
CAA->StreamUpdateParam = p->sn764xx[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
if (! CurChip || ! (ChipClk & 0x80000000))
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
if (p->VGMHead.lngHzPSG & 0x80000000)
|
|
ChipCnt = 0x01;
|
|
}
|
|
if (p->VGMHead.lngHzYM2413)
|
|
{
|
|
//ChipVol = UseFM ? 0x00 : 0x200/*0x155*/;
|
|
p->ChipOpts[0x01].YM2413.EmuCore = p->ChipOpts[0x00].YM2413.EmuCore;
|
|
|
|
ChipCnt = (p->VGMHead.lngHzYM2413 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].YM2413;
|
|
CAA->ChipType = 0x01;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_ym2413(&p->ym2413[CurChip], p->ChipOpts[CurChip].YM2413.EmuCore, ChipClk, p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &ym2413_stream_update;
|
|
CAA->StreamUpdateParam = p->ym2413[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
// WHY has this chip such a low volume???
|
|
//AbsVol += (CAA->Volume + 1) * 3 / 4;
|
|
AbsVol += CAA->Volume / 2;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzYM2612)
|
|
{
|
|
//ChipVol = 0x100;
|
|
p->ChipOpts[0x01].YM2612.EmuCore = p->ChipOpts[0x00].YM2612.EmuCore;
|
|
p->ChipOpts[0x01].YM2612.SpecialFlags = p->ChipOpts[0x00].YM2612.SpecialFlags;
|
|
|
|
ChipCnt = (p->VGMHead.lngHzYM2612 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].YM2612;
|
|
CAA->ChipType = 0x02;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_ym2612(&p->ym2612[CurChip], p->ChipOpts[CurChip].YM2612.EmuCore, p->ChipOpts[CurChip].YM2612.SpecialFlags, ChipClk, p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE, &p->IsVGMInit);
|
|
CAA->StreamUpdate = &ym2612_stream_update;
|
|
CAA->StreamUpdateParam = p->ym2612[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzYM2151)
|
|
{
|
|
//ChipVol = 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzYM2151 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].YM2151;
|
|
CAA->ChipType = 0x03;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_ym2151(&p->ym2151[CurChip], ChipClk, p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &ym2151_update;
|
|
CAA->StreamUpdateParam = p->ym2151[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzSPCM)
|
|
{
|
|
//ChipVol = 0x180;
|
|
ChipCnt = (p->VGMHead.lngHzSPCM & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].SegaPCM;
|
|
CAA->ChipType = 0x04;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_segapcm(&p->segapcm[CurChip], ChipClk, p->VGMHead.lngSPCMIntf);
|
|
CAA->StreamUpdate = &SEGAPCM_update;
|
|
CAA->StreamUpdateParam = p->segapcm[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzRF5C68)
|
|
{
|
|
//ChipVol = 0xB0; // that's right according to MAME, but it's almost too loud
|
|
ChipCnt = 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].RF5C68;
|
|
CAA->ChipType = 0x05;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_rf5c68(&p->rf5c68, ChipClk);
|
|
CAA->StreamUpdate = &rf5c68_update;
|
|
CAA->StreamUpdateParam = p->rf5c68;
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzYM2203)
|
|
{
|
|
//ChipVol = 0x100;
|
|
p->ChipOpts[0x01].YM2203.EmuCore = p->ChipOpts[0x00].YM2203.EmuCore;
|
|
p->ChipOpts[0x01].YM2203.SpecialFlags = p->ChipOpts[0x00].YM2203.SpecialFlags;
|
|
|
|
ChipCnt = (p->VGMHead.lngHzYM2203 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].YM2203;
|
|
COpt = &p->ChipOpts[CurChip].YM2203;
|
|
CAA->ChipType = 0x06;
|
|
CAA->Paired = &p->CA_Paired[CurChip][0x00];
|
|
CAA->Paired->ChipType = 0x80 | CAA->ChipType;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_ym2203(&p->ym2203[CurChip], COpt->EmuCore,
|
|
ChipClk, COpt->SpecialFlags & 0x01,
|
|
p->VGMHead.bytAYFlagYM2203,
|
|
(int*) &CAA->Paired->SmpRate,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &ym2203_stream_update;
|
|
CAA->StreamUpdateParam = p->ym2203[CurChip];
|
|
CAA->Paired->StreamUpdate = &ym2203_stream_update_ay;
|
|
CAA->Paired->StreamUpdateParam = p->ym2203[CurChip];
|
|
ym2203_set_srchg_cb(p->ym2203[CurChip], &ChangeChipSampleRate, CAA, CAA->Paired);
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
CAA->Paired->Volume = GetChipVolume(p, CAA->Paired->ChipType,
|
|
CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume + CAA->Paired->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzYM2608)
|
|
{
|
|
//ChipVol = 0x80;
|
|
p->ChipOpts[0x01].YM2608.EmuCore = p->ChipOpts[0x00].YM2608.EmuCore;
|
|
p->ChipOpts[0x01].YM2608.SpecialFlags = p->ChipOpts[0x00].YM2608.SpecialFlags;
|
|
|
|
ChipCnt = (p->VGMHead.lngHzYM2608 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].YM2608;
|
|
COpt = &p->ChipOpts[CurChip].YM2608;
|
|
CAA->ChipType = 0x07;
|
|
CAA->Paired = &p->CA_Paired[CurChip][0x01];
|
|
CAA->Paired->ChipType = 0x80 | CAA->ChipType;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_ym2608(&p->ym2608[CurChip], COpt->EmuCore,
|
|
ChipClk, COpt->SpecialFlags & 0x01,
|
|
p->VGMHead.bytAYFlagYM2608,
|
|
(int*) &CAA->Paired->SmpRate,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &ym2608_stream_update;
|
|
CAA->StreamUpdateParam = p->ym2608[CurChip];
|
|
CAA->Paired->StreamUpdate = &ym2608_stream_update_ay;
|
|
CAA->Paired->StreamUpdateParam = p->ym2608[CurChip];
|
|
ym2608_set_srchg_cb(p->ym2608[CurChip], &ChangeChipSampleRate, CAA, CAA->Paired);
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
CAA->Paired->Volume = GetChipVolume(p, CAA->Paired->ChipType,
|
|
CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume + CAA->Paired->Volume;
|
|
//CAA->Volume = ChipVol;
|
|
//CAA->Paired->Volume = ChipVol * 2;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzYM2610)
|
|
{
|
|
//ChipVol = 0x80;
|
|
p->ChipOpts[0x01].YM2610.EmuCore = p->ChipOpts[0x00].YM2610.EmuCore;
|
|
p->ChipOpts[0x01].YM2610.SpecialFlags = p->ChipOpts[0x00].YM2610.SpecialFlags;
|
|
|
|
ChipCnt = (p->VGMHead.lngHzYM2610 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].YM2610;
|
|
COpt = &p->ChipOpts[CurChip].YM2610;
|
|
CAA->ChipType = 0x08;
|
|
CAA->Paired = &p->CA_Paired[CurChip][0x02];
|
|
CAA->Paired->ChipType = 0x80 | CAA->ChipType;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_ym2610(&p->ym2610[CurChip], COpt->EmuCore,
|
|
ChipClk, COpt->SpecialFlags & 0x01,
|
|
(int*) &CAA->Paired->SmpRate,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = (ChipClk & 0x80000000) ? ym2610b_stream_update :
|
|
ym2610_stream_update;
|
|
CAA->StreamUpdateParam = p->ym2610[CurChip];
|
|
CAA->Paired->StreamUpdate = &ym2610_stream_update_ay;
|
|
CAA->Paired->StreamUpdateParam = p->ym2610[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
CAA->Paired->Volume = GetChipVolume(p, CAA->Paired->ChipType,
|
|
CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume + CAA->Paired->Volume;
|
|
//CAA->Volume = ChipVol;
|
|
//CAA->Paired->Volume = ChipVol * 2;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzYM3812)
|
|
{
|
|
//ChipVol = UseFM ? 0x00 : 0x100;
|
|
p->ChipOpts[0x01].YM3812.EmuCore = p->ChipOpts[0x00].YM3812.EmuCore;
|
|
|
|
ChipCnt = (p->VGMHead.lngHzYM3812 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].YM3812;
|
|
CAA->ChipType = 0x09;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_ym3812(&p->ym3812[CurChip],
|
|
p->ChipOpts[CurChip].YM3812.EmuCore,
|
|
ChipClk,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
if (ChipClk & 0x80000000)
|
|
{
|
|
struct dual_opl2_info * info = (struct dual_opl2_info *) malloc(sizeof(struct dual_opl2_info));
|
|
|
|
CAA->StreamUpdate = dual_opl2_stereo;
|
|
CAA->StreamUpdateParam = (void *) info;
|
|
|
|
info->chip = p->ym3812[CurChip];
|
|
info->ChipID = CurChip;
|
|
|
|
p->ym3812_dual_data[CurChip] = (void *) info;
|
|
}
|
|
else
|
|
{
|
|
CAA->StreamUpdate = ym3812_stream_update;
|
|
CAA->StreamUpdateParam = p->ym3812[CurChip];
|
|
p->ym3812_dual_data[CurChip] = NULL;
|
|
}
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
if (! CurChip || ! (ChipClk & 0x80000000))
|
|
AbsVol += CAA->Volume * 2;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzYM3526)
|
|
{
|
|
//ChipVol = UseFM ? 0x00 : 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzYM3526 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].YM3526;
|
|
CAA->ChipType = 0x0A;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_ym3526(&p->ym3526[CurChip], ChipClk,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &ym3526_stream_update;
|
|
CAA->StreamUpdateParam = p->ym3526[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume * 2;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzY8950)
|
|
{
|
|
//ChipVol = UseFM ? 0x00 : 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzY8950 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].Y8950;
|
|
CAA->ChipType = 0x0B;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_y8950(&p->y8950[CurChip], ChipClk,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &y8950_stream_update;
|
|
CAA->StreamUpdateParam = p->y8950[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume * 2;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzYMF262)
|
|
{
|
|
//ChipVol = UseFM ? 0x00 : 0x100;
|
|
p->ChipOpts[0x01].YMF262.EmuCore = p->ChipOpts[0x00].YMF262.EmuCore;
|
|
|
|
ChipCnt = (p->VGMHead.lngHzYMF262 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].YMF262;
|
|
CAA->ChipType = 0x0C;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_ymf262(&p->ymf262[CurChip],
|
|
p->ChipOpts[CurChip].YMF262.EmuCore,
|
|
ChipClk,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &ymf262_stream_update;
|
|
CAA->StreamUpdateParam = p->ymf262[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume * 2;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzYMF278B)
|
|
{
|
|
//ChipVol = 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzYMF278B & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].YMF278B;
|
|
CAA->ChipType = 0x0D;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_ymf278b(&p->ymf278b[CurChip], ChipClk);
|
|
CAA->StreamUpdate = &ymf278b_pcm_update;
|
|
CAA->StreamUpdateParam = p->ymf278b[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume; //good as long as it only uses WaveTable Synth
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzYMF271)
|
|
{
|
|
//ChipVol = 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzYMF271 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].YMF271;
|
|
CAA->ChipType = 0x0E;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_ymf271(&p->ymf271[CurChip], ChipClk);
|
|
CAA->StreamUpdate = &ymf271_update;
|
|
CAA->StreamUpdateParam = p->ymf271[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzYMZ280B)
|
|
{
|
|
//ChipVol = 0x98;
|
|
ChipCnt = (p->VGMHead.lngHzYMZ280B & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].YMZ280B;
|
|
CAA->ChipType = 0x0F;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_ymz280b(&p->ymz280b[CurChip], ChipClk);
|
|
CAA->StreamUpdate = &ymz280b_update;
|
|
CAA->StreamUpdateParam = p->ymz280b[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += (CAA->Volume * 0x20 / 0x13);
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzRF5C164)
|
|
{
|
|
//ChipVol = 0x80;
|
|
ChipCnt = 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].RF5C164;
|
|
CAA->ChipType = 0x10;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_rf5c164(&p->rf5c164, ChipClk,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &rf5c164_update;
|
|
CAA->StreamUpdateParam = p->rf5c164;
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume * 2;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzPWM)
|
|
{
|
|
//ChipVol = 0xE0; // 0xCD
|
|
ChipCnt = 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].PWM;
|
|
CAA->ChipType = 0x11;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_pwm(&p->pwm, ChipClk, p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &pwm_update;
|
|
CAA->StreamUpdateParam = p->pwm;
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzAY8910)
|
|
{
|
|
//ChipVol = 0x100;
|
|
p->ChipOpts[0x01].AY8910.EmuCore = p->ChipOpts[0x00].AY8910.EmuCore;
|
|
|
|
ChipCnt = (p->VGMHead.lngHzAY8910 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].AY8910;
|
|
CAA->ChipType = 0x12;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_ayxx(&p->ay8910[CurChip],
|
|
p->ChipOpts[CurChip].AY8910.EmuCore,
|
|
ChipClk, p->VGMHead.bytAYType, p->VGMHead.bytAYFlag,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &ayxx_stream_update;
|
|
CAA->StreamUpdateParam = p->ay8910[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume * 2;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzGBDMG)
|
|
{
|
|
//ChipVol = 0xC0;
|
|
p->ChipOpts[0x01].GameBoy.SpecialFlags = p->ChipOpts[0x00].GameBoy.SpecialFlags;
|
|
|
|
ChipCnt = (p->VGMHead.lngHzGBDMG & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].GameBoy;
|
|
CAA->ChipType = 0x13;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_gameboy_sound(&p->gbdmg[CurChip], ChipClk,
|
|
p->ChipOpts[CurChip].GameBoy.SpecialFlags,
|
|
p->SampleRate);
|
|
CAA->StreamUpdate = &gameboy_update;
|
|
CAA->StreamUpdateParam = p->gbdmg[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume * 2;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzNESAPU)
|
|
{
|
|
//ChipVol = 0x100;
|
|
p->ChipOpts[0x01].NES.EmuCore = p->ChipOpts[0x00].NES.EmuCore;
|
|
p->ChipOpts[0x01].NES.SpecialFlags = p->ChipOpts[0x00].NES.SpecialFlags;
|
|
|
|
ChipCnt = (p->VGMHead.lngHzNESAPU & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].NES;
|
|
COpt = &p->ChipOpts[CurChip].NES;
|
|
CAA->ChipType = 0x14;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_nes(&p->nesapu[CurChip], COpt->EmuCore,
|
|
ChipClk, COpt->SpecialFlags,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &nes_stream_update;
|
|
CAA->StreamUpdateParam = p->nesapu[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume * 2;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzMultiPCM)
|
|
{
|
|
//ChipVol = 0x40;
|
|
ChipCnt = (p->VGMHead.lngHzMultiPCM & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].MultiPCM;
|
|
CAA->ChipType = 0x15;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_multipcm(&p->multipcm[CurChip], ChipClk);
|
|
CAA->StreamUpdate = &MultiPCM_update;
|
|
CAA->StreamUpdateParam = p->multipcm[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume * 4;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzUPD7759)
|
|
{
|
|
//ChipVol = 0x11E;
|
|
ChipCnt = (p->VGMHead.lngHzUPD7759 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].UPD7759;
|
|
CAA->ChipType = 0x16;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_upd7759(&p->upd7759[CurChip], ChipClk);
|
|
CAA->StreamUpdate = &upd7759_update;
|
|
CAA->StreamUpdateParam = p->upd7759[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzOKIM6258)
|
|
{
|
|
//ChipVol = 0x1C0;
|
|
p->ChipOpts[0x01].OKIM6258.SpecialFlags = p->ChipOpts[0x00].OKIM6258.SpecialFlags;
|
|
|
|
ChipCnt = (p->VGMHead.lngHzOKIM6258 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].OKIM6258;
|
|
CAA->ChipType = 0x17;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_okim6258(&p->okim6258[CurChip], ChipClk,
|
|
p->ChipOpts[CurChip].OKIM6258.SpecialFlags,
|
|
(p->VGMHead.bytOKI6258Flags & 0x03) >> 0,
|
|
(p->VGMHead.bytOKI6258Flags & 0x04) >> 2,
|
|
(p->VGMHead.bytOKI6258Flags & 0x08) >> 3);
|
|
CAA->StreamUpdate = &okim6258_update;
|
|
CAA->StreamUpdateParam = p->okim6258[CurChip];
|
|
okim6258_set_srchg_cb(p->okim6258[CurChip], &ChangeChipSampleRate, CAA);
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume * 2;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzOKIM6295)
|
|
{
|
|
/*// Use the System Tag to decide between normal and CP System volume level.
|
|
// I know, this is very hackish, but ATM there's no better solution.
|
|
if (VGMTag.strSystemNameE != NULL && ! wcsncmp(VGMTag.strSystemNameE, L"CP", 0x02))
|
|
ChipVol = 110;
|
|
else
|
|
ChipVol = 0x100;*/
|
|
ChipCnt = (p->VGMHead.lngHzOKIM6295 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].OKIM6295;
|
|
CAA->ChipType = 0x18;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_okim6295(&p->okim6295[CurChip], ChipClk);
|
|
CAA->StreamUpdate = &okim6295_update;
|
|
CAA->StreamUpdateParam = p->okim6295[CurChip];
|
|
okim6295_set_srchg_cb(p->okim6295[CurChip], &ChangeChipSampleRate, CAA);
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume * 2;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzK051649)
|
|
{
|
|
//ChipVol = 0xA0;
|
|
ChipCnt = (p->VGMHead.lngHzK051649 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].K051649;
|
|
CAA->ChipType = 0x19;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_k051649(&p->k051649[CurChip], ChipClk);
|
|
CAA->StreamUpdate = &k051649_update;
|
|
CAA->StreamUpdateParam = p->k051649[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzK054539)
|
|
{
|
|
//ChipVol = 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzK054539 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].K054539;
|
|
CAA->ChipType = 0x1A;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_k054539(&p->k054539[CurChip], ChipClk);
|
|
CAA->StreamUpdate = &k054539_update;
|
|
CAA->StreamUpdateParam = p->k054539[CurChip];
|
|
k054539_init_flags(p->k054539[CurChip], p->VGMHead.bytK054539Flags);
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzHuC6280)
|
|
{
|
|
//ChipVol = 0x100;
|
|
p->ChipOpts[0x01].HuC6280.EmuCore = p->ChipOpts[0x00].HuC6280.EmuCore;
|
|
|
|
ChipCnt = (p->VGMHead.lngHzHuC6280 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].HuC6280;
|
|
CAA->ChipType = 0x1B;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_c6280(&p->huc6280[CurChip],
|
|
p->ChipOpts[CurChip].HuC6280.EmuCore,
|
|
ChipClk, p->SampleRate,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &c6280_update;
|
|
CAA->StreamUpdateParam = p->huc6280[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzC140)
|
|
{
|
|
//ChipVol = 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzC140 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].C140;
|
|
CAA->ChipType = 0x1C;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_c140(&p->c140[CurChip], ChipClk, p->VGMHead.bytC140Type,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &c140_update;
|
|
CAA->StreamUpdateParam = p->c140[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzK053260)
|
|
{
|
|
//ChipVol = 0xB3;
|
|
ChipCnt = (p->VGMHead.lngHzK053260 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].K053260;
|
|
CAA->ChipType = 0x1D;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_k053260(&p->k053260[CurChip], ChipClk);
|
|
CAA->StreamUpdate = &k053260_update;
|
|
CAA->StreamUpdateParam = p->k053260[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzPokey)
|
|
{
|
|
//ChipVol = 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzPokey & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].Pokey;
|
|
CAA->ChipType = 0x1E;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_pokey(&p->pokey[CurChip], ChipClk);
|
|
CAA->StreamUpdate = &pokey_update;
|
|
CAA->StreamUpdateParam = p->pokey[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzQSound)
|
|
{
|
|
//ChipVol = 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzQSound & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].QSound;
|
|
CAA->ChipType = 0x1F;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_qsound(&p->qsound[CurChip], ChipClk);
|
|
CAA->StreamUpdate = &qsound_update;
|
|
CAA->StreamUpdateParam = p->qsound[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzSCSP)
|
|
{
|
|
p->ChipOpts[0x01].SCSP.SpecialFlags = p->ChipOpts[0x00].SCSP.SpecialFlags;
|
|
|
|
//ChipVol = 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzSCSP & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].SCSP;
|
|
CAA->ChipType = 0x20;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_scsp(&p->scsp[CurChip], ChipClk, p->ChipOpts[CurChip].SCSP.SpecialFlags);
|
|
CAA->StreamUpdate = &SCSP_Update;
|
|
CAA->StreamUpdateParam = p->scsp[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzWSwan)
|
|
{
|
|
//ChipVol = 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzWSwan & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].WSwan;
|
|
CAA->ChipType = 0x21;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = ws_audio_init(&p->wswan[CurChip], ChipClk, p->SampleRate,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &ws_audio_update;
|
|
CAA->StreamUpdateParam = p->wswan[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzVSU)
|
|
{
|
|
//ChipVol = 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzVSU & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].VSU;
|
|
CAA->ChipType = 0x22;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_vsu(&p->vsu[CurChip], ChipClk, p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &vsu_stream_update;
|
|
CAA->StreamUpdateParam = p->vsu[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzSAA1099)
|
|
{
|
|
//ChipVol = 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzSAA1099 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].SAA1099;
|
|
CAA->ChipType = 0x23;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_saa1099(&p->saa1099[CurChip], ChipClk);
|
|
CAA->StreamUpdate = &saa1099_update;
|
|
CAA->StreamUpdateParam = p->saa1099[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzES5503)
|
|
{
|
|
//ChipVol = 0x40;
|
|
ChipCnt = (p->VGMHead.lngHzES5503 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].ES5503;
|
|
CAA->ChipType = 0x24;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_es5503(&p->es5503[CurChip], ChipClk, p->VGMHead.bytES5503Chns);
|
|
CAA->StreamUpdate = &es5503_pcm_update;
|
|
CAA->StreamUpdateParam = p->es5503[CurChip];
|
|
es5503_set_srchg_cb(p->es5503[CurChip], &ChangeChipSampleRate, CAA);
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume * 8;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzES5506)
|
|
{
|
|
//ChipVol = 0x20;
|
|
ChipCnt = (p->VGMHead.lngHzES5506 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].ES5506;
|
|
CAA->ChipType = 0x25;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_es5506(&p->es550x[CurChip], ChipClk, p->VGMHead.bytES5506Chns);
|
|
CAA->StreamUpdate = &es5506_update;
|
|
CAA->StreamUpdateParam = p->es550x[CurChip];
|
|
es5506_set_srchg_cb(p->es550x[CurChip], &ChangeChipSampleRate, CAA);
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume * 16;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzX1_010)
|
|
{
|
|
//ChipVol = 0x100;
|
|
ChipCnt = (p->VGMHead.lngHzX1_010 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].X1_010;
|
|
CAA->ChipType = 0x26;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_x1_010(&p->x1_010[CurChip], ChipClk,
|
|
p->CHIP_SAMPLING_MODE, p->CHIP_SAMPLE_RATE);
|
|
CAA->StreamUpdate = &seta_update;
|
|
CAA->StreamUpdateParam = p->x1_010[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzC352)
|
|
{
|
|
//ChipVol = 0x40;
|
|
ChipCnt = (p->VGMHead.lngHzC352 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].C352;
|
|
CAA->ChipType = 0x27;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_c352(&p->c352[CurChip], ChipClk, p->VGMHead.bytC352ClkDiv * 4);
|
|
CAA->StreamUpdate = &c352_update;
|
|
CAA->StreamUpdateParam = p->c352[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume * 8;
|
|
}
|
|
}
|
|
if (p->VGMHead.lngHzGA20)
|
|
{
|
|
//ChipVol = 0x280;
|
|
ChipCnt = (p->VGMHead.lngHzGA20 & 0x40000000) ? 0x02 : 0x01;
|
|
for (CurChip = 0x00; CurChip < ChipCnt; CurChip ++)
|
|
{
|
|
CAA = &p->ChipAudio[CurChip].GA20;
|
|
CAA->ChipType = 0x28;
|
|
|
|
ChipClk = GetChipClock(p, (CurChip << 7) | CAA->ChipType, NULL);
|
|
CAA->SmpRate = device_start_iremga20(&p->ga20[CurChip], ChipClk);
|
|
CAA->StreamUpdate = &IremGA20_update;
|
|
CAA->StreamUpdateParam = p->ga20[CurChip];
|
|
|
|
CAA->Volume = GetChipVolume(p, CAA->ChipType, CurChip, ChipCnt);
|
|
AbsVol += CAA->Volume;
|
|
}
|
|
}
|
|
|
|
// Initialize DAC Control and PCM Bank
|
|
p->DacCtrlUsed = 0x00;
|
|
//memset(DacCtrlUsg, 0x00, 0x01 * 0xFF);
|
|
for (CurChip = 0x00; CurChip < 0xFF; CurChip ++)
|
|
{
|
|
p->DacCtrl[CurChip].Enable = false;
|
|
}
|
|
//memset(p->DacCtrl, 0x00, sizeof(DACCTRL_DATA) * 0xFF);
|
|
|
|
memset(p->PCMBank, 0x00, sizeof(VGM_PCM_BANK) * PCM_BANK_COUNT);
|
|
memset(&p->PCMTbl, 0x00, sizeof(PCMBANK_TBL));
|
|
|
|
// Reset chips
|
|
Chips_GeneralActions(p, 0x01);
|
|
|
|
while(AbsVol < 0x200 && AbsVol)
|
|
{
|
|
for (CurCSet = 0x00; CurCSet < 0x02; CurCSet ++)
|
|
{
|
|
CAA = (CAUD_ATTR*)&p->ChipAudio[CurCSet];
|
|
for (CurChip = 0x00; CurChip < CHIP_COUNT; CurChip ++, CAA ++)
|
|
CAA->Volume *= 2;
|
|
CAA = p->CA_Paired[CurCSet];
|
|
for (CurChip = 0x00; CurChip < 0x03; CurChip ++, CAA ++)
|
|
CAA->Volume *= 2;
|
|
}
|
|
AbsVol *= 2;
|
|
}
|
|
while(AbsVol > 0x300)
|
|
{
|
|
for (CurCSet = 0x00; CurCSet < 0x02; CurCSet ++)
|
|
{
|
|
CAA = (CAUD_ATTR*)&p->ChipAudio[CurCSet];
|
|
for (CurChip = 0x00; CurChip < CHIP_COUNT; CurChip ++, CAA ++)
|
|
CAA->Volume /= 2;
|
|
CAA = p->CA_Paired[CurCSet];
|
|
for (CurChip = 0x00; CurChip < 0x03; CurChip ++, CAA ++)
|
|
CAA->Volume /= 2;
|
|
}
|
|
AbsVol /= 2;
|
|
}
|
|
|
|
// Initialize Resampler
|
|
for (CurCSet = 0x00; CurCSet < 0x02; CurCSet ++)
|
|
{
|
|
CAA = (CAUD_ATTR*)&p->ChipAudio[CurCSet];
|
|
for (CurChip = 0x00; CurChip < CHIP_COUNT; CurChip ++, CAA ++)
|
|
SetupResampler(p, CAA);
|
|
|
|
CAA = p->CA_Paired[CurCSet];
|
|
for (CurChip = 0x00; CurChip < 0x03; CurChip ++, CAA ++)
|
|
SetupResampler(p, CAA);
|
|
}
|
|
|
|
GeneralChipLists(p);
|
|
break;
|
|
case 0x01: // Reset chips
|
|
for (CurCSet = 0x00; CurCSet < 0x02; CurCSet ++)
|
|
{
|
|
|
|
CAA = (CAUD_ATTR*)&p->ChipAudio[CurCSet];
|
|
for (CurChip = 0x00; CurChip < CHIP_COUNT; CurChip ++, CAA ++)
|
|
{
|
|
if (CAA->ChipType == 0xFF) // chip unused
|
|
continue;
|
|
else if (CAA->ChipType == 0x00)
|
|
device_reset_sn764xx(p->sn764xx[CurCSet]);
|
|
else if (CAA->ChipType == 0x01)
|
|
device_reset_ym2413(p->ym2413[CurCSet]);
|
|
else if (CAA->ChipType == 0x02)
|
|
device_reset_ym2612(p->ym2612[CurCSet]);
|
|
else if (CAA->ChipType == 0x03)
|
|
device_reset_ym2151(p->ym2151[CurCSet]);
|
|
else if (CAA->ChipType == 0x04)
|
|
device_reset_segapcm(p->segapcm[CurCSet]);
|
|
else if (CAA->ChipType == 0x05 && CurCSet == 0x00)
|
|
device_reset_rf5c68(p->rf5c68);
|
|
else if (CAA->ChipType == 0x06)
|
|
device_reset_ym2203(p->ym2203[CurCSet]);
|
|
else if (CAA->ChipType == 0x07)
|
|
device_reset_ym2608(p->ym2608[CurCSet]);
|
|
else if (CAA->ChipType == 0x08)
|
|
device_reset_ym2610(p->ym2610[CurCSet]);
|
|
else if (CAA->ChipType == 0x09)
|
|
device_reset_ym3812(p->ym3812[CurCSet]);
|
|
else if (CAA->ChipType == 0x0A)
|
|
device_reset_ym3526(p->ym3526[CurCSet]);
|
|
else if (CAA->ChipType == 0x0B)
|
|
device_reset_y8950(p->y8950[CurCSet]);
|
|
else if (CAA->ChipType == 0x0C)
|
|
device_reset_ymf262(p->ymf262[CurCSet]);
|
|
else if (CAA->ChipType == 0x0D)
|
|
device_reset_ymf278b(p->ymf278b[CurCSet]);
|
|
else if (CAA->ChipType == 0x0E)
|
|
device_reset_ymf271(p->ymf271[CurCSet]);
|
|
else if (CAA->ChipType == 0x0F)
|
|
device_reset_ymz280b(p->ymz280b[CurCSet]);
|
|
else if (CAA->ChipType == 0x10 && CurCSet == 0x00)
|
|
device_reset_rf5c164(p->rf5c164);
|
|
else if (CAA->ChipType == 0x11 && CurCSet == 0x00)
|
|
device_reset_pwm(p->pwm);
|
|
else if (CAA->ChipType == 0x12)
|
|
device_reset_ayxx(p->ay8910[CurCSet]);
|
|
else if (CAA->ChipType == 0x13)
|
|
device_reset_gameboy_sound(p->gbdmg[CurCSet]);
|
|
else if (CAA->ChipType == 0x14)
|
|
device_reset_nes(p->nesapu[CurCSet]);
|
|
else if (CAA->ChipType == 0x15)
|
|
device_reset_multipcm(p->multipcm[CurCSet]);
|
|
else if (CAA->ChipType == 0x16)
|
|
device_reset_upd7759(p->upd7759[CurCSet]);
|
|
else if (CAA->ChipType == 0x17)
|
|
device_reset_okim6258(p->okim6258[CurCSet]);
|
|
else if (CAA->ChipType == 0x18)
|
|
device_reset_okim6295(p->okim6295[CurCSet]);
|
|
else if (CAA->ChipType == 0x19)
|
|
device_reset_k051649(p->k051649[CurCSet]);
|
|
else if (CAA->ChipType == 0x1A)
|
|
device_reset_k054539(p->k054539[CurCSet]);
|
|
else if (CAA->ChipType == 0x1B)
|
|
device_reset_c6280(p->huc6280[CurCSet]);
|
|
else if (CAA->ChipType == 0x1C)
|
|
device_reset_c140(p->c140[CurCSet]);
|
|
else if (CAA->ChipType == 0x1D)
|
|
device_reset_k053260(p->k053260[CurCSet]);
|
|
else if (CAA->ChipType == 0x1E)
|
|
device_reset_pokey(p->pokey[CurCSet]);
|
|
else if (CAA->ChipType == 0x1F)
|
|
device_reset_qsound(p->qsound[CurCSet]);
|
|
else if (CAA->ChipType == 0x20)
|
|
device_reset_scsp(p->scsp[CurCSet]);
|
|
else if (CAA->ChipType == 0x21)
|
|
ws_audio_reset(p->wswan[CurCSet]);
|
|
else if (CAA->ChipType == 0x22)
|
|
device_reset_vsu(p->vsu[CurCSet]);
|
|
else if (CAA->ChipType == 0x23)
|
|
device_reset_saa1099(p->saa1099[CurCSet]);
|
|
else if (CAA->ChipType == 0x24)
|
|
device_reset_es5503(p->es5503[CurCSet]);
|
|
else if (CAA->ChipType == 0x25)
|
|
device_reset_es5506(p->es550x[CurCSet]);
|
|
else if (CAA->ChipType == 0x26)
|
|
device_reset_x1_010(p->x1_010[CurCSet]);
|
|
else if (CAA->ChipType == 0x27)
|
|
device_reset_c352(p->c352[CurCSet]);
|
|
else if (CAA->ChipType == 0x28)
|
|
device_reset_iremga20(p->ga20[CurCSet]);
|
|
} // end for CurChip
|
|
|
|
} // end for CurCSet
|
|
|
|
Chips_GeneralActions(p, 0x10); // set muting mask
|
|
Chips_GeneralActions(p, 0x20); // set panning
|
|
|
|
for (CurChip = 0x00; CurChip < p->DacCtrlUsed; CurChip ++)
|
|
{
|
|
CurCSet = p->DacCtrlUsg[CurChip];
|
|
device_reset_daccontrol(p->daccontrol[CurCSet]);
|
|
//DacCtrl[CurCSet].Enable = false;
|
|
}
|
|
//DacCtrlUsed = 0x00;
|
|
//memset(DacCtrlUsg, 0x00, 0x01 * 0xFF);
|
|
|
|
for (CurChip = 0x00; CurChip < PCM_BANK_COUNT; CurChip ++)
|
|
{
|
|
// reset PCM Bank, but not the data
|
|
// (this way I don't need to decompress the data again when restarting)
|
|
p->PCMBank[CurChip].DataPos = 0x00000000;
|
|
p->PCMBank[CurChip].BnkPos = 0x00000000;
|
|
}
|
|
p->PCMTbl.EntryCount = 0x00;
|
|
break;
|
|
case 0x02: // Stop chips
|
|
for (CurCSet = 0x00; CurCSet < 0x02; CurCSet ++)
|
|
{
|
|
|
|
CAA = (CAUD_ATTR*)&p->ChipAudio[CurCSet];
|
|
for (CurChip = 0x00; CurChip < CHIP_COUNT; CurChip ++, CAA ++)
|
|
{
|
|
if (CAA->ChipType == 0xFF) // chip unused
|
|
continue;
|
|
else if (CAA->ChipType == 0x00)
|
|
device_stop_sn764xx(p->sn764xx[CurCSet]);
|
|
else if (CAA->ChipType == 0x01)
|
|
device_stop_ym2413(p->ym2413[CurCSet]);
|
|
else if (CAA->ChipType == 0x02)
|
|
device_stop_ym2612(p->ym2612[CurCSet]);
|
|
else if (CAA->ChipType == 0x03)
|
|
device_stop_ym2151(p->ym2151[CurCSet]);
|
|
else if (CAA->ChipType == 0x04)
|
|
device_stop_segapcm(p->segapcm[CurCSet]);
|
|
else if (CAA->ChipType == 0x05 && CurCSet == 0x00)
|
|
device_stop_rf5c68(p->rf5c68);
|
|
else if (CAA->ChipType == 0x06)
|
|
device_stop_ym2203(p->ym2203[CurCSet]);
|
|
else if (CAA->ChipType == 0x07)
|
|
device_stop_ym2608(p->ym2608[CurCSet]);
|
|
else if (CAA->ChipType == 0x08)
|
|
device_stop_ym2610(p->ym2610[CurCSet]);
|
|
else if (CAA->ChipType == 0x09)
|
|
{
|
|
device_stop_ym3812(p->ym3812[CurCSet]);
|
|
free(p->ym3812_dual_data[CurCSet]);
|
|
p->ym3812_dual_data[CurCSet] = NULL;
|
|
}
|
|
else if (CAA->ChipType == 0x0A)
|
|
device_stop_ym3526(p->ym3526[CurCSet]);
|
|
else if (CAA->ChipType == 0x0B)
|
|
device_stop_y8950(p->y8950[CurCSet]);
|
|
else if (CAA->ChipType == 0x0C)
|
|
device_stop_ymf262(p->ymf262[CurCSet]);
|
|
else if (CAA->ChipType == 0x0D)
|
|
device_stop_ymf278b(p->ymf278b[CurCSet]);
|
|
else if (CAA->ChipType == 0x0E)
|
|
device_stop_ymf271(p->ymf271[CurCSet]);
|
|
else if (CAA->ChipType == 0x0F)
|
|
device_stop_ymz280b(p->ymz280b[CurCSet]);
|
|
else if (CAA->ChipType == 0x10 && CurCSet == 0x00)
|
|
device_stop_rf5c164(p->rf5c164);
|
|
else if (CAA->ChipType == 0x11 && CurCSet == 0x00)
|
|
device_stop_pwm(p->pwm);
|
|
else if (CAA->ChipType == 0x12)
|
|
device_stop_ayxx(p->ay8910[CurCSet]);
|
|
else if (CAA->ChipType == 0x13)
|
|
device_stop_gameboy_sound(p->gbdmg[CurCSet]);
|
|
else if (CAA->ChipType == 0x14)
|
|
device_stop_nes(p->nesapu[CurCSet]);
|
|
else if (CAA->ChipType == 0x15)
|
|
device_stop_multipcm(p->multipcm[CurCSet]);
|
|
else if (CAA->ChipType == 0x16)
|
|
device_stop_upd7759(p->upd7759[CurCSet]);
|
|
else if (CAA->ChipType == 0x17)
|
|
device_stop_okim6258(p->okim6258[CurCSet]);
|
|
else if (CAA->ChipType == 0x18)
|
|
device_stop_okim6295(p->okim6295[CurCSet]);
|
|
else if (CAA->ChipType == 0x19)
|
|
device_stop_k051649(p->k051649[CurCSet]);
|
|
else if (CAA->ChipType == 0x1A)
|
|
device_stop_k054539(p->k054539[CurCSet]);
|
|
else if (CAA->ChipType == 0x1B)
|
|
device_stop_c6280(p->huc6280[CurCSet]);
|
|
else if (CAA->ChipType == 0x1C)
|
|
device_stop_c140(p->c140[CurCSet]);
|
|
else if (CAA->ChipType == 0x1D)
|
|
device_stop_k053260(p->k053260[CurCSet]);
|
|
else if (CAA->ChipType == 0x1E)
|
|
device_stop_pokey(p->pokey[CurCSet]);
|
|
else if (CAA->ChipType == 0x1F)
|
|
device_stop_qsound(p->qsound[CurCSet]);
|
|
else if (CAA->ChipType == 0x20)
|
|
device_stop_scsp(p->scsp[CurCSet]);
|
|
else if (CAA->ChipType == 0x21)
|
|
ws_audio_done(p->wswan[CurCSet]);
|
|
else if (CAA->ChipType == 0x22)
|
|
device_stop_vsu(p->vsu[CurCSet]);
|
|
else if (CAA->ChipType == 0x23)
|
|
device_stop_saa1099(p->saa1099[CurCSet]);
|
|
else if (CAA->ChipType == 0x24)
|
|
device_stop_es5503(p->es5503[CurCSet]);
|
|
else if (CAA->ChipType == 0x25)
|
|
device_stop_es5506(p->es550x[CurCSet]);
|
|
else if (CAA->ChipType == 0x26)
|
|
device_stop_x1_010(p->x1_010[CurCSet]);
|
|
else if (CAA->ChipType == 0x27)
|
|
device_stop_c352(p->c352[CurCSet]);
|
|
else if (CAA->ChipType == 0x28)
|
|
device_stop_iremga20(p->ga20[CurCSet]);
|
|
|
|
resampler_destroy(CAA->Resampler);
|
|
CAA->Resampler = 0x00;
|
|
|
|
CAA->ChipType = 0xFF; // mark as "unused"
|
|
} // end for CurChip
|
|
|
|
} // end for CurCSet
|
|
|
|
for (CurChip = 0x00; CurChip < p->DacCtrlUsed; CurChip ++)
|
|
{
|
|
CurCSet = p->DacCtrlUsg[CurChip];
|
|
device_stop_daccontrol(p->daccontrol[CurCSet]);
|
|
p->DacCtrl[CurCSet].Enable = false;
|
|
}
|
|
p->DacCtrlUsed = 0x00;
|
|
|
|
for (CurChip = 0x00; CurChip < PCM_BANK_COUNT; CurChip ++)
|
|
{
|
|
free(p->PCMBank[CurChip].Bank);
|
|
free(p->PCMBank[CurChip].Data);
|
|
}
|
|
//memset(PCMBank, 0x00, sizeof(VGM_PCM_BANK) * PCM_BANK_COUNT);
|
|
free(p->PCMTbl.Entries);
|
|
//memset(&PCMTbl, 0x00, sizeof(PCMBANK_TBL));
|
|
break;
|
|
case 0x10: // Set Muting Mask
|
|
for (CurCSet = 0x00; CurCSet < 0x02; CurCSet ++)
|
|
{
|
|
|
|
CAA = (CAUD_ATTR*)&p->ChipAudio[CurCSet];
|
|
for (CurChip = 0x00; CurChip < CHIP_COUNT; CurChip ++, CAA ++)
|
|
{
|
|
if (CAA->ChipType == 0xFF) // chip unused
|
|
continue;
|
|
else if (CAA->ChipType == 0x00)
|
|
sn764xx_set_mute_mask(p->sn764xx[CurCSet], p->ChipOpts[CurCSet].SN76496.ChnMute1);
|
|
else if (CAA->ChipType == 0x01)
|
|
ym2413_set_mute_mask(p->ym2413[CurCSet], p->ChipOpts[CurCSet].YM2413.ChnMute1);
|
|
else if (CAA->ChipType == 0x02)
|
|
ym2612_set_mute_mask(p->ym2612[CurCSet], p->ChipOpts[CurCSet].YM2612.ChnMute1);
|
|
else if (CAA->ChipType == 0x03)
|
|
ym2151_set_mute_mask(p->ym2151[CurCSet], p->ChipOpts[CurCSet].YM2151.ChnMute1);
|
|
else if (CAA->ChipType == 0x04)
|
|
segapcm_set_mute_mask(p->segapcm[CurCSet], p->ChipOpts[CurCSet].SegaPCM.ChnMute1);
|
|
else if (CAA->ChipType == 0x05 && CurCSet == 0x00)
|
|
rf5c68_set_mute_mask(p->rf5c68, p->ChipOpts[CurCSet].RF5C68.ChnMute1);
|
|
else if (CAA->ChipType == 0x06)
|
|
ym2203_set_mute_mask(p->ym2203[CurCSet], p->ChipOpts[CurCSet].YM2203.ChnMute1,
|
|
p->ChipOpts[CurCSet].YM2203.ChnMute3);
|
|
else if (CAA->ChipType == 0x07)
|
|
{
|
|
MaskVal = (p->ChipOpts[CurCSet].YM2608.ChnMute1 & 0x3F) << 0;
|
|
MaskVal |= (p->ChipOpts[CurCSet].YM2608.ChnMute2 & 0x7F) << 6;
|
|
ym2608_set_mute_mask(p->ym2608[CurCSet], MaskVal, p->ChipOpts[CurCSet].YM2608.ChnMute3);
|
|
}
|
|
else if (CAA->ChipType == 0x08)
|
|
{
|
|
MaskVal = (p->ChipOpts[CurCSet].YM2610.ChnMute1 & 0x3F) << 0;
|
|
MaskVal |= (p->ChipOpts[CurCSet].YM2610.ChnMute2 & 0x7F) << 6;
|
|
ym2610_set_mute_mask(p->ym2610[CurCSet], MaskVal, p->ChipOpts[CurCSet].YM2610.ChnMute3);
|
|
}
|
|
else if (CAA->ChipType == 0x09)
|
|
ym3812_set_mute_mask(p->ym3812[CurCSet], p->ChipOpts[CurCSet].YM3812.ChnMute1);
|
|
else if (CAA->ChipType == 0x0A)
|
|
ym3526_set_mute_mask(p->ym3526[CurCSet], p->ChipOpts[CurCSet].YM3526.ChnMute1);
|
|
else if (CAA->ChipType == 0x0B)
|
|
y8950_set_mute_mask(p->y8950[CurCSet], p->ChipOpts[CurCSet].Y8950.ChnMute1);
|
|
else if (CAA->ChipType == 0x0C)
|
|
ymf262_set_mute_mask(p->ymf262[CurCSet], p->ChipOpts[CurCSet].YMF262.ChnMute1);
|
|
else if (CAA->ChipType == 0x0D)
|
|
ymf278b_set_mute_mask(p->ymf278b[CurCSet], p->ChipOpts[CurCSet].YMF278B.ChnMute1,
|
|
p->ChipOpts[CurCSet].YMF278B.ChnMute2);
|
|
else if (CAA->ChipType == 0x0E)
|
|
ymf271_set_mute_mask(p->ymf271[CurCSet], p->ChipOpts[CurCSet].YMF271.ChnMute1);
|
|
else if (CAA->ChipType == 0x0F)
|
|
ymz280b_set_mute_mask(p->ymz280b[CurCSet], p->ChipOpts[CurCSet].YMZ280B.ChnMute1);
|
|
else if (CAA->ChipType == 0x10 && CurCSet == 0x00)
|
|
rf5c164_set_mute_mask(p->rf5c164, p->ChipOpts[CurCSet].RF5C164.ChnMute1);
|
|
else if (CAA->ChipType == 0x11)
|
|
; // PWM - nothing to mute
|
|
else if (CAA->ChipType == 0x12)
|
|
ayxx_set_mute_mask(p->ay8910[CurCSet], p->ChipOpts[CurCSet].AY8910.ChnMute1);
|
|
else if (CAA->ChipType == 0x13)
|
|
gameboy_sound_set_mute_mask(p->gbdmg[CurCSet], p->ChipOpts[CurCSet].GameBoy.ChnMute1);
|
|
else if (CAA->ChipType == 0x14)
|
|
nes_set_mute_mask(p->nesapu[CurCSet], p->ChipOpts[CurCSet].NES.ChnMute1);
|
|
else if (CAA->ChipType == 0x15)
|
|
multipcm_set_mute_mask(p->multipcm[CurCSet], p->ChipOpts[CurCSet].MultiPCM.ChnMute1);
|
|
else if (CAA->ChipType == 0x16)
|
|
; // UPD7759 - nothing to mute
|
|
else if (CAA->ChipType == 0x17)
|
|
; // OKIM6258 - nothing to mute
|
|
else if (CAA->ChipType == 0x18)
|
|
okim6295_set_mute_mask(p->okim6295[CurCSet], p->ChipOpts[CurCSet].OKIM6295.ChnMute1);
|
|
else if (CAA->ChipType == 0x19)
|
|
k051649_set_mute_mask(p->k051649[CurCSet], p->ChipOpts[CurCSet].K051649.ChnMute1);
|
|
else if (CAA->ChipType == 0x1A)
|
|
k054539_set_mute_mask(p->k054539[CurCSet], p->ChipOpts[CurCSet].K054539.ChnMute1);
|
|
else if (CAA->ChipType == 0x1B)
|
|
c6280_set_mute_mask(p->huc6280[CurCSet], p->ChipOpts[CurCSet].HuC6280.ChnMute1);
|
|
else if (CAA->ChipType == 0x1C)
|
|
c140_set_mute_mask(p->c140[CurCSet], p->ChipOpts[CurCSet].C140.ChnMute1);
|
|
else if (CAA->ChipType == 0x1D)
|
|
k053260_set_mute_mask(p->k053260[CurCSet], p->ChipOpts[CurCSet].K053260.ChnMute1);
|
|
else if (CAA->ChipType == 0x1E)
|
|
pokey_set_mute_mask(p->pokey[CurCSet], p->ChipOpts[CurCSet].Pokey.ChnMute1);
|
|
else if (CAA->ChipType == 0x1F)
|
|
qsound_set_mute_mask(p->qsound[CurCSet], p->ChipOpts[CurCSet].QSound.ChnMute1);
|
|
else if (CAA->ChipType == 0x20)
|
|
scsp_set_mute_mask(p->scsp[CurCSet], p->ChipOpts[CurCSet].SCSP.ChnMute1);
|
|
else if (CAA->ChipType == 0x21)
|
|
ws_set_mute_mask(p->wswan[CurCSet], p->ChipOpts[CurCSet].WSwan.ChnMute1);
|
|
else if (CAA->ChipType == 0x22)
|
|
vsu_set_mute_mask(p->vsu[CurCSet], p->ChipOpts[CurCSet].VSU.ChnMute1);
|
|
else if (CAA->ChipType == 0x23)
|
|
saa1099_set_mute_mask(p->saa1099[CurCSet], p->ChipOpts[CurCSet].SAA1099.ChnMute1);
|
|
else if (CAA->ChipType == 0x24)
|
|
es5503_set_mute_mask(p->es5503[CurCSet], p->ChipOpts[CurCSet].ES5503.ChnMute1);
|
|
else if (CAA->ChipType == 0x25)
|
|
es5506_set_mute_mask(p->es550x[CurCSet], p->ChipOpts[CurCSet].ES5506.ChnMute1);
|
|
else if (CAA->ChipType == 0x26)
|
|
x1_010_set_mute_mask(p->x1_010[CurCSet], p->ChipOpts[CurCSet].X1_010.ChnMute1);
|
|
else if (CAA->ChipType == 0x27)
|
|
c352_set_mute_mask(p->c352[CurCSet], p->ChipOpts[CurCSet].C352.ChnMute1);
|
|
else if (CAA->ChipType == 0x28)
|
|
iremga20_set_mute_mask(p->ga20[CurCSet], p->ChipOpts[CurCSet].GA20.ChnMute1);
|
|
} // end for CurChip
|
|
|
|
} // end for CurCSet
|
|
break;
|
|
case 0x20: // Set Panning
|
|
for (CurCSet = 0x00; CurCSet < 0x02; CurCSet ++)
|
|
{
|
|
|
|
CAA = (CAUD_ATTR*)&p->ChipAudio[CurCSet];
|
|
for (CurChip = 0x00; CurChip < CHIP_COUNT; CurChip ++, CAA ++)
|
|
{
|
|
if (CAA->ChipType == 0xFF) // chip unused
|
|
continue;
|
|
else if (CAA->ChipType == 0x00)
|
|
sn764xx_set_panning(p->sn764xx[CurCSet], p->ChipOpts[CurCSet].SN76496.Panning);
|
|
else if (CAA->ChipType == 0x01)
|
|
ym2413_set_panning(p->ym2413[CurCSet], p->ChipOpts[CurCSet].YM2413.Panning);
|
|
} // end for CurChip
|
|
|
|
} // end for CurCSet
|
|
break;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
INLINE INT32 SampleVGM2Pbk_I(VGM_PLAYER* p, INT32 SampleVal)
|
|
{
|
|
return (INT32)((INT64)SampleVal * p->VGMSmplRateMul / p->VGMSmplRateDiv);
|
|
}
|
|
|
|
INLINE INT32 SamplePbk2VGM_I(VGM_PLAYER* p, INT32 SampleVal)
|
|
{
|
|
return (INT32)((INT64)SampleVal * p->VGMSmplRateDiv / p->VGMSmplRateMul);
|
|
}
|
|
|
|
INT32 SampleVGM2Playback(void* _p, INT32 SampleVal)
|
|
{
|
|
VGM_PLAYER* p = (VGM_PLAYER *)_p;
|
|
return (INT32)((INT64)SampleVal * p->VGMSmplRateMul / p->VGMSmplRateDiv);
|
|
}
|
|
|
|
INT32 SamplePlayback2VGM(void* _p, INT32 SampleVal)
|
|
{
|
|
VGM_PLAYER* p = (VGM_PLAYER *)_p;
|
|
return (INT32)((INT64)SampleVal * p->VGMSmplRateDiv / p->VGMSmplRateMul);
|
|
}
|
|
|
|
|
|
static void InterpretFile(VGM_PLAYER* p, UINT32 SampleCount)
|
|
{
|
|
UINT32 TempLng;
|
|
UINT8 CurChip;
|
|
|
|
if (p->DacCtrlUsed && SampleCount > 1) // handle skipping
|
|
{
|
|
for (CurChip = 0x00; CurChip < p->DacCtrlUsed; CurChip ++)
|
|
{
|
|
daccontrol_update(p->daccontrol[p->DacCtrlUsg[CurChip]], SampleCount - 1);
|
|
}
|
|
}
|
|
|
|
if (! p->FileMode)
|
|
InterpretVGM(p, SampleCount);
|
|
#ifdef ADDITIONAL_FORMATS
|
|
else
|
|
InterpretOther(p, SampleCount);
|
|
#endif
|
|
|
|
if (p->DacCtrlUsed && SampleCount)
|
|
{
|
|
// calling this here makes "Emulating while Paused" nicer
|
|
for (CurChip = 0x00; CurChip < p->DacCtrlUsed; CurChip ++)
|
|
{
|
|
daccontrol_update(p->daccontrol[p->DacCtrlUsg[CurChip]], 1);
|
|
}
|
|
}
|
|
|
|
p->VGMSmplPlayed += SampleCount;
|
|
p->PlayingTime += SampleCount;
|
|
|
|
//if (FadePlay && ! FadeTime)
|
|
// EndPlay = true;
|
|
|
|
return;
|
|
}
|
|
|
|
static void AddPCMData(VGM_PLAYER* p, UINT8 Type, UINT32 DataSize, const UINT8* Data)
|
|
{
|
|
UINT32 CurBnk;
|
|
VGM_PCM_BANK* TempPCM;
|
|
VGM_PCM_DATA* TempBnk;
|
|
UINT32 BankSize;
|
|
bool RetVal;
|
|
UINT8 BnkType;
|
|
UINT8 CurDAC;
|
|
|
|
BnkType = Type & 0x3F;
|
|
if (BnkType >= PCM_BANK_COUNT || p->VGMCurLoop)
|
|
return;
|
|
|
|
if (Type == 0x7F)
|
|
{
|
|
ReadPCMTable(p, DataSize, Data);
|
|
return;
|
|
}
|
|
|
|
TempPCM = &p->PCMBank[BnkType];
|
|
TempPCM->BnkPos ++;
|
|
if (TempPCM->BnkPos <= TempPCM->BankCount)
|
|
return; // Speed hack for restarting playback (skip already loaded blocks)
|
|
CurBnk = TempPCM->BankCount;
|
|
TempPCM->BankCount ++;
|
|
if (p->Last95Max != 0xFFFF)
|
|
p->Last95Max = TempPCM->BankCount;
|
|
TempPCM->Bank = (VGM_PCM_DATA*)realloc(TempPCM->Bank,
|
|
sizeof(VGM_PCM_DATA) * TempPCM->BankCount);
|
|
|
|
if (! (Type & 0x40))
|
|
BankSize = DataSize;
|
|
else
|
|
BankSize = ReadLE32(&Data[0x01]);
|
|
TempPCM->Data = realloc(TempPCM->Data, TempPCM->DataSize + BankSize);
|
|
TempBnk = &TempPCM->Bank[CurBnk];
|
|
TempBnk->DataStart = TempPCM->DataSize;
|
|
if (! (Type & 0x40))
|
|
{
|
|
TempBnk->DataSize = DataSize;
|
|
TempBnk->Data = TempPCM->Data + TempBnk->DataStart;
|
|
memcpy(TempBnk->Data, Data, DataSize);
|
|
}
|
|
else
|
|
{
|
|
TempBnk->Data = TempPCM->Data + TempBnk->DataStart;
|
|
RetVal = DecompressDataBlk(p, TempBnk, DataSize, Data);
|
|
if (! RetVal)
|
|
{
|
|
TempBnk->Data = NULL;
|
|
TempBnk->DataSize = 0x00;
|
|
//return;
|
|
goto RefreshDACStrm; // sorry for the goto, but I don't want to copy-paste the code
|
|
}
|
|
}
|
|
if (BankSize != TempBnk->DataSize)
|
|
printf("Error reading Data Block! Data Size conflict!\n");
|
|
TempPCM->DataSize += BankSize;
|
|
|
|
// realloc may've moved the Bank block, so refresh all DAC Streams
|
|
RefreshDACStrm:
|
|
for (CurDAC = 0x00; CurDAC < p->DacCtrlUsed; CurDAC ++)
|
|
{
|
|
if (p->DacCtrl[p->DacCtrlUsg[CurDAC]].Bank == BnkType)
|
|
daccontrol_refresh_data(p->daccontrol[p->DacCtrlUsg[CurDAC]], TempPCM->Data, TempPCM->DataSize);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
/*INLINE FUINT16 ReadBits(UINT8* Data, UINT32* Pos, FUINT8* BitPos, FUINT8 BitsToRead)
|
|
{
|
|
FUINT8 BitReadVal;
|
|
UINT32 InPos;
|
|
FUINT8 InVal;
|
|
FUINT8 BitMask;
|
|
FUINT8 InShift;
|
|
FUINT8 OutBit;
|
|
FUINT16 RetVal;
|
|
|
|
InPos = *Pos;
|
|
InShift = *BitPos;
|
|
OutBit = 0x00;
|
|
RetVal = 0x0000;
|
|
while(BitsToRead)
|
|
{
|
|
BitReadVal = (BitsToRead >= 8) ? 8 : BitsToRead;
|
|
BitsToRead -= BitReadVal;
|
|
BitMask = (1 << BitReadVal) - 1;
|
|
|
|
InShift += BitReadVal;
|
|
InVal = (Data[InPos] << InShift >> 8) & BitMask;
|
|
if (InShift >= 8)
|
|
{
|
|
InShift -= 8;
|
|
InPos ++;
|
|
if (InShift)
|
|
InVal |= (Data[InPos] << InShift >> 8) & BitMask;
|
|
}
|
|
|
|
RetVal |= InVal << OutBit;
|
|
OutBit += BitReadVal;
|
|
}
|
|
|
|
*Pos = InPos;
|
|
*BitPos = InShift;
|
|
return RetVal;
|
|
}
|
|
|
|
static void DecompressDataBlk(VGM_PCM_DATA* Bank, UINT32 DataSize, const UINT8* Data)
|
|
{
|
|
UINT8 ComprType;
|
|
UINT8 BitDec;
|
|
FUINT8 BitCmp;
|
|
UINT8 CmpSubType;
|
|
UINT16 AddVal;
|
|
UINT32 InPos;
|
|
UINT32 OutPos;
|
|
FUINT16 InVal;
|
|
FUINT16 OutVal;
|
|
FUINT8 ValSize;
|
|
FUINT8 InShift;
|
|
FUINT8 OutShift;
|
|
UINT8* Ent1B;
|
|
UINT16* Ent2B;
|
|
//UINT32 Time;
|
|
|
|
//Time = GetTickCount();
|
|
ComprType = Data[0x00];
|
|
Bank->DataSize = ReadLE32(&Data[0x01]);
|
|
BitDec = Data[0x05];
|
|
BitCmp = Data[0x06];
|
|
CmpSubType = Data[0x07];
|
|
AddVal = ReadLE16(&Data[0x08]);
|
|
|
|
switch(ComprType)
|
|
{
|
|
case 0x00: // n-Bit compression
|
|
if (CmpSubType == 0x02)
|
|
{
|
|
Ent1B = (UINT8*)PCMTbl.Entries;
|
|
Ent2B = (UINT16*)PCMTbl.Entries;
|
|
if (! PCMTbl.EntryCount)
|
|
{
|
|
printf("Error loading table-compressed data block! No table loaded!\n");
|
|
return;
|
|
}
|
|
else if (BitDec != PCMTbl.BitDec || BitCmp != PCMTbl.BitCmp)
|
|
{
|
|
printf("Warning! Data block and loaded value table incompatible!\n");
|
|
return;
|
|
}
|
|
}
|
|
|
|
ValSize = (BitDec + 7) / 8;
|
|
InPos = 0x0A;
|
|
InShift = 0;
|
|
OutShift = BitDec - BitCmp;
|
|
|
|
for (OutPos = 0x00; OutPos < Bank->DataSize; OutPos += ValSize)
|
|
{
|
|
if (InPos >= DataSize)
|
|
break;
|
|
InVal = ReadBits(Data, &InPos, &InShift, BitCmp);
|
|
switch(CmpSubType)
|
|
{
|
|
case 0x00: // Copy
|
|
OutVal = InVal + AddVal;
|
|
break;
|
|
case 0x01: // Shift Left
|
|
OutVal = (InVal << OutShift) + AddVal;
|
|
break;
|
|
case 0x02: // Table
|
|
switch(ValSize)
|
|
{
|
|
case 0x01:
|
|
OutVal = Ent1B[InVal];
|
|
break;
|
|
case 0x02:
|
|
OutVal = Ent2B[InVal];
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
memcpy(&Bank->Data[OutPos], &OutVal, ValSize);
|
|
}
|
|
break;
|
|
}
|
|
|
|
//Time = GetTickCount() - Time;
|
|
//printf("Decompression Time: %lu\n", Time);
|
|
|
|
return;
|
|
}*/
|
|
|
|
static bool DecompressDataBlk(VGM_PLAYER* p, VGM_PCM_DATA* Bank, UINT32 DataSize, const UINT8* Data)
|
|
{
|
|
UINT8 ComprType;
|
|
UINT8 BitDec;
|
|
FUINT8 BitCmp;
|
|
UINT8 CmpSubType;
|
|
UINT16 AddVal;
|
|
const UINT8* InPos;
|
|
const UINT8* InDataEnd;
|
|
UINT8* OutPos;
|
|
const UINT8* OutDataEnd;
|
|
FUINT16 InVal;
|
|
FUINT16 OutVal;
|
|
FUINT8 ValSize;
|
|
FUINT8 InShift;
|
|
FUINT8 OutShift;
|
|
UINT8* Ent1B;
|
|
UINT16* Ent2B;
|
|
|
|
// ReadBits Variables
|
|
FUINT8 BitsToRead;
|
|
FUINT8 BitReadVal;
|
|
FUINT8 InValB;
|
|
FUINT8 BitMask;
|
|
FUINT8 OutBit;
|
|
|
|
// Variables for DPCM
|
|
UINT16 OutMask;
|
|
|
|
ComprType = Data[0x00];
|
|
Bank->DataSize = ReadLE32(&Data[0x01]);
|
|
|
|
switch(ComprType)
|
|
{
|
|
case 0x00: // n-Bit compression
|
|
BitDec = Data[0x05];
|
|
BitCmp = Data[0x06];
|
|
CmpSubType = Data[0x07];
|
|
AddVal = ReadLE16(&Data[0x08]);
|
|
|
|
if (CmpSubType == 0x02)
|
|
{
|
|
Ent1B = (UINT8*)p->PCMTbl.Entries; // Big Endian note: Those are stored in LE and converted when reading.
|
|
Ent2B = (UINT16*)p->PCMTbl.Entries;
|
|
if (! p->PCMTbl.EntryCount)
|
|
{
|
|
Bank->DataSize = 0x00;
|
|
printf("Error loading table-compressed data block! No table loaded!\n");
|
|
return false;
|
|
}
|
|
else if (BitDec != p->PCMTbl.BitDec || BitCmp != p->PCMTbl.BitCmp)
|
|
{
|
|
Bank->DataSize = 0x00;
|
|
printf("Warning! Data block and loaded value table incompatible!\n");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
ValSize = (BitDec + 7) / 8;
|
|
InPos = Data + 0x0A;
|
|
InDataEnd = Data + DataSize;
|
|
InShift = 0;
|
|
OutShift = BitDec - BitCmp;
|
|
OutDataEnd = Bank->Data + Bank->DataSize;
|
|
|
|
for (OutPos = Bank->Data; OutPos < OutDataEnd && InPos < InDataEnd; OutPos += ValSize)
|
|
{
|
|
//InVal = ReadBits(Data, InPos, &InShift, BitCmp);
|
|
// inlined - is 30% faster
|
|
OutBit = 0x00;
|
|
InVal = 0x0000;
|
|
BitsToRead = BitCmp;
|
|
while(BitsToRead)
|
|
{
|
|
BitReadVal = (BitsToRead >= 8) ? 8 : BitsToRead;
|
|
BitsToRead -= BitReadVal;
|
|
BitMask = (1 << BitReadVal) - 1;
|
|
|
|
InShift += BitReadVal;
|
|
InValB = (*InPos << InShift >> 8) & BitMask;
|
|
if (InShift >= 8)
|
|
{
|
|
InShift -= 8;
|
|
InPos ++;
|
|
if (InShift)
|
|
InValB |= (*InPos << InShift >> 8) & BitMask;
|
|
}
|
|
|
|
InVal |= InValB << OutBit;
|
|
OutBit += BitReadVal;
|
|
}
|
|
|
|
switch(CmpSubType)
|
|
{
|
|
case 0x00: // Copy
|
|
OutVal = InVal + AddVal;
|
|
break;
|
|
case 0x01: // Shift Left
|
|
OutVal = (InVal << OutShift) + AddVal;
|
|
break;
|
|
case 0x02: // Table
|
|
switch(ValSize)
|
|
{
|
|
case 0x01:
|
|
OutVal = Ent1B[InVal];
|
|
break;
|
|
case 0x02:
|
|
#ifndef VGM_BIG_ENDIAN
|
|
OutVal = Ent2B[InVal];
|
|
#else
|
|
OutVal = ReadLE16((UINT8*)&Ent2B[InVal]);
|
|
#endif
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
|
|
#ifndef VGM_BIG_ENDIAN
|
|
//memcpy(OutPos, &OutVal, ValSize);
|
|
if (ValSize == 0x01)
|
|
*((UINT8*)OutPos) = (UINT8)OutVal;
|
|
else //if (ValSize == 0x02)
|
|
*((UINT16*)OutPos) = (UINT16)OutVal;
|
|
#else
|
|
if (ValSize == 0x01)
|
|
{
|
|
*OutPos = (UINT8)OutVal;
|
|
}
|
|
else //if (ValSize == 0x02)
|
|
{
|
|
OutPos[0x00] = (UINT8)((OutVal & 0x00FF) >> 0);
|
|
OutPos[0x01] = (UINT8)((OutVal & 0xFF00) >> 8);
|
|
}
|
|
#endif
|
|
}
|
|
break;
|
|
case 0x01: // Delta-PCM
|
|
BitDec = Data[0x05];
|
|
BitCmp = Data[0x06];
|
|
OutVal = ReadLE16(&Data[0x08]);
|
|
|
|
Ent1B = (UINT8*)p->PCMTbl.Entries;
|
|
Ent2B = (UINT16*)p->PCMTbl.Entries;
|
|
if (! p->PCMTbl.EntryCount)
|
|
{
|
|
Bank->DataSize = 0x00;
|
|
printf("Error loading table-compressed data block! No table loaded!\n");
|
|
return false;
|
|
}
|
|
else if (BitDec != p->PCMTbl.BitDec || BitCmp != p->PCMTbl.BitCmp)
|
|
{
|
|
Bank->DataSize = 0x00;
|
|
printf("Warning! Data block and loaded value table incompatible!\n");
|
|
return false;
|
|
}
|
|
|
|
ValSize = (BitDec + 7) / 8;
|
|
OutMask = (1 << BitDec) - 1;
|
|
InPos = Data + 0x0A;
|
|
InDataEnd = Data + DataSize;
|
|
InShift = 0;
|
|
OutShift = BitDec - BitCmp;
|
|
OutDataEnd = Bank->Data + Bank->DataSize;
|
|
AddVal = 0x0000;
|
|
|
|
for (OutPos = Bank->Data; OutPos < OutDataEnd && InPos < InDataEnd; OutPos += ValSize)
|
|
{
|
|
//InVal = ReadBits(Data, InPos, &InShift, BitCmp);
|
|
// inlined - is 30% faster
|
|
OutBit = 0x00;
|
|
InVal = 0x0000;
|
|
BitsToRead = BitCmp;
|
|
while(BitsToRead)
|
|
{
|
|
BitReadVal = (BitsToRead >= 8) ? 8 : BitsToRead;
|
|
BitsToRead -= BitReadVal;
|
|
BitMask = (1 << BitReadVal) - 1;
|
|
|
|
InShift += BitReadVal;
|
|
InValB = (*InPos << InShift >> 8) & BitMask;
|
|
if (InShift >= 8)
|
|
{
|
|
InShift -= 8;
|
|
InPos ++;
|
|
if (InShift)
|
|
InValB |= (*InPos << InShift >> 8) & BitMask;
|
|
}
|
|
|
|
InVal |= InValB << OutBit;
|
|
OutBit += BitReadVal;
|
|
}
|
|
|
|
switch(ValSize)
|
|
{
|
|
case 0x01:
|
|
AddVal = Ent1B[InVal];
|
|
OutVal += AddVal;
|
|
OutVal &= OutMask;
|
|
*((UINT8*)OutPos) = (UINT8)OutVal;
|
|
break;
|
|
case 0x02:
|
|
#ifndef VGM_BIG_ENDIAN
|
|
AddVal = Ent2B[InVal];
|
|
#else
|
|
AddVal = ReadLE16((UINT8*)&Ent2B[InVal]);
|
|
#endif
|
|
OutVal += AddVal;
|
|
OutVal &= OutMask;
|
|
#ifndef VGM_BIG_ENDIAN
|
|
*((UINT16*)OutPos) = (UINT16)OutVal;
|
|
#else
|
|
OutPos[0x00] = (UINT8)((OutVal & 0x00FF) >> 0);
|
|
OutPos[0x01] = (UINT8)((OutVal & 0xFF00) >> 8);
|
|
#endif
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
printf("Error: Unknown data block compression!\n");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static UINT8 GetDACFromPCMBank(VGM_PLAYER* p)
|
|
{
|
|
// for YM2612 DAC data only
|
|
/*VGM_PCM_BANK* TempPCM;
|
|
UINT32 CurBnk;*/
|
|
UINT32 DataPos;
|
|
|
|
/*TempPCM = &PCMBank[0x00];
|
|
DataPos = TempPCM->DataPos;
|
|
for (CurBnk = 0x00; CurBnk < TempPCM->BankCount; CurBnk ++)
|
|
{
|
|
if (DataPos < TempPCM->Bank[CurBnk].DataSize)
|
|
{
|
|
if (TempPCM->DataPos < TempPCM->DataSize)
|
|
TempPCM->DataPos ++;
|
|
return TempPCM->Bank[CurBnk].Data[DataPos];
|
|
}
|
|
DataPos -= TempPCM->Bank[CurBnk].DataSize;
|
|
}
|
|
return 0x80;*/
|
|
|
|
DataPos = p->PCMBank[0x00].DataPos;
|
|
if (DataPos >= p->PCMBank[0x00].DataSize)
|
|
return 0x80;
|
|
|
|
p->PCMBank[0x00].DataPos ++;
|
|
return p->PCMBank[0x00].Data[DataPos];
|
|
}
|
|
|
|
static UINT8* GetPointerFromPCMBank(VGM_PLAYER* p, UINT8 Type, UINT32 DataPos)
|
|
{
|
|
if (Type >= PCM_BANK_COUNT)
|
|
return NULL;
|
|
|
|
if (DataPos >= p->PCMBank[Type].DataSize)
|
|
return NULL;
|
|
|
|
return &p->PCMBank[Type].Data[DataPos];
|
|
}
|
|
|
|
static void ReadPCMTable(VGM_PLAYER* p, UINT32 DataSize, const UINT8* Data)
|
|
{
|
|
UINT8 ValSize;
|
|
UINT32 TblSize;
|
|
|
|
p->PCMTbl.ComprType = Data[0x00];
|
|
p->PCMTbl.CmpSubType = Data[0x01];
|
|
p->PCMTbl.BitDec = Data[0x02];
|
|
p->PCMTbl.BitCmp = Data[0x03];
|
|
p->PCMTbl.EntryCount = ReadLE16(&Data[0x04]);
|
|
|
|
ValSize = (p->PCMTbl.BitDec + 7) / 8;
|
|
TblSize = p->PCMTbl.EntryCount * ValSize;
|
|
|
|
p->PCMTbl.Entries = realloc(p->PCMTbl.Entries, TblSize);
|
|
memcpy(p->PCMTbl.Entries, &Data[0x06], TblSize);
|
|
|
|
if (DataSize < 0x06 + TblSize)
|
|
printf("Warning! Bad PCM Table Length!\n");
|
|
|
|
return;
|
|
}
|
|
|
|
#define CHIP_CHECK(name) (p->ChipAudio[CurChip].name.ChipType != 0xFF)
|
|
static void InterpretVGM(VGM_PLAYER* p, UINT32 SampleCount)
|
|
{
|
|
INT32 SmplPlayed;
|
|
UINT8 Command;
|
|
UINT8 TempByt;
|
|
UINT16 TempSht;
|
|
UINT32 TempLng;
|
|
VGM_PCM_BANK* TempPCM;
|
|
VGM_PCM_DATA* TempBnk;
|
|
UINT32 ROMSize;
|
|
UINT32 DataStart;
|
|
UINT32 DataLen;
|
|
const UINT8* ROMData;
|
|
UINT8 CurChip;
|
|
const UINT8* VGMPnt;
|
|
|
|
if (p->VGMEnd)
|
|
return;
|
|
|
|
SmplPlayed = SamplePbk2VGM_I(p, p->VGMSmplPlayed + SampleCount);
|
|
while(p->VGMSmplPos <= SmplPlayed)
|
|
{
|
|
Command = p->VGMData[p->VGMPos + 0x00];
|
|
if (Command >= 0x70 && Command <= 0x8F)
|
|
{
|
|
switch(Command & 0xF0)
|
|
{
|
|
case 0x70:
|
|
p->VGMSmplPos += (Command & 0x0F) + 0x01;
|
|
break;
|
|
case 0x80:
|
|
TempByt = GetDACFromPCMBank(p);
|
|
if (p->VGMHead.lngHzYM2612)
|
|
{
|
|
chip_reg_write(p, 0x02, 0x00, 0x00, 0x2A, TempByt);
|
|
}
|
|
p->VGMSmplPos += (Command & 0x0F);
|
|
break;
|
|
}
|
|
p->VGMPos += 0x01;
|
|
}
|
|
else
|
|
{
|
|
VGMPnt = &p->VGMData[p->VGMPos];
|
|
|
|
// Cheat Mode (to use 2 instances of 1 chip)
|
|
CurChip = 0x00;
|
|
switch(Command)
|
|
{
|
|
case 0x30:
|
|
if (p->VGMHead.lngHzPSG & 0x40000000)
|
|
{
|
|
Command += 0x20;
|
|
CurChip = 0x01;
|
|
}
|
|
break;
|
|
case 0x3F:
|
|
if (p->VGMHead.lngHzPSG & 0x40000000)
|
|
{
|
|
Command += 0x10;
|
|
CurChip = 0x01;
|
|
}
|
|
break;
|
|
case 0xA1:
|
|
if (p->VGMHead.lngHzYM2413 & 0x40000000)
|
|
{
|
|
Command -= 0x50;
|
|
CurChip = 0x01;
|
|
}
|
|
break;
|
|
case 0xA2:
|
|
case 0xA3:
|
|
if (p->VGMHead.lngHzYM2612 & 0x40000000)
|
|
{
|
|
Command -= 0x50;
|
|
CurChip = 0x01;
|
|
}
|
|
break;
|
|
case 0xA4:
|
|
if (p->VGMHead.lngHzYM2151 & 0x40000000)
|
|
{
|
|
Command -= 0x50;
|
|
CurChip = 0x01;
|
|
}
|
|
break;
|
|
case 0xA5:
|
|
if (p->VGMHead.lngHzYM2203 & 0x40000000)
|
|
{
|
|
Command -= 0x50;
|
|
CurChip = 0x01;
|
|
}
|
|
break;
|
|
case 0xA6:
|
|
case 0xA7:
|
|
if (p->VGMHead.lngHzYM2608 & 0x40000000)
|
|
{
|
|
Command -= 0x50;
|
|
CurChip = 0x01;
|
|
}
|
|
break;
|
|
case 0xA8:
|
|
case 0xA9:
|
|
if (p->VGMHead.lngHzYM2610 & 0x40000000)
|
|
{
|
|
Command -= 0x50;
|
|
CurChip = 0x01;
|
|
}
|
|
break;
|
|
case 0xAA:
|
|
if (p->VGMHead.lngHzYM3812 & 0x40000000)
|
|
{
|
|
Command -= 0x50;
|
|
CurChip = 0x01;
|
|
}
|
|
break;
|
|
case 0xAB:
|
|
if (p->VGMHead.lngHzYM3526 & 0x40000000)
|
|
{
|
|
Command -= 0x50;
|
|
CurChip = 0x01;
|
|
}
|
|
break;
|
|
case 0xAC:
|
|
if (p->VGMHead.lngHzY8950 & 0x40000000)
|
|
{
|
|
Command -= 0x50;
|
|
CurChip = 0x01;
|
|
}
|
|
break;
|
|
case 0xAE:
|
|
case 0xAF:
|
|
if (p->VGMHead.lngHzYMF262 & 0x40000000)
|
|
{
|
|
Command -= 0x50;
|
|
CurChip = 0x01;
|
|
}
|
|
break;
|
|
case 0xAD:
|
|
if (p->VGMHead.lngHzYMZ280B & 0x40000000)
|
|
{
|
|
Command -= 0x50;
|
|
CurChip = 0x01;
|
|
}
|
|
break;
|
|
}
|
|
|
|
switch(Command)
|
|
{
|
|
case 0x66: // End Of File
|
|
if (p->VGMHead.lngLoopOffset)
|
|
{
|
|
p->VGMPos = p->VGMHead.lngLoopOffset;
|
|
p->VGMSmplPos -= p->VGMHead.lngLoopSamples;
|
|
p->VGMSmplPlayed -= SampleVGM2Pbk_I(p, p->VGMHead.lngLoopSamples);
|
|
SmplPlayed = SamplePbk2VGM_I(p, p->VGMSmplPlayed + SampleCount);
|
|
p->VGMCurLoop ++;
|
|
|
|
if (p->VGMMaxLoopM && p->VGMCurLoop >= p->VGMMaxLoopM)
|
|
{
|
|
if (! p->FadePlay)
|
|
{
|
|
p->FadeStart = SampleVGM2Pbk_I(p, p->VGMHead.lngTotalSamples +
|
|
(p->VGMCurLoop - 1) * p->VGMHead.lngLoopSamples);
|
|
}
|
|
p->FadePlay = true;
|
|
}
|
|
if (p->FadePlay && ! p->FadeTime)
|
|
p->VGMEnd = true;
|
|
}
|
|
else
|
|
{
|
|
if (p->VGMHead.lngTotalSamples != (UINT32)p->VGMSmplPos)
|
|
{
|
|
#ifdef CONSOLE_MODE
|
|
printf("Warning! Header Samples: %u\t Counted Samples: %u\n",
|
|
p->VGMHead.lngTotalSamples, p->VGMSmplPos);
|
|
p->ErrorHappened = true;
|
|
#endif
|
|
p->VGMHead.lngTotalSamples = p->VGMSmplPos;
|
|
}
|
|
p->VGMEnd = true;
|
|
break;
|
|
}
|
|
break;
|
|
case 0x62: // 1/60s delay
|
|
p->VGMSmplPos += 735;
|
|
p->VGMPos += 0x01;
|
|
break;
|
|
case 0x63: // 1/50s delay
|
|
p->VGMSmplPos += 882;
|
|
p->VGMPos += 0x01;
|
|
break;
|
|
case 0x61: // xx Sample Delay
|
|
TempSht = ReadLE16(&VGMPnt[0x01]);
|
|
p->VGMSmplPos += TempSht;
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0x50: // SN76496 write
|
|
if (CHIP_CHECK(SN76496))
|
|
{
|
|
chip_reg_write(p, 0x00, CurChip, 0x00, 0x00, VGMPnt[0x01]);
|
|
}
|
|
p->VGMPos += 0x02;
|
|
break;
|
|
case 0x51: // YM2413 write
|
|
if (CHIP_CHECK(YM2413))
|
|
{
|
|
chip_reg_write(p, 0x01, CurChip, 0x00, VGMPnt[0x01], VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0x52: // YM2612 write port 0
|
|
case 0x53: // YM2612 write port 1
|
|
if (CHIP_CHECK(YM2612))
|
|
{
|
|
chip_reg_write(p, 0x02, CurChip, Command & 0x01, VGMPnt[0x01], VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0x67: // PCM Data Stream
|
|
TempByt = VGMPnt[0x02];
|
|
TempLng = ReadLE32(&VGMPnt[0x03]);
|
|
if (TempLng & 0x80000000)
|
|
{
|
|
TempLng &= 0x7FFFFFFF;
|
|
CurChip = 0x01;
|
|
}
|
|
|
|
switch(TempByt & 0xC0)
|
|
{
|
|
case 0x00: // Database Block
|
|
case 0x40:
|
|
AddPCMData(p, TempByt, TempLng, &VGMPnt[0x07]);
|
|
/*switch(TempByt)
|
|
{
|
|
case 0x00: // YM2612 PCM Database
|
|
break;
|
|
case 0x01: // RF5C68 PCM Database
|
|
break;
|
|
case 0x02: // RF5C164 PCM Database
|
|
break;
|
|
}*/
|
|
break;
|
|
case 0x80: // ROM/RAM Dump
|
|
if (p->VGMCurLoop)
|
|
break;
|
|
|
|
ROMSize = ReadLE32(&VGMPnt[0x07]);
|
|
DataStart = ReadLE32(&VGMPnt[0x0B]);
|
|
DataLen = TempLng - 0x08;
|
|
ROMData = &VGMPnt[0x0F];
|
|
switch(TempByt)
|
|
{
|
|
case 0x80: // SegaPCM ROM
|
|
if (! CHIP_CHECK(SegaPCM))
|
|
break;
|
|
sega_pcm_write_rom(p->segapcm[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x81: // YM2608 DELTA-T ROM Image
|
|
if (! CHIP_CHECK(YM2608))
|
|
break;
|
|
ym2608_write_data_pcmrom(p->ym2608[CurChip], 0x02, ROMSize, DataStart, DataLen,
|
|
ROMData);
|
|
break;
|
|
case 0x82: // YM2610 ADPCM ROM Image
|
|
case 0x83: // YM2610 DELTA-T ROM Image
|
|
if (! CHIP_CHECK(YM2610))
|
|
break;
|
|
TempByt = 0x01 + (TempByt - 0x82);
|
|
ym2610_write_data_pcmrom(p->ym2610[CurChip], TempByt, ROMSize, DataStart, DataLen,
|
|
ROMData);
|
|
break;
|
|
case 0x84: // YMF278B ROM Image
|
|
if (! CHIP_CHECK(YMF278B))
|
|
break;
|
|
ymf278b_write_rom(p->ymf278b[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x85: // YMF271 ROM Image
|
|
if (! CHIP_CHECK(YMF271))
|
|
break;
|
|
ymf271_write_rom(p->ymf271[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x86: // YMZ280B ROM Image
|
|
if (! CHIP_CHECK(YMZ280B))
|
|
break;
|
|
ymz280b_write_rom(p->ymz280b[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x87: // YMF278B RAM Image
|
|
if (! CHIP_CHECK(YMF278B))
|
|
break;
|
|
//ymf278b_write_ram(CurChip, ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x88: // Y8950 DELTA-T ROM Image
|
|
if (! CHIP_CHECK(Y8950) || p->PlayingMode == 0x01)
|
|
break;
|
|
y8950_write_data_pcmrom(p->y8950[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x89: // MultiPCM ROM Image
|
|
if (! CHIP_CHECK(MultiPCM))
|
|
break;
|
|
multipcm_write_rom(p->multipcm[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x8A: // UPD7759 ROM Image
|
|
if (! CHIP_CHECK(UPD7759))
|
|
break;
|
|
upd7759_write_rom(p->upd7759[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x8B: // OKIM6295 ROM Image
|
|
if (! CHIP_CHECK(OKIM6295))
|
|
break;
|
|
okim6295_write_rom(p->okim6295[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x8C: // K054539 ROM Image
|
|
if (! CHIP_CHECK(K054539))
|
|
break;
|
|
k054539_write_rom(p->k054539[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x8D: // C140 ROM Image
|
|
if (! CHIP_CHECK(C140))
|
|
break;
|
|
c140_write_rom(p->c140[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x8E: // K053260 ROM Image
|
|
if (! CHIP_CHECK(K053260))
|
|
break;
|
|
k053260_write_rom(p->k053260[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x8F: // QSound ROM Image
|
|
if (! CHIP_CHECK(QSound))
|
|
break;
|
|
qsound_write_rom(p->qsound[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x90: // ES5506 ROM Image
|
|
if (! CHIP_CHECK(ES5506))
|
|
break;
|
|
es5506_write_rom(p->es550x[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x91: // X1-010 ROM Image
|
|
if (! CHIP_CHECK(X1_010))
|
|
break;
|
|
x1_010_write_rom(p->x1_010[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x92: // C352 ROM Image
|
|
if (! CHIP_CHECK(C352))
|
|
break;
|
|
c352_write_rom(p->c352[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0x93: // GA20 ROM Image
|
|
if (! CHIP_CHECK(GA20))
|
|
break;
|
|
iremga20_write_rom(p->ga20[CurChip], ROMSize, DataStart, DataLen, ROMData);
|
|
break;
|
|
// case 0x8C: // OKIM6376 ROM Image
|
|
// if (! CHIP_CHECK(OKIM6376))
|
|
// break;
|
|
// break;
|
|
}
|
|
break;
|
|
case 0xC0: // RAM Write
|
|
if (! (TempByt & 0x20))
|
|
{
|
|
DataStart = ReadLE16(&VGMPnt[0x07]);
|
|
DataLen = TempLng - 0x02;
|
|
ROMData = &VGMPnt[0x09];
|
|
}
|
|
else
|
|
{
|
|
DataStart = ReadLE32(&VGMPnt[0x07]);
|
|
DataLen = TempLng - 0x04;
|
|
ROMData = &VGMPnt[0x0B];
|
|
}
|
|
switch(TempByt)
|
|
{
|
|
case 0xC0: // RF5C68 RAM Database
|
|
if (! CHIP_CHECK(RF5C68))
|
|
break;
|
|
rf5c68_write_ram(p->rf5c68, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0xC1: // RF5C164 RAM Database
|
|
if (! CHIP_CHECK(RF5C164))
|
|
break;
|
|
rf5c164_write_ram(p->rf5c164, DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0xC2: // NES APU RAM
|
|
if (! CHIP_CHECK(NES))
|
|
break;
|
|
nes_write_ram(p->nesapu[CurChip], DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0xE0: // SCSP RAM
|
|
if (! CHIP_CHECK(SCSP))
|
|
break;
|
|
scsp_write_ram(p->scsp[CurChip], DataStart, DataLen, ROMData);
|
|
break;
|
|
case 0xE1: // ES5503 RAM
|
|
if (! CHIP_CHECK(ES5503))
|
|
break;
|
|
es5503_write_ram(p->es5503[CurChip], DataStart, DataLen, ROMData);
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
p->VGMPos += 0x07 + TempLng;
|
|
break;
|
|
case 0xE0: // Seek to PCM Data Bank Pos
|
|
p->PCMBank[0x00].DataPos = ReadLE32(&VGMPnt[0x01]);
|
|
p->VGMPos += 0x05;
|
|
break;
|
|
case 0x4F: // GG Stereo
|
|
if (CHIP_CHECK(SN76496))
|
|
{
|
|
chip_reg_write(p, 0x00, CurChip, 0x01, 0x00, VGMPnt[0x01]);
|
|
}
|
|
p->VGMPos += 0x02;
|
|
break;
|
|
case 0x54: // YM2151 write
|
|
if (CHIP_CHECK(YM2151))
|
|
{
|
|
chip_reg_write(p, 0x03, CurChip, 0x01, VGMPnt[0x01], VGMPnt[0x02]);
|
|
}
|
|
//p->VGMSmplPos += 80;
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xC0: // Sega PCM memory write
|
|
TempSht = ReadLE16(&VGMPnt[0x01]);
|
|
CurChip = (TempSht & 0x8000) >> 15;
|
|
if (CHIP_CHECK(SegaPCM))
|
|
{
|
|
sega_pcm_w(p->segapcm[CurChip], TempSht & 0x7FFF, VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xB0: // RF5C68 register write
|
|
if (CHIP_CHECK(RF5C68))
|
|
{
|
|
chip_reg_write(p, 0x05, CurChip, 0x00, VGMPnt[0x01], VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xC1: // RF5C68 memory write
|
|
if (CHIP_CHECK(RF5C68))
|
|
{
|
|
TempSht = ReadLE16(&VGMPnt[0x01]);
|
|
rf5c68_mem_w(p->rf5c68, TempSht, VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0x55: // YM2203
|
|
if (CHIP_CHECK(YM2203))
|
|
{
|
|
chip_reg_write(p, 0x06, CurChip, 0x00, VGMPnt[0x01], VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0x56: // YM2608 write port 0
|
|
case 0x57: // YM2608 write port 1
|
|
if (CHIP_CHECK(YM2608))
|
|
{
|
|
chip_reg_write(p, 0x07, CurChip, Command & 0x01, VGMPnt[0x01], VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0x58: // YM2610 write port 0
|
|
case 0x59: // YM2610 write port 1
|
|
if (CHIP_CHECK(YM2610))
|
|
{
|
|
chip_reg_write(p, 0x08, CurChip, Command & 0x01, VGMPnt[0x01], VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0x5A: // YM3812 write
|
|
if (CHIP_CHECK(YM3812))
|
|
{
|
|
chip_reg_write(p, 0x09, CurChip, 0x00, VGMPnt[0x01], VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0x5B: // YM3526 write
|
|
if (CHIP_CHECK(YM3526))
|
|
{
|
|
chip_reg_write(p, 0x0A, CurChip, 0x00, VGMPnt[0x01], VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0x5C: // Y8950 write
|
|
if (CHIP_CHECK(Y8950))
|
|
{
|
|
chip_reg_write(p, 0x0B, CurChip, 0x00, VGMPnt[0x01], VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0x5E: // YMF262 write port 0
|
|
case 0x5F: // YMF262 write port 1
|
|
if (CHIP_CHECK(YMF262))
|
|
{
|
|
chip_reg_write(p, 0x0C, CurChip, Command & 0x01, VGMPnt[0x01], VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0x5D: // YMZ280B write
|
|
if (CHIP_CHECK(YMZ280B))
|
|
{
|
|
chip_reg_write(p, 0x0F, CurChip, 0x00, VGMPnt[0x01], VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xD0: // YMF278B write
|
|
if (CHIP_CHECK(YMF278B))
|
|
{
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
chip_reg_write(p, 0x0D, CurChip, VGMPnt[0x01] & 0x7F, VGMPnt[0x02], VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xD1: // YMF271 write
|
|
if (CHIP_CHECK(YMF271))
|
|
{
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
chip_reg_write(p, 0x0E, CurChip, VGMPnt[0x01] & 0x7F, VGMPnt[0x02], VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xB1: // RF5C164 register write
|
|
if (CHIP_CHECK(RF5C164))
|
|
{
|
|
chip_reg_write(p, 0x10, CurChip, 0x00, VGMPnt[0x01], VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xC2: // RF5C164 memory write
|
|
if (CHIP_CHECK(RF5C164))
|
|
{
|
|
TempSht = ReadLE16(&VGMPnt[0x01]);
|
|
rf5c164_mem_w(p->rf5c164, TempSht, VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xB2: // PWM channel write
|
|
if (CHIP_CHECK(PWM))
|
|
{
|
|
chip_reg_write(p, 0x11, CurChip, (VGMPnt[0x01] & 0xF0) >> 4,
|
|
VGMPnt[0x01] & 0x0F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0x68: // PCM RAM write
|
|
CurChip = (VGMPnt[0x02] & 0x80) >> 7;
|
|
TempByt = VGMPnt[0x02] & 0x7F;
|
|
|
|
DataStart = ReadLE24(&VGMPnt[0x03]);
|
|
TempLng = ReadLE24(&VGMPnt[0x06]);
|
|
DataLen = ReadLE24(&VGMPnt[0x09]);
|
|
if (! DataLen)
|
|
DataLen += 0x01000000;
|
|
ROMData = GetPointerFromPCMBank(p, TempByt, DataStart);
|
|
if (ROMData == NULL)
|
|
{
|
|
p->VGMPos += 0x0C;
|
|
break;
|
|
}
|
|
|
|
switch(TempByt)
|
|
{
|
|
case 0x01:
|
|
if (! CHIP_CHECK(RF5C68))
|
|
break;
|
|
rf5c68_write_ram(p->rf5c68, TempLng, DataLen, ROMData);
|
|
break;
|
|
case 0x02:
|
|
if (! CHIP_CHECK(RF5C164))
|
|
break;
|
|
rf5c164_write_ram(p->rf5c164, TempLng, DataLen, ROMData);
|
|
break;
|
|
case 0x06:
|
|
if (! CHIP_CHECK(SCSP))
|
|
break;
|
|
scsp_write_ram(p->scsp[CurChip], TempLng, DataLen, ROMData);
|
|
break;
|
|
case 0x07:
|
|
if (! CHIP_CHECK(NES))
|
|
break;
|
|
p->Last95Drum = DataStart / DataLen - 1;
|
|
p->Last95Max = p->PCMBank[TempByt].DataSize / DataLen;
|
|
nes_write_ram(p->nesapu[CurChip], TempLng, DataLen, ROMData);
|
|
break;
|
|
}
|
|
p->VGMPos += 0x0C;
|
|
break;
|
|
case 0xA0: // AY8910 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(AY8910))
|
|
{
|
|
chip_reg_write(p, 0x12, CurChip, 0x00, VGMPnt[0x01] & 0x7F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xB3: // GameBoy DMG write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(GameBoy))
|
|
{
|
|
chip_reg_write(p, 0x13, CurChip, 0x00, VGMPnt[0x01] & 0x7F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xB4: // NES APU write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(NES))
|
|
{
|
|
chip_reg_write(p, 0x14, CurChip, 0x00, VGMPnt[0x01] & 0x7F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xB5: // MultiPCM write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(MultiPCM))
|
|
{
|
|
chip_reg_write(p, 0x15, CurChip, 0x00, VGMPnt[0x01] & 0x7F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xC3: // MultiPCM memory write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(MultiPCM))
|
|
{
|
|
TempSht = ReadLE16(&VGMPnt[0x02]);
|
|
multipcm_bank_write(p->multipcm[CurChip], VGMPnt[0x01] & 0x7F, TempSht);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xB6: // UPD7759 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(UPD7759))
|
|
{
|
|
chip_reg_write(p, 0x16, CurChip, 0x00, VGMPnt[0x01] & 0x7F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xB7: // OKIM6258 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(OKIM6258))
|
|
{
|
|
chip_reg_write(p, 0x17, CurChip, 0x00, VGMPnt[0x01] & 0x7F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xB8: // OKIM6295 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(OKIM6295))
|
|
{
|
|
chip_reg_write(p, 0x18, CurChip, 0x00, VGMPnt[0x01] & 0x7F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xD2: // SCC1 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(K051649))
|
|
{
|
|
chip_reg_write(p, 0x19, CurChip, VGMPnt[0x01] & 0x7F, VGMPnt[0x02],
|
|
VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xD3: // K054539 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(K054539))
|
|
{
|
|
chip_reg_write(p, 0x1A, CurChip, VGMPnt[0x01] & 0x7F, VGMPnt[0x02],
|
|
VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xB9: // HuC6280 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(HuC6280))
|
|
{
|
|
chip_reg_write(p, 0x1B, CurChip, 0x00, VGMPnt[0x01] & 0x7F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xD4: // C140 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(C140))
|
|
{
|
|
chip_reg_write(p, 0x1C, CurChip, VGMPnt[0x01] & 0x7F, VGMPnt[0x02],
|
|
VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xBA: // K053260 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(K053260))
|
|
{
|
|
chip_reg_write(p, 0x1D, CurChip, 0x00, VGMPnt[0x01] & 0x7F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xBB: // Pokey write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(Pokey))
|
|
{
|
|
chip_reg_write(p, 0x1E, CurChip, 0x00, VGMPnt[0x01] & 0x7F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xC4: // QSound write
|
|
if (CHIP_CHECK(QSound))
|
|
{
|
|
chip_reg_write(p, 0x1F, CurChip, VGMPnt[0x01], VGMPnt[0x02], VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xC5: // YMF292/SCSP write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(SCSP))
|
|
{
|
|
chip_reg_write(p, 0x20, CurChip, VGMPnt[0x01] & 0x7F, VGMPnt[0x02],
|
|
VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xBC: // WonderSwan write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(WSwan))
|
|
{
|
|
chip_reg_write(p, 0x21, CurChip, 0x00, VGMPnt[0x01] & 0x7F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xC6: // WonderSwan memory write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(WSwan))
|
|
{
|
|
TempSht = ReadBE16(&VGMPnt[0x01]) & 0x7FFF;
|
|
ws_write_ram(p->wswan[CurChip], TempSht, VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xC7: // VSU write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(VSU))
|
|
{
|
|
chip_reg_write(p, 0x22, CurChip, VGMPnt[0x01] & 0x7F, VGMPnt[0x02],
|
|
VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xBD: // SAA1099 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(SAA1099))
|
|
{
|
|
chip_reg_write(p, 0x23, CurChip, 0x00, VGMPnt[0x01] & 0x7F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xD5: // ES5503 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(ES5503))
|
|
{
|
|
chip_reg_write(p, 0x24, CurChip, VGMPnt[0x01] & 0x7F, VGMPnt[0x02],
|
|
VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xBE: // ES5506 write (8-bit data)
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(ES5506))
|
|
{
|
|
chip_reg_write(p, 0x25, CurChip, VGMPnt[0x01] & 0x7F, 0x00, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xD6: // ES5506 write (16-bit data)
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(ES5506))
|
|
{
|
|
chip_reg_write(p, 0x25, CurChip, 0x80 | (VGMPnt[0x01] & 0x7F),
|
|
VGMPnt[0x02], VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xC8: // X1-010 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(X1_010))
|
|
{
|
|
chip_reg_write(p, 0x26, CurChip, VGMPnt[0x01] & 0x7F, VGMPnt[0x02],
|
|
VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
#if 0 // for ctr's WIP rips
|
|
case 0xC9: // C352 write
|
|
CurChip = 0x00;
|
|
if (CHIP_CHECK(C352))
|
|
{
|
|
if (VGMPnt[0x01] == 0x03 && VGMPnt[0x02] == 0xFF && VGMPnt[0x03] == 0xFF)
|
|
c352_w(p->c352[CurChip], 0x202, 0x0020);
|
|
else
|
|
chip_reg_write(p, 0x27, CurChip, VGMPnt[0x01], VGMPnt[0x02],
|
|
VGMPnt[0x03]);
|
|
}
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
#endif
|
|
case 0xE1: // C352 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(C352))
|
|
{
|
|
TempSht = ((VGMPnt[0x01] & 0x7F) << 8) | (VGMPnt[0x02] << 0);
|
|
c352_w(p->c352[CurChip], TempSht, (VGMPnt[0x03] << 8) | VGMPnt[0x04]);
|
|
}
|
|
p->VGMPos += 0x05;
|
|
break;
|
|
case 0xBF: // GA20 write
|
|
CurChip = (VGMPnt[0x01] & 0x80) >> 7;
|
|
if (CHIP_CHECK(GA20))
|
|
{
|
|
chip_reg_write(p, 0x28, CurChip, 0x00, VGMPnt[0x01] & 0x7F, VGMPnt[0x02]);
|
|
}
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0x90: // DAC Ctrl: Setup Chip
|
|
CurChip = VGMPnt[0x01];
|
|
if (CurChip == 0xFF)
|
|
{
|
|
p->VGMPos += 0x05;
|
|
break;
|
|
}
|
|
if (! p->DacCtrl[CurChip].Enable)
|
|
{
|
|
device_start_daccontrol(&p->daccontrol[CurChip], p, p->SampleRate);
|
|
device_reset_daccontrol(p->daccontrol[CurChip]);
|
|
p->DacCtrl[CurChip].Enable = true;
|
|
p->DacCtrlUsg[p->DacCtrlUsed] = CurChip;
|
|
p->DacCtrlUsed ++;
|
|
}
|
|
TempByt = VGMPnt[0x02]; // Chip Type
|
|
TempSht = ReadBE16(&VGMPnt[0x03]);
|
|
daccontrol_setup_chip(p->daccontrol[CurChip], TempByt & 0x7F, (TempByt & 0x80) >> 7, TempSht);
|
|
p->VGMPos += 0x05;
|
|
break;
|
|
case 0x91: // DAC Ctrl: Set Data
|
|
CurChip = VGMPnt[0x01];
|
|
if (CurChip == 0xFF || ! p->DacCtrl[CurChip].Enable)
|
|
{
|
|
p->VGMPos += 0x05;
|
|
break;
|
|
}
|
|
p->DacCtrl[CurChip].Bank = VGMPnt[0x02];
|
|
if (p->DacCtrl[CurChip].Bank >= PCM_BANK_COUNT)
|
|
p->DacCtrl[CurChip].Bank = 0x00;
|
|
|
|
TempPCM = &p->PCMBank[p->DacCtrl[CurChip].Bank];
|
|
p->Last95Max = TempPCM->BankCount;
|
|
daccontrol_set_data(p->daccontrol[CurChip], TempPCM->Data, TempPCM->DataSize,
|
|
VGMPnt[0x03], VGMPnt[0x04]);
|
|
p->VGMPos += 0x05;
|
|
break;
|
|
case 0x92: // DAC Ctrl: Set Freq
|
|
CurChip = VGMPnt[0x01];
|
|
if (CurChip == 0xFF || ! p->DacCtrl[CurChip].Enable)
|
|
{
|
|
p->VGMPos += 0x06;
|
|
break;
|
|
}
|
|
TempLng = ReadLE32(&VGMPnt[0x02]);
|
|
p->Last95Freq = TempLng;
|
|
daccontrol_set_frequency(p->daccontrol[CurChip], TempLng);
|
|
p->VGMPos += 0x06;
|
|
break;
|
|
case 0x93: // DAC Ctrl: Play from Start Pos
|
|
CurChip = VGMPnt[0x01];
|
|
if (CurChip == 0xFF || ! p->DacCtrl[CurChip].Enable ||
|
|
! p->PCMBank[p->DacCtrl[CurChip].Bank].BankCount)
|
|
{
|
|
p->VGMPos += 0x0B;
|
|
break;
|
|
}
|
|
DataStart = ReadLE32(&VGMPnt[0x02]);
|
|
p->Last95Drum = 0xFFFF;
|
|
TempByt = VGMPnt[0x06];
|
|
DataLen = ReadLE32(&VGMPnt[0x07]);
|
|
daccontrol_start(p->daccontrol[CurChip], DataStart, TempByt, DataLen);
|
|
p->VGMPos += 0x0B;
|
|
break;
|
|
case 0x94: // DAC Ctrl: Stop immediately
|
|
CurChip = VGMPnt[0x01];
|
|
if (! p->DacCtrl[CurChip].Enable)
|
|
{
|
|
p->VGMPos += 0x02;
|
|
break;
|
|
}
|
|
p->Last95Drum = 0xFFFF;
|
|
if (CurChip < 0xFF)
|
|
{
|
|
daccontrol_stop(p->daccontrol[CurChip]);
|
|
}
|
|
else
|
|
{
|
|
for (CurChip = 0x00; CurChip < 0xFF; CurChip ++)
|
|
daccontrol_stop(p->daccontrol[CurChip]);
|
|
}
|
|
p->VGMPos += 0x02;
|
|
break;
|
|
case 0x95: // DAC Ctrl: Play Block (small)
|
|
CurChip = VGMPnt[0x01];
|
|
if (CurChip == 0xFF || ! p->DacCtrl[CurChip].Enable ||
|
|
! p->PCMBank[p->DacCtrl[CurChip].Bank].BankCount)
|
|
{
|
|
p->VGMPos += 0x05;
|
|
break;
|
|
}
|
|
TempPCM = &p->PCMBank[p->DacCtrl[CurChip].Bank];
|
|
TempSht = ReadLE16(&VGMPnt[0x02]);
|
|
p->Last95Drum = TempSht;
|
|
p->Last95Max = TempPCM->BankCount;
|
|
if (TempSht >= TempPCM->BankCount)
|
|
TempSht = 0x00;
|
|
TempBnk = &TempPCM->Bank[TempSht];
|
|
|
|
TempByt = DCTRL_LMODE_BYTES |
|
|
(VGMPnt[0x04] & 0x10) | // Reverse Mode
|
|
((VGMPnt[0x04] & 0x01) << 7); // Looping
|
|
daccontrol_start(p->daccontrol[CurChip], TempBnk->DataStart, TempByt, TempBnk->DataSize);
|
|
p->VGMPos += 0x05;
|
|
break;
|
|
default:
|
|
|
|
switch(Command & 0xF0)
|
|
{
|
|
case 0x00:
|
|
case 0x10:
|
|
case 0x20:
|
|
p->VGMPos += 0x01;
|
|
break;
|
|
case 0x30:
|
|
p->VGMPos += 0x02;
|
|
break;
|
|
case 0x40:
|
|
case 0x50:
|
|
case 0xA0:
|
|
case 0xB0:
|
|
p->VGMPos += 0x03;
|
|
break;
|
|
case 0xC0:
|
|
case 0xD0:
|
|
p->VGMPos += 0x04;
|
|
break;
|
|
case 0xE0:
|
|
case 0xF0:
|
|
p->VGMPos += 0x05;
|
|
break;
|
|
default:
|
|
p->VGMEnd = true;
|
|
p->EndPlay = true;
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (p->VGMPos >= p->VGMHead.lngEOFOffset)
|
|
p->VGMEnd = true;
|
|
|
|
if (p->VGMEnd)
|
|
break;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
static void GeneralChipLists(VGM_PLAYER* p)
|
|
{
|
|
// Generate Chip List for playback loop
|
|
UINT16 CurBufIdx;
|
|
CA_LIST* CLstOld;
|
|
CA_LIST* CLst;
|
|
CA_LIST* CurLst;
|
|
UINT8 CurChip;
|
|
UINT8 CurCSet;
|
|
CAUD_ATTR* CAA;
|
|
|
|
p->ChipListAll = NULL;
|
|
//ChipListPause = NULL;
|
|
//ChipListOpt = NULL;
|
|
|
|
// generate list of all chips that are used in the current VGM
|
|
CurBufIdx = 0x00;
|
|
CLstOld = NULL;
|
|
for (CurChip = 0x00; CurChip < CHIP_COUNT; CurChip ++)
|
|
{
|
|
for (CurCSet = 0x00; CurCSet < 0x02; CurCSet ++)
|
|
{
|
|
CAA = (CAUD_ATTR*)&p->ChipAudio[CurCSet] + CurChip;
|
|
if (CAA->ChipType != 0xFF)
|
|
{
|
|
CLst = &p->ChipListBuffer[CurBufIdx];
|
|
CurBufIdx ++;
|
|
if (CLstOld == NULL)
|
|
p->ChipListAll = CLst;
|
|
else
|
|
CLstOld->next = CLst;
|
|
|
|
CLst->CAud = CAA;
|
|
CLst->COpts = (CHIP_OPTS*)&p->ChipOpts[CurCSet] + CurChip;
|
|
CLstOld = CLst;
|
|
}
|
|
}
|
|
}
|
|
if (CLstOld != NULL)
|
|
CLstOld->next = NULL;
|
|
|
|
/*// Go through the chip list and copy all chips to a new list, except for a few
|
|
// selected ones.
|
|
CLstOld = NULL;
|
|
CurLst = ChipListAll;
|
|
while(CurLst != NULL)
|
|
{
|
|
// don't emulate the RF5Cxx chips when paused+emulated
|
|
if (CurLst->CAud->ChipType != 0x05 && CurLst->CAud->ChipType != 0x10)
|
|
{
|
|
CLst = &p->ChipListBuffer[CurBufIdx];
|
|
CurBufIdx ++;
|
|
if (CLstOld == NULL)
|
|
p->ChipListPause = CLst;
|
|
else
|
|
CLstOld->next = CLst;
|
|
|
|
*CLst = *CurLst;
|
|
CLstOld = CLst;
|
|
}
|
|
CurLst = CurLst->next;
|
|
}
|
|
if (CLstOld != NULL)
|
|
CLstOld->next = NULL;*/
|
|
|
|
return;
|
|
}
|
|
|
|
static void SetupResampler(VGM_PLAYER* p, CAUD_ATTR* CAA)
|
|
{
|
|
if (! CAA->SmpRate)
|
|
{
|
|
CAA->Resampler = 0x00;
|
|
return;
|
|
}
|
|
|
|
CAA->TargetSmpRate = p->SampleRate;
|
|
|
|
CAA->Resampler = resampler_create();
|
|
|
|
return;
|
|
}
|
|
|
|
static void ChangeChipSampleRate(void* DataPtr, UINT32 NewSmplRate)
|
|
{
|
|
CAUD_ATTR* CAA = (CAUD_ATTR*)DataPtr;
|
|
|
|
if (CAA->SmpRate == NewSmplRate)
|
|
return;
|
|
|
|
CAA->SmpRate = NewSmplRate;
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
INLINE INT16 Limit2Short(INT32 Value)
|
|
{
|
|
INT32 NewValue;
|
|
|
|
NewValue = Value;
|
|
if (NewValue < -0x8000)
|
|
NewValue = -0x8000;
|
|
else if (NewValue > 0x7FFF)
|
|
NewValue = 0x7FFF;
|
|
|
|
return (INT16)NewValue;
|
|
}
|
|
|
|
INLINE INT32 LimitScaleAdd(INT32 Target, INT32 Value, UINT16 Scale)
|
|
{
|
|
INT64 NewValue;
|
|
|
|
NewValue = (INT64)Value;
|
|
NewValue *= (INT64)Scale;
|
|
NewValue += (INT64)Target;
|
|
|
|
if (NewValue < -0x80000000LL)
|
|
NewValue = -0x80000000LL;
|
|
else if (NewValue > 0x7FFFFFFFLL)
|
|
NewValue = 0x7FFFFFFFLL;
|
|
|
|
return (INT32)NewValue;
|
|
}
|
|
|
|
static void null_update(void *param, stream_sample_t **outputs, int samples)
|
|
{
|
|
memset(outputs[0x00], 0x00, sizeof(stream_sample_t) * samples);
|
|
memset(outputs[0x01], 0x00, sizeof(stream_sample_t) * samples);
|
|
|
|
return;
|
|
}
|
|
|
|
static void dual_opl2_stereo(void *param, stream_sample_t **outputs, int samples)
|
|
{
|
|
struct dual_opl2_info * info = (struct dual_opl2_info *) param;
|
|
|
|
ym3812_stream_update(info->chip, outputs, samples);
|
|
|
|
// Dual-OPL with Stereo
|
|
if (info->ChipID & 0x01)
|
|
memset(outputs[0x00], 0x00, sizeof(stream_sample_t) * samples); // Mute Left Chanel
|
|
else
|
|
memset(outputs[0x01], 0x00, sizeof(stream_sample_t) * samples); // Mute Right Chanel
|
|
|
|
return;
|
|
}
|
|
|
|
static void ResampleChipStream(VGM_PLAYER* p, CA_LIST* CLst, WAVE_32BS* RetSample, UINT32 Length)
|
|
{
|
|
CAUD_ATTR* CAA;
|
|
INT32* CurBufL;
|
|
INT32* CurBufR;
|
|
INT32 SmpCnt; // must be signed, else I'm getting calculation errors
|
|
INT32 CurSmpl;
|
|
UINT32 SampleRate;
|
|
UINT32 OutPos;
|
|
sample_t ls, rs;
|
|
|
|
CAA = CLst->CAud;
|
|
if (!CAA->Resampler)
|
|
return;
|
|
|
|
CurBufL = p->StreamBufs[0x00];
|
|
CurBufR = p->StreamBufs[0x01];
|
|
|
|
SampleRate = p->SampleRate;
|
|
|
|
OutPos = 0;
|
|
|
|
// This Do-While-Loop gets and resamples the chip output of one or more chips.
|
|
// It's a loop to support the AY8910 paired with the YM2203/YM2608/YM2610.
|
|
do
|
|
{
|
|
for (OutPos = 0; OutPos < Length; OutPos++)
|
|
{
|
|
if (CAA->LastSmpRate != CAA->SmpRate)
|
|
{
|
|
resampler_set_rate(CAA->Resampler, (double)CAA->SmpRate / (double)CAA->TargetSmpRate);
|
|
CAA->LastSmpRate = CAA->SmpRate;
|
|
}
|
|
|
|
SmpCnt = resampler_get_min_fill(CAA->Resampler) / 2;
|
|
|
|
if (SmpCnt)
|
|
{
|
|
CAA->StreamUpdate(CAA->StreamUpdateParam, p->StreamBufs, SmpCnt);
|
|
for (CurSmpl = 0; CurSmpl < SmpCnt; CurSmpl++)
|
|
resampler_write_pair(CAA->Resampler, CurBufL[CurSmpl], CurBufR[CurSmpl]);
|
|
}
|
|
|
|
resampler_read_pair(CAA->Resampler, &ls, &rs);
|
|
|
|
RetSample[OutPos].Left = LimitScaleAdd(RetSample[OutPos].Left, ls, CAA->Volume);
|
|
RetSample[OutPos].Right = LimitScaleAdd(RetSample[OutPos].Right, rs, CAA->Volume);
|
|
}
|
|
|
|
CAA = CAA->Paired;
|
|
} while(CAA != NULL);
|
|
|
|
return;
|
|
}
|
|
|
|
static INT32 RecalcFadeVolume(VGM_PLAYER* p)
|
|
{
|
|
float TempSng;
|
|
|
|
if (p->FadePlay)
|
|
{
|
|
if (! p->FadeStart)
|
|
p->FadeStart = p->PlayingTime;
|
|
|
|
TempSng = (p->PlayingTime - p->FadeStart) / (float)p->SampleRate;
|
|
p->MasterVol = 1.0f - TempSng / (p->FadeTime * 0.001f);
|
|
if (p->MasterVol < 0.0f)
|
|
{
|
|
p->MasterVol = 0.0f;
|
|
//EndPlay = true;
|
|
p->VGMEnd = true;
|
|
}
|
|
p->FinalVol = p->VolumeLevelM * p->MasterVol * p->MasterVol;
|
|
}
|
|
|
|
return (INT32)(0x100 * p->FinalVol + 0.5f);
|
|
}
|
|
|
|
UINT32 FillBuffer(void *_p, WAVE_16BS* Buffer, UINT32 BufferSize)
|
|
{
|
|
UINT32 CurSmpl;
|
|
WAVE_32BS TempBuf;
|
|
INT32 CurMstVol;
|
|
UINT32 RecalcStep;
|
|
CA_LIST* CurCLst;
|
|
|
|
VGM_PLAYER* p = (VGM_PLAYER *)_p;
|
|
|
|
//memset(Buffer, 0x00, sizeof(WAVE_16BS) * BufferSize);
|
|
|
|
RecalcStep = p->FadePlay ? p->SampleRate / 44100 : 0;
|
|
CurMstVol = RecalcFadeVolume(p);
|
|
|
|
if (Buffer == NULL)
|
|
{
|
|
//for (CurSmpl = 0x00; CurSmpl < BufferSize; CurSmpl ++)
|
|
// InterpretFile(1);
|
|
InterpretFile(p, BufferSize);
|
|
|
|
if (p->FadePlay && ! p->FadeStart)
|
|
{
|
|
p->FadeStart = p->PlayingTime;
|
|
RecalcStep = p->FadePlay ? p->SampleRate / 100 : 0;
|
|
}
|
|
//if (RecalcStep && ! (CurSmpl % RecalcStep))
|
|
if (RecalcStep)
|
|
CurMstVol = RecalcFadeVolume(p);
|
|
|
|
if (p->VGMEnd)
|
|
{
|
|
p->EndPlay = true;
|
|
}
|
|
|
|
return BufferSize;
|
|
}
|
|
|
|
for (CurSmpl = 0x00; CurSmpl < BufferSize; CurSmpl ++)
|
|
{
|
|
InterpretFile(p, 1);
|
|
|
|
// Sample Structures
|
|
// 00 - SN76496
|
|
// 01 - YM2413
|
|
// 02 - YM2612
|
|
// 03 - YM2151
|
|
// 04 - SegaPCM
|
|
// 05 - RF5C68
|
|
// 06 - YM2203
|
|
// 07 - YM2608
|
|
// 08 - YM2610/YM2610B
|
|
// 09 - YM3812
|
|
// 0A - YM3526
|
|
// 0B - Y8950
|
|
// 0C - YMF262
|
|
// 0D - YMF278B
|
|
// 0E - YMF271
|
|
// 0F - YMZ280B
|
|
// 10 - RF5C164
|
|
// 11 - PWM
|
|
// 12 - AY8910
|
|
// 13 - GameBoy
|
|
// 14 - NES APU
|
|
// 15 - MultiPCM
|
|
// 16 - UPD7759
|
|
// 17 - OKIM6258
|
|
// 18 - OKIM6295
|
|
// 19 - K051649
|
|
// 1A - K054539
|
|
// 1B - HuC6280
|
|
// 1C - C140
|
|
// 1D - K053260
|
|
// 1E - Pokey
|
|
// 1F - QSound
|
|
// 20 - YMF292/SCSP
|
|
// 21 - WonderSwan
|
|
// 22 - VSU
|
|
// 23 - SAA1099
|
|
// 24 - ES5503
|
|
// 25 - ES5506
|
|
// 26 - X1-010
|
|
// 27 - C352
|
|
// 28 - GA20
|
|
TempBuf.Left = 0x00;
|
|
TempBuf.Right = 0x00;
|
|
CurCLst = p->ChipListAll;
|
|
while(CurCLst != NULL)
|
|
{
|
|
if (! CurCLst->COpts->Disabled)
|
|
{
|
|
ResampleChipStream(p, CurCLst, &TempBuf, 1);
|
|
}
|
|
CurCLst = CurCLst->next;
|
|
}
|
|
|
|
// ChipData << 9 [ChipVol] >> 5 << 8 [MstVol] >> 11 -> 9-5+8-11 = <<1
|
|
TempBuf.Left = ((TempBuf.Left >> 5) * CurMstVol) >> 11;
|
|
TempBuf.Right = ((TempBuf.Right >> 5) * CurMstVol) >> 11;
|
|
if (p->SurroundSound)
|
|
TempBuf.Right *= -1;
|
|
Buffer[CurSmpl].Left = Limit2Short(TempBuf.Left);
|
|
Buffer[CurSmpl].Right = Limit2Short(TempBuf.Right);
|
|
|
|
if (p->FadePlay && ! p->FadeStart)
|
|
{
|
|
p->FadeStart = p->PlayingTime;
|
|
RecalcStep = p->FadePlay ? p->SampleRate / 100 : 0;
|
|
}
|
|
if (RecalcStep && ! (CurSmpl % RecalcStep))
|
|
CurMstVol = RecalcFadeVolume(p);
|
|
|
|
if (p->VGMEnd)
|
|
{
|
|
if (! p->EndPlay)
|
|
{
|
|
p->EndPlay = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return CurSmpl;
|
|
}
|