2016-06-25 20:46:54 -04:00
|
|
|
#pragma once
|
|
|
|
#include "stdafx.h"
|
2016-12-12 20:33:48 -05:00
|
|
|
#include <cstring>
|
2016-06-25 20:46:54 -04:00
|
|
|
#include <algorithm>
|
2016-12-12 20:33:48 -05:00
|
|
|
#include "RomData.h"
|
2016-06-25 20:46:54 -04:00
|
|
|
#include "NsfLoader.h"
|
|
|
|
|
|
|
|
class NsfeLoader : public NsfLoader
|
|
|
|
{
|
|
|
|
private:
|
2016-12-12 20:33:48 -05:00
|
|
|
void CopyString(char* dest, string source, int maxLength)
|
|
|
|
{
|
|
|
|
memset(dest, 0, maxLength);
|
|
|
|
memcpy(dest, source.c_str(), std::max(source.size(), (size_t)maxLength - 1));
|
|
|
|
}
|
|
|
|
|
2016-06-25 20:46:54 -04:00
|
|
|
void Read(uint8_t* &data, uint8_t& dest)
|
|
|
|
{
|
|
|
|
dest = data[0];
|
|
|
|
data++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Read(uint8_t* &data, uint16_t& dest)
|
|
|
|
{
|
|
|
|
dest = data[0] | (data[1] << 8);
|
|
|
|
data += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Read(uint8_t* &data, uint32_t& dest)
|
|
|
|
{
|
|
|
|
dest = data[0] | (data[1] << 8) | (data[2] << 16) | (data[3] << 24);
|
|
|
|
data += 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Read(uint8_t* &data, int32_t& dest)
|
|
|
|
{
|
|
|
|
dest = data[0] | (data[1] << 8) | (data[2] << 16) | (data[3] << 24);
|
|
|
|
data += 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Read(uint8_t* &data, char* dest, size_t len)
|
|
|
|
{
|
|
|
|
memcpy(dest, data, len);
|
|
|
|
data += len;
|
|
|
|
}
|
|
|
|
|
|
|
|
vector<string> ReadStrings(uint8_t* &data, uint8_t* chunkEnd)
|
|
|
|
{
|
|
|
|
vector<string> strings;
|
|
|
|
stringstream ss;
|
|
|
|
while(data < chunkEnd) {
|
|
|
|
if(data[0] == 0) {
|
|
|
|
//end of string
|
|
|
|
strings.push_back(ss.str());
|
|
|
|
ss = stringstream();
|
|
|
|
} else {
|
|
|
|
ss << (char)data[0];
|
|
|
|
}
|
|
|
|
data++;
|
|
|
|
}
|
|
|
|
|
|
|
|
//truncate all strings to 255 characters + null
|
2016-06-26 16:31:29 -04:00
|
|
|
for(size_t i = 0; i < strings.size(); i++) {
|
2016-06-25 20:46:54 -04:00
|
|
|
strings[i] = strings[i].substr(0, std::min((int)strings[i].size(), 255));
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings;
|
|
|
|
}
|
|
|
|
|
|
|
|
string ReadFourCC(uint8_t* &data)
|
|
|
|
{
|
|
|
|
stringstream ss;
|
|
|
|
for(int i = 0; i < 4; i++) {
|
|
|
|
ss << (char)data[i];
|
|
|
|
}
|
|
|
|
data += 4;
|
|
|
|
return ss.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ReadChunk(uint8_t* &data, uint8_t* dataEnd, RomData& romData)
|
|
|
|
{
|
2016-08-14 20:12:50 -04:00
|
|
|
if(data + 4 > dataEnd) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-12-11 11:18:37 -05:00
|
|
|
NsfHeader& header = romData.NsfInfo;
|
2016-06-25 20:46:54 -04:00
|
|
|
|
|
|
|
uint32_t length;
|
|
|
|
Read(data, length);
|
|
|
|
|
|
|
|
uint8_t* chunkEnd = data + 4 + length;
|
|
|
|
|
|
|
|
if(chunkEnd > dataEnd) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
string fourCC = ReadFourCC(data);
|
|
|
|
if(fourCC.compare("INFO") == 0) {
|
|
|
|
Read(data, header.LoadAddress);
|
|
|
|
Read(data, header.InitAddress);
|
|
|
|
Read(data, header.PlayAddress);
|
|
|
|
Read(data, header.Flags);
|
|
|
|
Read(data, header.SoundChips);
|
|
|
|
Read(data, header.TotalSongs);
|
|
|
|
Read(data, header.StartingSong);
|
|
|
|
|
|
|
|
header.PlaySpeedNtsc = 16639;
|
|
|
|
header.PlaySpeedPal = 19997;
|
|
|
|
|
|
|
|
//Adjust to match NSF spec
|
|
|
|
header.StartingSong++;
|
|
|
|
} else if(fourCC.compare("DATA") == 0) {
|
|
|
|
//Pad start of file to make the first block start at a multiple of 4k
|
|
|
|
romData.PrgRom.insert(romData.PrgRom.end(), header.LoadAddress % 4096, 0);
|
|
|
|
|
|
|
|
romData.PrgRom.insert(romData.PrgRom.end(), (uint8_t*)data, data+length);
|
|
|
|
|
|
|
|
//Pad out the last block to be a multiple of 4k
|
|
|
|
if(romData.PrgRom.size() % 4096 != 0) {
|
|
|
|
romData.PrgRom.insert(romData.PrgRom.end(), 4096 - (romData.PrgRom.size() % 4096), 0);
|
|
|
|
}
|
|
|
|
} else if(fourCC.compare("NEND") == 0) {
|
|
|
|
//End of file
|
|
|
|
romData.Error = false;
|
|
|
|
return false;
|
|
|
|
} else if(fourCC.compare("BANK") == 0) {
|
|
|
|
memset(header.BankSetup, 0, sizeof(header.BankSetup));
|
|
|
|
Read(data, (char*)header.BankSetup, std::min(8, (int)length));
|
|
|
|
} else if(fourCC.compare("plst") == 0) {
|
|
|
|
//not supported
|
|
|
|
} else if(fourCC.compare("time") == 0) {
|
|
|
|
int i = 0;
|
|
|
|
while(data < chunkEnd) {
|
|
|
|
Read(data, header.TrackLength[i]);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
} else if(fourCC.compare("fade") == 0) {
|
|
|
|
int i = 0;
|
|
|
|
while(data < chunkEnd) {
|
|
|
|
Read(data, header.TrackFade[i]);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
} else if(fourCC.compare("tlbl") == 0) {
|
|
|
|
vector<string> trackNames = ReadStrings(data, chunkEnd);
|
|
|
|
stringstream ss;
|
|
|
|
for(string &trackName : trackNames) {
|
|
|
|
ss << trackName;
|
|
|
|
ss << "[!|!]";
|
|
|
|
}
|
2016-12-12 20:33:48 -05:00
|
|
|
CopyString(header.TrackName, ss.str(), 20000);
|
2016-06-25 20:46:54 -04:00
|
|
|
} else if(fourCC.compare("auth") == 0) {
|
|
|
|
vector<string> infoStrings = ReadStrings(data, chunkEnd);
|
|
|
|
|
|
|
|
if(infoStrings.size() > 0) {
|
2016-12-12 20:33:48 -05:00
|
|
|
CopyString(header.SongName, infoStrings[0], 256);
|
2016-06-25 20:46:54 -04:00
|
|
|
if(infoStrings.size() > 1) {
|
2016-12-12 20:33:48 -05:00
|
|
|
CopyString(header.ArtistName, infoStrings[1], 256);
|
2016-06-25 20:46:54 -04:00
|
|
|
if(infoStrings.size() > 2) {
|
2016-12-12 20:33:48 -05:00
|
|
|
CopyString(header.CopyrightHolder, infoStrings[2], 256);
|
2016-06-25 20:46:54 -04:00
|
|
|
if(infoStrings.size() > 3) {
|
2016-12-12 20:33:48 -05:00
|
|
|
CopyString(header.RipperName, infoStrings[3], 256);
|
2016-06-25 20:46:54 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if(fourCC.compare("text") == 0) {
|
|
|
|
//not supported
|
|
|
|
} else {
|
|
|
|
if(fourCC[0] >= 'A' && fourCC[0] <= 'Z') {
|
|
|
|
//unknown required block, can't read file
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
data = chunkEnd;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
RomData LoadRom(vector<uint8_t>& romFile)
|
|
|
|
{
|
|
|
|
RomData romData;
|
2016-12-11 11:18:37 -05:00
|
|
|
NsfHeader &header = romData.NsfInfo;
|
2016-06-25 20:46:54 -04:00
|
|
|
|
|
|
|
InitHeader(header);
|
|
|
|
|
2017-05-06 15:27:48 -04:00
|
|
|
romData.Format = RomFormat::Nsf;
|
|
|
|
|
2016-06-25 20:46:54 -04:00
|
|
|
uint8_t* data = romFile.data() + 4;
|
2016-08-14 20:12:50 -04:00
|
|
|
uint8_t* endOfData = romFile.data() + romFile.size();
|
2016-06-25 20:46:54 -04:00
|
|
|
|
|
|
|
memset(header.SongName, 0, sizeof(header.SongName));
|
|
|
|
memset(header.ArtistName, 0, sizeof(header.ArtistName));
|
|
|
|
memset(header.CopyrightHolder, 0, sizeof(header.CopyrightHolder));
|
|
|
|
|
|
|
|
//Will be set to false when we read NEND block
|
|
|
|
romData.Error = true;
|
2016-08-14 20:12:50 -04:00
|
|
|
while(ReadChunk(data, endOfData, romData)) {
|
2016-06-25 20:46:54 -04:00
|
|
|
//Read all chunks
|
|
|
|
}
|
|
|
|
|
|
|
|
InitializeFromHeader(romData);
|
|
|
|
|
|
|
|
return romData;
|
|
|
|
}
|
|
|
|
};
|