mirror of
https://github.com/libretro/Mesen.git
synced 2025-01-07 09:20:57 +00:00
293 lines
6.9 KiB
C++
293 lines
6.9 KiB
C++
#include "stdafx.h"
|
|
|
|
#include "AutoRomTest.h"
|
|
#include "Console.h"
|
|
#include "EmulationSettings.h"
|
|
#include "MessageManager.h"
|
|
#include "Debugger.h"
|
|
#include "../Utilities/FolderUtilities.h"
|
|
#include "../Utilities/md5.h"
|
|
#include "../Utilities/ZipWriter.h"
|
|
#include "../Utilities/ZipReader.h"
|
|
|
|
AutoRomTest::AutoRomTest()
|
|
{
|
|
Reset();
|
|
|
|
MessageManager::RegisterNotificationListener(this);
|
|
}
|
|
|
|
AutoRomTest::~AutoRomTest()
|
|
{
|
|
Reset();
|
|
|
|
MessageManager::UnregisterNotificationListener(this);
|
|
}
|
|
|
|
void AutoRomTest::SaveFrame(uint16_t* ppuFrameBuffer)
|
|
{
|
|
uint8_t md5Hash[16];
|
|
GetMd5Sum(md5Hash, ppuFrameBuffer, PPU::PixelCount * sizeof(uint16_t));
|
|
|
|
if(memcmp(_previousHash, md5Hash, 16) == 0 && _currentCount < 255) {
|
|
_currentCount++;
|
|
} else {
|
|
uint8_t* hash = new uint8_t[16];
|
|
memcpy(hash, md5Hash, 16);
|
|
_screenshotHashes.push_back(hash);
|
|
if(_currentCount > 0) {
|
|
_repetitionCount.push_back(_currentCount);
|
|
}
|
|
_currentCount = 1;
|
|
|
|
memcpy(_previousHash, md5Hash, 16);
|
|
|
|
_signal.Signal();
|
|
}
|
|
}
|
|
|
|
void AutoRomTest::ValidateFrame(uint16_t* ppuFrameBuffer)
|
|
{
|
|
uint8_t md5Hash[16];
|
|
GetMd5Sum(md5Hash, ppuFrameBuffer, PPU::PixelCount * sizeof(uint16_t));
|
|
|
|
if(_currentCount == 0) {
|
|
_currentCount = _repetitionCount.front();
|
|
_repetitionCount.pop_front();
|
|
_screenshotHashes.pop_front();
|
|
}
|
|
_currentCount--;
|
|
|
|
if(memcmp(_screenshotHashes.front(), md5Hash, 16) != 0) {
|
|
_badFrameCount++;
|
|
Debugger::BreakIfDebugging();
|
|
}
|
|
|
|
if (_currentCount == 0 && _repetitionCount.empty()) {
|
|
//End of test
|
|
_runningTest = false;
|
|
_signal.Signal();
|
|
}
|
|
}
|
|
|
|
void AutoRomTest::ProcessNotification(ConsoleNotificationType type, void* parameter)
|
|
{
|
|
switch(type) {
|
|
case ConsoleNotificationType::PpuFrameDone:
|
|
if(_recording) {
|
|
SaveFrame((uint16_t*)parameter);
|
|
} else if(_runningTest) {
|
|
ValidateFrame((uint16_t*)parameter);
|
|
}
|
|
break;
|
|
case ConsoleNotificationType::MovieEnded:
|
|
if(_recordingFromMovie) {
|
|
Save();
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void AutoRomTest::Reset()
|
|
{
|
|
memset(_previousHash, 0xFF, 16);
|
|
|
|
_currentCount = 0;
|
|
_repetitionCount.clear();
|
|
|
|
for(uint8_t* hash : _screenshotHashes) {
|
|
delete[] hash;
|
|
}
|
|
_screenshotHashes.clear();
|
|
|
|
_runningTest = false;
|
|
_recording = false;
|
|
_badFrameCount = 0;
|
|
_recordingFromMovie = false;
|
|
}
|
|
|
|
void AutoRomTest::Record(string filename, bool reset)
|
|
{
|
|
_filename = filename;
|
|
|
|
string mrtFilename = FolderUtilities::CombinePath(FolderUtilities::GetFolderName(filename), FolderUtilities::GetFilename(filename, false) + ".mrt");
|
|
_file.open(mrtFilename, ios::out | ios::binary);
|
|
|
|
if(_file) {
|
|
Console::Pause();
|
|
Reset();
|
|
|
|
_recording = true;
|
|
|
|
//Start recording movie alongside with screenshots
|
|
Movie::Record(FolderUtilities::CombinePath(FolderUtilities::GetFolderName(filename), FolderUtilities::GetFilename(filename, false) + ".mmo"), reset);
|
|
|
|
Console::Resume();
|
|
}
|
|
}
|
|
|
|
void AutoRomTest::RecordFromMovie(string testFilename, stringstream &movieStream, bool autoLoadRom)
|
|
{
|
|
_filename = testFilename;
|
|
|
|
string mrtFilename = FolderUtilities::CombinePath(FolderUtilities::GetFolderName(testFilename), FolderUtilities::GetFilename(testFilename, false) + ".mrt");
|
|
_file.open(mrtFilename, ios::out | ios::binary);
|
|
|
|
if(_file) {
|
|
Console::Pause();
|
|
Reset();
|
|
|
|
_recording = true;
|
|
|
|
//Start playing movie
|
|
Movie::Play(movieStream, autoLoadRom);
|
|
movieStream.seekg(0, ios::beg);
|
|
_movieStream << movieStream.rdbuf();
|
|
|
|
_recordingFromMovie = true;
|
|
|
|
Console::Resume();
|
|
}
|
|
}
|
|
|
|
void AutoRomTest::RecordFromMovie(string testFilename, string movieFilename)
|
|
{
|
|
stringstream ss;
|
|
ifstream file(movieFilename, ios::in | ios::binary);
|
|
if(file) {
|
|
ss << file.rdbuf();
|
|
file.close();
|
|
RecordFromMovie(testFilename, ss, true);
|
|
}
|
|
}
|
|
|
|
void AutoRomTest::RecordFromTest(string newTestFilename, string existingTestFilename)
|
|
{
|
|
ZipReader zipReader;
|
|
zipReader.LoadZipArchive(existingTestFilename);
|
|
std::stringstream testMovie = zipReader.ExtractFile("TestMovie.mmo");
|
|
std::stringstream testRom = zipReader.ExtractFile("TestRom.nes");
|
|
|
|
if(testMovie && testRom) {
|
|
Console::Pause();
|
|
Console::LoadROM("TestRom", &testRom);
|
|
testRom.seekg(0, ios::beg);
|
|
_romStream << testRom.rdbuf();
|
|
|
|
RecordFromMovie(newTestFilename, testMovie, false);
|
|
Console::Resume();
|
|
}
|
|
}
|
|
|
|
int32_t AutoRomTest::Run(string filename)
|
|
{
|
|
ZipReader zipReader;
|
|
zipReader.LoadZipArchive(filename);
|
|
std::stringstream testData = zipReader.ExtractFile("TestData.mrt");
|
|
std::stringstream testMovie = zipReader.ExtractFile("TestMovie.mmo");
|
|
std::stringstream testRom = zipReader.ExtractFile("TestRom.nes");
|
|
|
|
if(testData) {
|
|
char header[3];
|
|
testData.read((char*)&header, 3);
|
|
if(memcmp((char*)&header, "MRT", 3) != 0) {
|
|
//Invalid test file
|
|
return false;
|
|
}
|
|
|
|
EmulationSettings::SetEmulationSpeed(0);
|
|
EmulationSettings::SetMasterVolume(0);
|
|
|
|
Console::Pause();
|
|
|
|
Reset();
|
|
|
|
uint32_t hashCount;
|
|
testData.read((char*)&hashCount, sizeof(uint32_t));
|
|
|
|
for(uint32_t i = 0; i < hashCount; i++) {
|
|
uint8_t repeatCount = 0;
|
|
testData.read((char*)&repeatCount, sizeof(uint8_t));
|
|
_repetitionCount.push_back(repeatCount);
|
|
|
|
uint8_t* screenshotHash = new uint8_t[16];
|
|
testData.read((char*)screenshotHash, 16);
|
|
_screenshotHashes.push_back(screenshotHash);
|
|
}
|
|
|
|
_currentCount = _repetitionCount.front();
|
|
_repetitionCount.pop_front();
|
|
|
|
_runningTest = true;
|
|
|
|
//Start playing movie
|
|
Console::LoadROM("TestRom", &testRom);
|
|
Movie::Play(testMovie, false);
|
|
|
|
Console::Resume();
|
|
|
|
_signal.Wait();
|
|
_runningTest = false;
|
|
|
|
Console::GetInstance()->Stop();
|
|
|
|
|
|
EmulationSettings::SetEmulationSpeed(100);
|
|
EmulationSettings::SetMasterVolume(1.0);
|
|
|
|
return _badFrameCount;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
void AutoRomTest::Stop()
|
|
{
|
|
if(_recording) {
|
|
Save();
|
|
}
|
|
Reset();
|
|
}
|
|
|
|
void AutoRomTest::Save()
|
|
{
|
|
//Wait until the next frame is captured to end the recording
|
|
_signal.Wait();
|
|
_repetitionCount.push_back(_currentCount);
|
|
_recording = false;
|
|
|
|
//Stop playing/recording the movie
|
|
Movie::Stop();
|
|
|
|
_file.write("MRT", 3);
|
|
|
|
uint32_t hashCount = (uint32_t)_screenshotHashes.size();
|
|
_file.write((char*)&hashCount, sizeof(uint32_t));
|
|
|
|
for(uint32_t i = 0; i < hashCount; i++) {
|
|
_file.write((char*)&_repetitionCount[i], sizeof(uint8_t));
|
|
_file.write((char*)&_screenshotHashes[i][0], 16);
|
|
}
|
|
|
|
_file.close();
|
|
|
|
ZipWriter writer(_filename);
|
|
|
|
string mrtFilename = FolderUtilities::CombinePath(FolderUtilities::GetFolderName(_filename), FolderUtilities::GetFilename(_filename, false) + ".mrt");
|
|
writer.AddFile(mrtFilename, "TestData.mrt");
|
|
std::remove(mrtFilename.c_str());
|
|
|
|
if(_recordingFromMovie) {
|
|
writer.AddFile(_movieStream, "TestMovie.mmo");
|
|
writer.AddFile(_romStream, "TestRom.nes");
|
|
} else {
|
|
string mmoFilename = FolderUtilities::CombinePath(FolderUtilities::GetFolderName(_filename), FolderUtilities::GetFilename(_filename, false) + ".mmo");
|
|
writer.AddFile(mmoFilename, "TestMovie.mmo");
|
|
std::remove(mmoFilename.c_str());
|
|
|
|
writer.AddFile(Console::GetROMPath(), "TestRom.nes");
|
|
}
|
|
|
|
|
|
MessageManager::DisplayMessage("Test", "Test file saved to: " + FolderUtilities::GetFilename(_filename, true));
|
|
} |