Add E2E and ZSCustomOverworld test suites for comprehensive testing
- Introduced new E2E test suite for comprehensive ROM testing, validating the complete ROM editing workflow. - Added ZSCustomOverworld test suite to validate version upgrades and data integrity. - Updated `EditorManager` to register the new test suites. - Enhanced CMake configuration to include the new test files. - Updated README to reflect the new testing capabilities and best practices for AI agent testing.
This commit is contained in:
@@ -30,6 +30,8 @@
|
|||||||
#include "app/test/test_manager.h"
|
#include "app/test/test_manager.h"
|
||||||
#include "app/test/integrated_test_suite.h"
|
#include "app/test/integrated_test_suite.h"
|
||||||
#include "app/test/rom_dependent_test_suite.h"
|
#include "app/test/rom_dependent_test_suite.h"
|
||||||
|
#include "app/test/e2e_test_suite.h"
|
||||||
|
#include "app/test/zscustomoverworld_test_suite.h"
|
||||||
#ifdef YAZE_ENABLE_GTEST
|
#ifdef YAZE_ENABLE_GTEST
|
||||||
#include "app/test/unit_test_suite.h"
|
#include "app/test/unit_test_suite.h"
|
||||||
#endif
|
#endif
|
||||||
@@ -160,6 +162,10 @@ void EditorManager::InitializeTestSuites() {
|
|||||||
test_manager.RegisterTestSuite(std::make_unique<test::UITestSuite>());
|
test_manager.RegisterTestSuite(std::make_unique<test::UITestSuite>());
|
||||||
test_manager.RegisterTestSuite(std::make_unique<test::RomDependentTestSuite>());
|
test_manager.RegisterTestSuite(std::make_unique<test::RomDependentTestSuite>());
|
||||||
|
|
||||||
|
// Register new E2E and ZSCustomOverworld test suites
|
||||||
|
test_manager.RegisterTestSuite(std::make_unique<test::E2ETestSuite>());
|
||||||
|
test_manager.RegisterTestSuite(std::make_unique<test::ZSCustomOverworldTestSuite>());
|
||||||
|
|
||||||
// Register Google Test suite if available
|
// Register Google Test suite if available
|
||||||
#ifdef YAZE_ENABLE_GTEST
|
#ifdef YAZE_ENABLE_GTEST
|
||||||
test_manager.RegisterTestSuite(std::make_unique<test::UnitTestSuite>());
|
test_manager.RegisterTestSuite(std::make_unique<test::UnitTestSuite>());
|
||||||
|
|||||||
420
src/app/test/e2e_test_suite.h
Normal file
420
src/app/test/e2e_test_suite.h
Normal file
@@ -0,0 +1,420 @@
|
|||||||
|
#ifndef YAZE_APP_TEST_E2E_TEST_SUITE_H
|
||||||
|
#define YAZE_APP_TEST_E2E_TEST_SUITE_H
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
|
#include <filesystem>
|
||||||
|
|
||||||
|
#include "absl/strings/str_format.h"
|
||||||
|
#include "app/test/test_manager.h"
|
||||||
|
#include "app/rom.h"
|
||||||
|
#include "app/transaction.h"
|
||||||
|
#include "app/gui/icons.h"
|
||||||
|
|
||||||
|
namespace yaze {
|
||||||
|
namespace test {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief End-to-End test suite for comprehensive ROM testing
|
||||||
|
*
|
||||||
|
* This test suite provides comprehensive E2E testing capabilities including:
|
||||||
|
* - ROM loading/saving validation
|
||||||
|
* - Data integrity testing
|
||||||
|
* - Transaction system testing
|
||||||
|
* - Large-scale editing validation
|
||||||
|
*/
|
||||||
|
class E2ETestSuite : public TestSuite {
|
||||||
|
public:
|
||||||
|
E2ETestSuite() = default;
|
||||||
|
~E2ETestSuite() override = default;
|
||||||
|
|
||||||
|
std::string GetName() const override { return "End-to-End ROM Tests"; }
|
||||||
|
TestCategory GetCategory() const override { return TestCategory::kIntegration; }
|
||||||
|
|
||||||
|
absl::Status RunTests(TestResults& results) override {
|
||||||
|
Rom* current_rom = TestManager::Get().GetCurrentRom();
|
||||||
|
|
||||||
|
// Check ROM availability
|
||||||
|
if (!current_rom || !current_rom->is_loaded()) {
|
||||||
|
AddSkippedTest(results, "ROM_Availability_Check", "No ROM loaded");
|
||||||
|
return absl::OkStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Run E2E tests
|
||||||
|
if (test_rom_load_save_) {
|
||||||
|
RunRomLoadSaveTest(results, current_rom);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (test_data_integrity_) {
|
||||||
|
RunDataIntegrityTest(results, current_rom);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (test_transaction_system_) {
|
||||||
|
RunTransactionSystemTest(results, current_rom);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (test_large_scale_editing_) {
|
||||||
|
RunLargeScaleEditingTest(results, current_rom);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (test_corruption_detection_) {
|
||||||
|
RunCorruptionDetectionTest(results, current_rom);
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
void DrawConfiguration() override {
|
||||||
|
Rom* current_rom = TestManager::Get().GetCurrentRom();
|
||||||
|
|
||||||
|
ImGui::Text("%s E2E Test Configuration", ICON_MD_VERIFIED_USER);
|
||||||
|
|
||||||
|
if (current_rom && current_rom->is_loaded()) {
|
||||||
|
ImGui::TextColored(ImVec4(0.0F, 1.0F, 0.0F, 1.0F),
|
||||||
|
"%s Current ROM: %s", ICON_MD_CHECK_CIRCLE, current_rom->title().c_str());
|
||||||
|
ImGui::Text("Size: %.2F MB", current_rom->size() / 1048576.0F);
|
||||||
|
} else {
|
||||||
|
ImGui::TextColored(ImVec4(1.0F, 0.5F, 0.0F, 1.0F),
|
||||||
|
"%s No ROM currently loaded", ICON_MD_WARNING);
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::Separator();
|
||||||
|
ImGui::Checkbox("Test ROM load/save", &test_rom_load_save_);
|
||||||
|
ImGui::Checkbox("Test data integrity", &test_data_integrity_);
|
||||||
|
ImGui::Checkbox("Test transaction system", &test_transaction_system_);
|
||||||
|
ImGui::Checkbox("Test large-scale editing", &test_large_scale_editing_);
|
||||||
|
ImGui::Checkbox("Test corruption detection", &test_corruption_detection_);
|
||||||
|
|
||||||
|
if (test_large_scale_editing_) {
|
||||||
|
ImGui::Indent();
|
||||||
|
ImGui::InputInt("Number of edits", &num_edits_);
|
||||||
|
if (num_edits_ < 1) num_edits_ = 1;
|
||||||
|
if (num_edits_ > 100) num_edits_ = 100;
|
||||||
|
ImGui::Unindent();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
void AddSkippedTest(TestResults& results, const std::string& test_name, const std::string& reason) {
|
||||||
|
TestResult result;
|
||||||
|
result.name = test_name;
|
||||||
|
result.suite_name = GetName();
|
||||||
|
result.category = GetCategory();
|
||||||
|
result.status = TestStatus::kSkipped;
|
||||||
|
result.error_message = reason;
|
||||||
|
result.duration = std::chrono::milliseconds{0};
|
||||||
|
result.timestamp = std::chrono::steady_clock::now();
|
||||||
|
results.AddResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RunRomLoadSaveTest(TestResults& results, Rom* rom) {
|
||||||
|
auto start_time = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
TestResult result;
|
||||||
|
result.name = "ROM_Load_Save_Test";
|
||||||
|
result.suite_name = GetName();
|
||||||
|
result.category = GetCategory();
|
||||||
|
result.timestamp = start_time;
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto& test_manager = TestManager::Get();
|
||||||
|
|
||||||
|
auto test_status = test_manager.TestRomWithCopy(rom, [&](Rom* test_rom) -> absl::Status {
|
||||||
|
// Test basic ROM operations
|
||||||
|
if (test_rom->size() != rom->size()) {
|
||||||
|
return absl::InternalError("ROM copy size mismatch");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test save and reload
|
||||||
|
std::string test_filename = test_manager.GenerateTestRomFilename("e2e_test");
|
||||||
|
auto save_status = test_rom->SaveToFile(Rom::SaveSettings{.filename = test_filename});
|
||||||
|
if (!save_status.ok()) {
|
||||||
|
return save_status;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Clean up test file
|
||||||
|
std::filesystem::remove(test_filename);
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
});
|
||||||
|
|
||||||
|
if (test_status.ok()) {
|
||||||
|
result.status = TestStatus::kPassed;
|
||||||
|
result.error_message = "ROM load/save test completed successfully";
|
||||||
|
} else {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "ROM load/save test failed: " + test_status.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "ROM load/save test exception: " + std::string(e.what());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto end_time = std::chrono::steady_clock::now();
|
||||||
|
result.duration = std::chrono::duration_cast<std::chrono::milliseconds>(
|
||||||
|
end_time - start_time);
|
||||||
|
|
||||||
|
results.AddResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RunDataIntegrityTest(TestResults& results, Rom* rom) {
|
||||||
|
auto start_time = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
TestResult result;
|
||||||
|
result.name = "Data_Integrity_Test";
|
||||||
|
result.suite_name = GetName();
|
||||||
|
result.category = GetCategory();
|
||||||
|
result.timestamp = start_time;
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto& test_manager = TestManager::Get();
|
||||||
|
|
||||||
|
auto test_status = test_manager.TestRomWithCopy(rom, [&](Rom* test_rom) -> absl::Status {
|
||||||
|
// Test data integrity by comparing key areas
|
||||||
|
std::vector<uint32_t> test_addresses = {0x7FC0, 0x8000, 0x10000, 0x20000};
|
||||||
|
|
||||||
|
for (uint32_t addr : test_addresses) {
|
||||||
|
auto original_byte = rom->ReadByte(addr);
|
||||||
|
auto copy_byte = test_rom->ReadByte(addr);
|
||||||
|
|
||||||
|
if (!original_byte.ok() || !copy_byte.ok()) {
|
||||||
|
return absl::InternalError("Failed to read ROM data for comparison");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*original_byte != *copy_byte) {
|
||||||
|
return absl::InternalError(absl::StrFormat(
|
||||||
|
"Data integrity check failed at address 0x%X: original=0x%02X, copy=0x%02X",
|
||||||
|
addr, *original_byte, *copy_byte));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
});
|
||||||
|
|
||||||
|
if (test_status.ok()) {
|
||||||
|
result.status = TestStatus::kPassed;
|
||||||
|
result.error_message = "Data integrity test passed - all checked addresses match";
|
||||||
|
} else {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Data integrity test failed: " + test_status.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Data integrity test exception: " + std::string(e.what());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto end_time = std::chrono::steady_clock::now();
|
||||||
|
result.duration = std::chrono::duration_cast<std::chrono::milliseconds>(
|
||||||
|
end_time - start_time);
|
||||||
|
|
||||||
|
results.AddResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RunTransactionSystemTest(TestResults& results, Rom* rom) {
|
||||||
|
auto start_time = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
TestResult result;
|
||||||
|
result.name = "Transaction_System_Test";
|
||||||
|
result.suite_name = GetName();
|
||||||
|
result.category = GetCategory();
|
||||||
|
result.timestamp = start_time;
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto& test_manager = TestManager::Get();
|
||||||
|
|
||||||
|
auto test_status = test_manager.TestRomWithCopy(rom, [&](Rom* test_rom) -> absl::Status {
|
||||||
|
// Test transaction system
|
||||||
|
Transaction transaction(*test_rom);
|
||||||
|
|
||||||
|
// Store original values
|
||||||
|
auto original_byte1 = test_rom->ReadByte(0x1000);
|
||||||
|
auto original_byte2 = test_rom->ReadByte(0x2000);
|
||||||
|
auto original_word = test_rom->ReadWord(0x3000);
|
||||||
|
|
||||||
|
if (!original_byte1.ok() || !original_byte2.ok() || !original_word.ok()) {
|
||||||
|
return absl::InternalError("Failed to read original ROM data");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make changes in transaction
|
||||||
|
transaction.WriteByte(0x1000, 0xAA)
|
||||||
|
.WriteByte(0x2000, 0xBB)
|
||||||
|
.WriteWord(0x3000, 0xCCDD);
|
||||||
|
|
||||||
|
// Commit transaction
|
||||||
|
RETURN_IF_ERROR(transaction.Commit());
|
||||||
|
|
||||||
|
// Verify changes
|
||||||
|
auto new_byte1 = test_rom->ReadByte(0x1000);
|
||||||
|
auto new_byte2 = test_rom->ReadByte(0x2000);
|
||||||
|
auto new_word = test_rom->ReadWord(0x3000);
|
||||||
|
|
||||||
|
if (!new_byte1.ok() || !new_byte2.ok() || !new_word.ok()) {
|
||||||
|
return absl::InternalError("Failed to read modified ROM data");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*new_byte1 != 0xAA || *new_byte2 != 0xBB || *new_word != 0xCCDD) {
|
||||||
|
return absl::InternalError("Transaction changes not applied correctly");
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
});
|
||||||
|
|
||||||
|
if (test_status.ok()) {
|
||||||
|
result.status = TestStatus::kPassed;
|
||||||
|
result.error_message = "Transaction system test completed successfully";
|
||||||
|
} else {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Transaction system test failed: " + test_status.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Transaction system test exception: " + std::string(e.what());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto end_time = std::chrono::steady_clock::now();
|
||||||
|
result.duration = std::chrono::duration_cast<std::chrono::milliseconds>(
|
||||||
|
end_time - start_time);
|
||||||
|
|
||||||
|
results.AddResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RunLargeScaleEditingTest(TestResults& results, Rom* rom) {
|
||||||
|
auto start_time = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
TestResult result;
|
||||||
|
result.name = "Large_Scale_Editing_Test";
|
||||||
|
result.suite_name = GetName();
|
||||||
|
result.category = GetCategory();
|
||||||
|
result.timestamp = start_time;
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto& test_manager = TestManager::Get();
|
||||||
|
|
||||||
|
auto test_status = test_manager.TestRomWithCopy(rom, [&](Rom* test_rom) -> absl::Status {
|
||||||
|
// Test large-scale editing
|
||||||
|
for (int i = 0; i < num_edits_; i++) {
|
||||||
|
uint32_t addr = 0x1000 + (i * 4);
|
||||||
|
uint8_t value = i % 256;
|
||||||
|
|
||||||
|
RETURN_IF_ERROR(test_rom->WriteByte(addr, value));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify all changes
|
||||||
|
for (int i = 0; i < num_edits_; i++) {
|
||||||
|
uint32_t addr = 0x1000 + (i * 4);
|
||||||
|
uint8_t expected_value = i % 256;
|
||||||
|
|
||||||
|
auto actual_value = test_rom->ReadByte(addr);
|
||||||
|
if (!actual_value.ok()) {
|
||||||
|
return absl::InternalError(absl::StrFormat("Failed to read address 0x%X", addr));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*actual_value != expected_value) {
|
||||||
|
return absl::InternalError(absl::StrFormat(
|
||||||
|
"Value mismatch at 0x%X: expected=0x%02X, actual=0x%02X",
|
||||||
|
addr, expected_value, *actual_value));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
});
|
||||||
|
|
||||||
|
if (test_status.ok()) {
|
||||||
|
result.status = TestStatus::kPassed;
|
||||||
|
result.error_message = absl::StrFormat("Large-scale editing test passed: %d edits", num_edits_);
|
||||||
|
} else {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Large-scale editing test failed: " + test_status.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Large-scale editing test exception: " + std::string(e.what());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto end_time = std::chrono::steady_clock::now();
|
||||||
|
result.duration = std::chrono::duration_cast<std::chrono::milliseconds>(
|
||||||
|
end_time - start_time);
|
||||||
|
|
||||||
|
results.AddResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RunCorruptionDetectionTest(TestResults& results, Rom* rom) {
|
||||||
|
auto start_time = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
TestResult result;
|
||||||
|
result.name = "Corruption_Detection_Test";
|
||||||
|
result.suite_name = GetName();
|
||||||
|
result.category = GetCategory();
|
||||||
|
result.timestamp = start_time;
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto& test_manager = TestManager::Get();
|
||||||
|
|
||||||
|
auto test_status = test_manager.TestRomWithCopy(rom, [&](Rom* test_rom) -> absl::Status {
|
||||||
|
// Intentionally corrupt some data
|
||||||
|
RETURN_IF_ERROR(test_rom->WriteByte(0x1000, 0xFF));
|
||||||
|
RETURN_IF_ERROR(test_rom->WriteByte(0x2000, 0xAA));
|
||||||
|
|
||||||
|
// Verify corruption is detected
|
||||||
|
auto corrupted_byte1 = test_rom->ReadByte(0x1000);
|
||||||
|
auto corrupted_byte2 = test_rom->ReadByte(0x2000);
|
||||||
|
|
||||||
|
if (!corrupted_byte1.ok() || !corrupted_byte2.ok()) {
|
||||||
|
return absl::InternalError("Failed to read corrupted data");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*corrupted_byte1 != 0xFF || *corrupted_byte2 != 0xAA) {
|
||||||
|
return absl::InternalError("Corruption not applied correctly");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify original data is different
|
||||||
|
auto original_byte1 = rom->ReadByte(0x1000);
|
||||||
|
auto original_byte2 = rom->ReadByte(0x2000);
|
||||||
|
|
||||||
|
if (!original_byte1.ok() || !original_byte2.ok()) {
|
||||||
|
return absl::InternalError("Failed to read original data for comparison");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*corrupted_byte1 == *original_byte1 || *corrupted_byte2 == *original_byte2) {
|
||||||
|
return absl::InternalError("Corruption detection test failed - data not actually corrupted");
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
});
|
||||||
|
|
||||||
|
if (test_status.ok()) {
|
||||||
|
result.status = TestStatus::kPassed;
|
||||||
|
result.error_message = "Corruption detection test passed - corruption successfully applied and detected";
|
||||||
|
} else {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Corruption detection test failed: " + test_status.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Corruption detection test exception: " + std::string(e.what());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto end_time = std::chrono::steady_clock::now();
|
||||||
|
result.duration = std::chrono::duration_cast<std::chrono::milliseconds>(
|
||||||
|
end_time - start_time);
|
||||||
|
|
||||||
|
results.AddResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Configuration
|
||||||
|
bool test_rom_load_save_ = true;
|
||||||
|
bool test_data_integrity_ = true;
|
||||||
|
bool test_transaction_system_ = true;
|
||||||
|
bool test_large_scale_editing_ = true;
|
||||||
|
bool test_corruption_detection_ = true;
|
||||||
|
int num_edits_ = 10;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace test
|
||||||
|
} // namespace yaze
|
||||||
|
|
||||||
|
#endif // YAZE_APP_TEST_E2E_TEST_SUITE_H
|
||||||
@@ -4,6 +4,10 @@ set(YAZE_TEST_CORE_SOURCES
|
|||||||
app/test/test_manager.cc
|
app/test/test_manager.cc
|
||||||
app/test/test_manager.h
|
app/test/test_manager.h
|
||||||
app/test/unit_test_suite.h
|
app/test/unit_test_suite.h
|
||||||
|
app/test/integrated_test_suite.h
|
||||||
|
app/test/rom_dependent_test_suite.h
|
||||||
|
app/test/e2e_test_suite.h
|
||||||
|
app/test/zscustomoverworld_test_suite.h
|
||||||
)
|
)
|
||||||
|
|
||||||
# Add test sources to the main app target if testing is enabled
|
# Add test sources to the main app target if testing is enabled
|
||||||
|
|||||||
591
src/app/test/zscustomoverworld_test_suite.h
Normal file
591
src/app/test/zscustomoverworld_test_suite.h
Normal file
@@ -0,0 +1,591 @@
|
|||||||
|
#ifndef YAZE_APP_TEST_ZSCUSTOMOVERWORLD_TEST_SUITE_H
|
||||||
|
#define YAZE_APP_TEST_ZSCUSTOMOVERWORLD_TEST_SUITE_H
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
|
#include <map>
|
||||||
|
|
||||||
|
#include "absl/strings/str_format.h"
|
||||||
|
#include "app/test/test_manager.h"
|
||||||
|
#include "app/rom.h"
|
||||||
|
#include "app/gui/icons.h"
|
||||||
|
|
||||||
|
namespace yaze {
|
||||||
|
namespace test {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief ZSCustomOverworld upgrade testing suite
|
||||||
|
*
|
||||||
|
* This test suite validates ZSCustomOverworld version upgrades:
|
||||||
|
* - Vanilla -> v2 -> v3 upgrade path testing
|
||||||
|
* - Address validation for each version
|
||||||
|
* - Feature enablement/disablement testing
|
||||||
|
* - Data integrity validation during upgrades
|
||||||
|
* - Save compatibility between versions
|
||||||
|
*/
|
||||||
|
class ZSCustomOverworldTestSuite : public TestSuite {
|
||||||
|
public:
|
||||||
|
ZSCustomOverworldTestSuite() = default;
|
||||||
|
~ZSCustomOverworldTestSuite() override = default;
|
||||||
|
|
||||||
|
std::string GetName() const override { return "ZSCustomOverworld Upgrade Tests"; }
|
||||||
|
TestCategory GetCategory() const override { return TestCategory::kIntegration; }
|
||||||
|
|
||||||
|
absl::Status RunTests(TestResults& results) override {
|
||||||
|
Rom* current_rom = TestManager::Get().GetCurrentRom();
|
||||||
|
|
||||||
|
// Check ROM availability
|
||||||
|
if (!current_rom || !current_rom->is_loaded()) {
|
||||||
|
AddSkippedTest(results, "ROM_Availability_Check", "No ROM loaded");
|
||||||
|
return absl::OkStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initialize version data
|
||||||
|
InitializeVersionData();
|
||||||
|
|
||||||
|
// Run ZSCustomOverworld tests
|
||||||
|
if (test_vanilla_baseline_) {
|
||||||
|
RunVanillaBaselineTest(results, current_rom);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (test_v2_upgrade_) {
|
||||||
|
RunV2UpgradeTest(results, current_rom);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (test_v3_upgrade_) {
|
||||||
|
RunV3UpgradeTest(results, current_rom);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (test_address_validation_) {
|
||||||
|
RunAddressValidationTest(results, current_rom);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (test_feature_toggle_) {
|
||||||
|
RunFeatureToggleTest(results, current_rom);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (test_data_integrity_) {
|
||||||
|
RunDataIntegrityTest(results, current_rom);
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
void DrawConfiguration() override {
|
||||||
|
Rom* current_rom = TestManager::Get().GetCurrentRom();
|
||||||
|
|
||||||
|
ImGui::Text("%s ZSCustomOverworld Test Configuration", ICON_MD_UPGRADE);
|
||||||
|
|
||||||
|
if (current_rom && current_rom->is_loaded()) {
|
||||||
|
ImGui::TextColored(ImVec4(0.0F, 1.0F, 0.0F, 1.0F),
|
||||||
|
"%s Current ROM: %s", ICON_MD_CHECK_CIRCLE, current_rom->title().c_str());
|
||||||
|
|
||||||
|
// Check current version
|
||||||
|
auto version_byte = current_rom->ReadByte(0x140145);
|
||||||
|
if (version_byte.ok()) {
|
||||||
|
std::string version_name = "Unknown";
|
||||||
|
if (*version_byte == 0xFF) version_name = "Vanilla";
|
||||||
|
else if (*version_byte == 0x02) version_name = "v2";
|
||||||
|
else if (*version_byte == 0x03) version_name = "v3";
|
||||||
|
|
||||||
|
ImGui::Text("Current ZSCustomOverworld version: %s (0x%02X)",
|
||||||
|
version_name.c_str(), *version_byte);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ImGui::TextColored(ImVec4(1.0F, 0.5F, 0.0F, 1.0F),
|
||||||
|
"%s No ROM currently loaded", ICON_MD_WARNING);
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::Separator();
|
||||||
|
ImGui::Checkbox("Test vanilla baseline", &test_vanilla_baseline_);
|
||||||
|
ImGui::Checkbox("Test v2 upgrade", &test_v2_upgrade_);
|
||||||
|
ImGui::Checkbox("Test v3 upgrade", &test_v3_upgrade_);
|
||||||
|
ImGui::Checkbox("Test address validation", &test_address_validation_);
|
||||||
|
ImGui::Checkbox("Test feature toggle", &test_feature_toggle_);
|
||||||
|
ImGui::Checkbox("Test data integrity", &test_data_integrity_);
|
||||||
|
|
||||||
|
if (ImGui::CollapsingHeader("Version Settings")) {
|
||||||
|
ImGui::Text("Version-specific addresses and features:");
|
||||||
|
ImGui::Text("Vanilla: 0x140145 = 0xFF");
|
||||||
|
ImGui::Text("v2: 0x140145 = 0x02, main palettes enabled");
|
||||||
|
ImGui::Text("v3: 0x140145 = 0x03, all features enabled");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
void InitializeVersionData() {
|
||||||
|
// Vanilla ROM addresses and values
|
||||||
|
vanilla_data_ = {
|
||||||
|
{"version_flag", {0x140145, 0xFF}}, // OverworldCustomASMHasBeenApplied
|
||||||
|
{"message_ids", {0x3F51D, 0x00}}, // Message ID table start
|
||||||
|
{"area_graphics", {0x7C9C, 0x00}}, // Area graphics table
|
||||||
|
{"area_palettes", {0x7D1C, 0x00}}, // Area palettes table
|
||||||
|
};
|
||||||
|
|
||||||
|
// v2 ROM addresses and values
|
||||||
|
v2_data_ = {
|
||||||
|
{"version_flag", {0x140145, 0x02}}, // v2 version
|
||||||
|
{"message_ids", {0x1417F8, 0x00}}, // Expanded message ID table
|
||||||
|
{"area_graphics", {0x7C9C, 0x00}}, // Same as vanilla
|
||||||
|
{"area_palettes", {0x7D1C, 0x00}}, // Same as vanilla
|
||||||
|
{"main_palettes", {0x140160, 0x00}}, // New v2 feature
|
||||||
|
};
|
||||||
|
|
||||||
|
// v3 ROM addresses and values
|
||||||
|
v3_data_ = {
|
||||||
|
{"version_flag", {0x140145, 0x03}}, // v3 version
|
||||||
|
{"message_ids", {0x1417F8, 0x00}}, // Same as v2
|
||||||
|
{"area_graphics", {0x7C9C, 0x00}}, // Same as vanilla
|
||||||
|
{"area_palettes", {0x7D1C, 0x00}}, // Same as vanilla
|
||||||
|
{"main_palettes", {0x140160, 0x00}}, // Same as v2
|
||||||
|
{"bg_colors", {0x140000, 0x00}}, // New v3 feature
|
||||||
|
{"subscreen_overlays", {0x140340, 0x00}}, // New v3 feature
|
||||||
|
{"animated_gfx", {0x1402A0, 0x00}}, // New v3 feature
|
||||||
|
{"custom_tiles", {0x140480, 0x00}}, // New v3 feature
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
void AddSkippedTest(TestResults& results, const std::string& test_name, const std::string& reason) {
|
||||||
|
TestResult result;
|
||||||
|
result.name = test_name;
|
||||||
|
result.suite_name = GetName();
|
||||||
|
result.category = GetCategory();
|
||||||
|
result.status = TestStatus::kSkipped;
|
||||||
|
result.error_message = reason;
|
||||||
|
result.duration = std::chrono::milliseconds{0};
|
||||||
|
result.timestamp = std::chrono::steady_clock::now();
|
||||||
|
results.AddResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
absl::Status ApplyVersionPatch(Rom& rom, const std::string& version) {
|
||||||
|
const auto* data = &vanilla_data_;
|
||||||
|
if (version == "v2") {
|
||||||
|
data = &v2_data_;
|
||||||
|
} else if (version == "v3") {
|
||||||
|
data = &v3_data_;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Apply version-specific data
|
||||||
|
for (const auto& [key, value] : *data) {
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(value.first, value.second));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Apply version-specific features
|
||||||
|
if (version == "v2") {
|
||||||
|
// Enable v2 features
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x140146, 0x01)); // Enable main palettes
|
||||||
|
} else if (version == "v3") {
|
||||||
|
// Enable v3 features
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x140146, 0x01)); // Enable main palettes
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x140147, 0x01)); // Enable area-specific BG
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x140148, 0x01)); // Enable subscreen overlay
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x140149, 0x01)); // Enable animated GFX
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x14014A, 0x01)); // Enable custom tile GFX groups
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x14014B, 0x01)); // Enable mosaic
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ValidateVersionAddresses(Rom& rom, const std::string& version) {
|
||||||
|
const auto* data = &vanilla_data_;
|
||||||
|
if (version == "v2") {
|
||||||
|
data = &v2_data_;
|
||||||
|
} else if (version == "v3") {
|
||||||
|
data = &v3_data_;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const auto& [key, value] : *data) {
|
||||||
|
auto byte_value = rom.ReadByte(value.first);
|
||||||
|
if (!byte_value.ok() || *byte_value != value.second) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void RunVanillaBaselineTest(TestResults& results, Rom* rom) {
|
||||||
|
auto start_time = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
TestResult result;
|
||||||
|
result.name = "Vanilla_Baseline_Test";
|
||||||
|
result.suite_name = GetName();
|
||||||
|
result.category = GetCategory();
|
||||||
|
result.timestamp = start_time;
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto& test_manager = TestManager::Get();
|
||||||
|
|
||||||
|
auto test_status = test_manager.TestRomWithCopy(rom, [&](Rom* test_rom) -> absl::Status {
|
||||||
|
// Validate vanilla addresses
|
||||||
|
if (!ValidateVersionAddresses(*test_rom, "vanilla")) {
|
||||||
|
return absl::InternalError("Vanilla address validation failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify version flag
|
||||||
|
auto version_byte = test_rom->ReadByte(0x140145);
|
||||||
|
if (!version_byte.ok()) {
|
||||||
|
return absl::InternalError("Failed to read version flag");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*version_byte != 0xFF) {
|
||||||
|
return absl::InternalError(absl::StrFormat(
|
||||||
|
"Expected vanilla version flag (0xFF), got 0x%02X", *version_byte));
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
});
|
||||||
|
|
||||||
|
if (test_status.ok()) {
|
||||||
|
result.status = TestStatus::kPassed;
|
||||||
|
result.error_message = "Vanilla baseline test passed - ROM is in vanilla state";
|
||||||
|
} else {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Vanilla baseline test failed: " + test_status.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Vanilla baseline test exception: " + std::string(e.what());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto end_time = std::chrono::steady_clock::now();
|
||||||
|
result.duration = std::chrono::duration_cast<std::chrono::milliseconds>(
|
||||||
|
end_time - start_time);
|
||||||
|
|
||||||
|
results.AddResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RunV2UpgradeTest(TestResults& results, Rom* rom) {
|
||||||
|
auto start_time = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
TestResult result;
|
||||||
|
result.name = "V2_Upgrade_Test";
|
||||||
|
result.suite_name = GetName();
|
||||||
|
result.category = GetCategory();
|
||||||
|
result.timestamp = start_time;
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto& test_manager = TestManager::Get();
|
||||||
|
|
||||||
|
auto test_status = test_manager.TestRomWithCopy(rom, [&](Rom* test_rom) -> absl::Status {
|
||||||
|
// Apply v2 patch
|
||||||
|
RETURN_IF_ERROR(ApplyVersionPatch(*test_rom, "v2"));
|
||||||
|
|
||||||
|
// Validate v2 addresses
|
||||||
|
if (!ValidateVersionAddresses(*test_rom, "v2")) {
|
||||||
|
return absl::InternalError("v2 address validation failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify version flag
|
||||||
|
auto version_byte = test_rom->ReadByte(0x140145);
|
||||||
|
if (!version_byte.ok()) {
|
||||||
|
return absl::InternalError("Failed to read version flag");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*version_byte != 0x02) {
|
||||||
|
return absl::InternalError(absl::StrFormat(
|
||||||
|
"Expected v2 version flag (0x02), got 0x%02X", *version_byte));
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
});
|
||||||
|
|
||||||
|
if (test_status.ok()) {
|
||||||
|
result.status = TestStatus::kPassed;
|
||||||
|
result.error_message = "v2 upgrade test passed - ROM successfully upgraded to v2";
|
||||||
|
} else {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "v2 upgrade test failed: " + test_status.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "v2 upgrade test exception: " + std::string(e.what());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto end_time = std::chrono::steady_clock::now();
|
||||||
|
result.duration = std::chrono::duration_cast<std::chrono::milliseconds>(
|
||||||
|
end_time - start_time);
|
||||||
|
|
||||||
|
results.AddResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RunV3UpgradeTest(TestResults& results, Rom* rom) {
|
||||||
|
auto start_time = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
TestResult result;
|
||||||
|
result.name = "V3_Upgrade_Test";
|
||||||
|
result.suite_name = GetName();
|
||||||
|
result.category = GetCategory();
|
||||||
|
result.timestamp = start_time;
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto& test_manager = TestManager::Get();
|
||||||
|
|
||||||
|
auto test_status = test_manager.TestRomWithCopy(rom, [&](Rom* test_rom) -> absl::Status {
|
||||||
|
// Apply v3 patch
|
||||||
|
RETURN_IF_ERROR(ApplyVersionPatch(*test_rom, "v3"));
|
||||||
|
|
||||||
|
// Validate v3 addresses
|
||||||
|
if (!ValidateVersionAddresses(*test_rom, "v3")) {
|
||||||
|
return absl::InternalError("v3 address validation failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify version flag
|
||||||
|
auto version_byte = test_rom->ReadByte(0x140145);
|
||||||
|
if (!version_byte.ok()) {
|
||||||
|
return absl::InternalError("Failed to read version flag");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*version_byte != 0x03) {
|
||||||
|
return absl::InternalError(absl::StrFormat(
|
||||||
|
"Expected v3 version flag (0x03), got 0x%02X", *version_byte));
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
});
|
||||||
|
|
||||||
|
if (test_status.ok()) {
|
||||||
|
result.status = TestStatus::kPassed;
|
||||||
|
result.error_message = "v3 upgrade test passed - ROM successfully upgraded to v3";
|
||||||
|
} else {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "v3 upgrade test failed: " + test_status.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "v3 upgrade test exception: " + std::string(e.what());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto end_time = std::chrono::steady_clock::now();
|
||||||
|
result.duration = std::chrono::duration_cast<std::chrono::milliseconds>(
|
||||||
|
end_time - start_time);
|
||||||
|
|
||||||
|
results.AddResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RunAddressValidationTest(TestResults& results, Rom* rom) {
|
||||||
|
auto start_time = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
TestResult result;
|
||||||
|
result.name = "Address_Validation_Test";
|
||||||
|
result.suite_name = GetName();
|
||||||
|
result.category = GetCategory();
|
||||||
|
result.timestamp = start_time;
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto& test_manager = TestManager::Get();
|
||||||
|
|
||||||
|
auto test_status = test_manager.TestRomWithCopy(rom, [&](Rom* test_rom) -> absl::Status {
|
||||||
|
// Test vanilla addresses
|
||||||
|
if (!ValidateVersionAddresses(*test_rom, "vanilla")) {
|
||||||
|
return absl::InternalError("Vanilla address validation failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test v2 addresses
|
||||||
|
RETURN_IF_ERROR(ApplyVersionPatch(*test_rom, "v2"));
|
||||||
|
if (!ValidateVersionAddresses(*test_rom, "v2")) {
|
||||||
|
return absl::InternalError("v2 address validation failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test v3 addresses
|
||||||
|
RETURN_IF_ERROR(ApplyVersionPatch(*test_rom, "v3"));
|
||||||
|
if (!ValidateVersionAddresses(*test_rom, "v3")) {
|
||||||
|
return absl::InternalError("v3 address validation failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
});
|
||||||
|
|
||||||
|
if (test_status.ok()) {
|
||||||
|
result.status = TestStatus::kPassed;
|
||||||
|
result.error_message = "Address validation test passed - all version addresses valid";
|
||||||
|
} else {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Address validation test failed: " + test_status.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Address validation test exception: " + std::string(e.what());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto end_time = std::chrono::steady_clock::now();
|
||||||
|
result.duration = std::chrono::duration_cast<std::chrono::milliseconds>(
|
||||||
|
end_time - start_time);
|
||||||
|
|
||||||
|
results.AddResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RunFeatureToggleTest(TestResults& results, Rom* rom) {
|
||||||
|
auto start_time = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
TestResult result;
|
||||||
|
result.name = "Feature_Toggle_Test";
|
||||||
|
result.suite_name = GetName();
|
||||||
|
result.category = GetCategory();
|
||||||
|
result.timestamp = start_time;
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto& test_manager = TestManager::Get();
|
||||||
|
|
||||||
|
auto test_status = test_manager.TestRomWithCopy(rom, [&](Rom* test_rom) -> absl::Status {
|
||||||
|
// Apply v3 patch
|
||||||
|
RETURN_IF_ERROR(ApplyVersionPatch(*test_rom, "v3"));
|
||||||
|
|
||||||
|
// Test feature flags
|
||||||
|
auto main_palettes = test_rom->ReadByte(0x140146);
|
||||||
|
auto area_bg = test_rom->ReadByte(0x140147);
|
||||||
|
auto subscreen_overlay = test_rom->ReadByte(0x140148);
|
||||||
|
auto animated_gfx = test_rom->ReadByte(0x140149);
|
||||||
|
auto custom_tiles = test_rom->ReadByte(0x14014A);
|
||||||
|
auto mosaic = test_rom->ReadByte(0x14014B);
|
||||||
|
|
||||||
|
if (!main_palettes.ok() || !area_bg.ok() || !subscreen_overlay.ok() ||
|
||||||
|
!animated_gfx.ok() || !custom_tiles.ok() || !mosaic.ok()) {
|
||||||
|
return absl::InternalError("Failed to read feature flags");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*main_palettes != 0x01 || *area_bg != 0x01 || *subscreen_overlay != 0x01 ||
|
||||||
|
*animated_gfx != 0x01 || *custom_tiles != 0x01 || *mosaic != 0x01) {
|
||||||
|
return absl::InternalError("Feature flags not properly enabled");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Disable some features
|
||||||
|
RETURN_IF_ERROR(test_rom->WriteByte(0x140147, 0x00)); // Disable area-specific BG
|
||||||
|
RETURN_IF_ERROR(test_rom->WriteByte(0x140149, 0x00)); // Disable animated GFX
|
||||||
|
|
||||||
|
// Verify features are disabled
|
||||||
|
auto disabled_area_bg = test_rom->ReadByte(0x140147);
|
||||||
|
auto disabled_animated_gfx = test_rom->ReadByte(0x140149);
|
||||||
|
|
||||||
|
if (!disabled_area_bg.ok() || !disabled_animated_gfx.ok()) {
|
||||||
|
return absl::InternalError("Failed to read disabled feature flags");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*disabled_area_bg != 0x00 || *disabled_animated_gfx != 0x00) {
|
||||||
|
return absl::InternalError("Feature flags not properly disabled");
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
});
|
||||||
|
|
||||||
|
if (test_status.ok()) {
|
||||||
|
result.status = TestStatus::kPassed;
|
||||||
|
result.error_message = "Feature toggle test passed - features can be enabled/disabled";
|
||||||
|
} else {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Feature toggle test failed: " + test_status.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Feature toggle test exception: " + std::string(e.what());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto end_time = std::chrono::steady_clock::now();
|
||||||
|
result.duration = std::chrono::duration_cast<std::chrono::milliseconds>(
|
||||||
|
end_time - start_time);
|
||||||
|
|
||||||
|
results.AddResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RunDataIntegrityTest(TestResults& results, Rom* rom) {
|
||||||
|
auto start_time = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
TestResult result;
|
||||||
|
result.name = "Data_Integrity_Test";
|
||||||
|
result.suite_name = GetName();
|
||||||
|
result.category = GetCategory();
|
||||||
|
result.timestamp = start_time;
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto& test_manager = TestManager::Get();
|
||||||
|
|
||||||
|
auto test_status = test_manager.TestRomWithCopy(rom, [&](Rom* test_rom) -> absl::Status {
|
||||||
|
// Store some original data
|
||||||
|
auto original_graphics = test_rom->ReadByte(0x7C9C);
|
||||||
|
auto original_palette = test_rom->ReadByte(0x7D1C);
|
||||||
|
auto original_sprite_set = test_rom->ReadByte(0x7A41);
|
||||||
|
|
||||||
|
if (!original_graphics.ok() || !original_palette.ok() || !original_sprite_set.ok()) {
|
||||||
|
return absl::InternalError("Failed to read original data");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Upgrade to v3
|
||||||
|
RETURN_IF_ERROR(ApplyVersionPatch(*test_rom, "v3"));
|
||||||
|
|
||||||
|
// Verify original data is preserved
|
||||||
|
auto preserved_graphics = test_rom->ReadByte(0x7C9C);
|
||||||
|
auto preserved_palette = test_rom->ReadByte(0x7D1C);
|
||||||
|
auto preserved_sprite_set = test_rom->ReadByte(0x7A41);
|
||||||
|
|
||||||
|
if (!preserved_graphics.ok() || !preserved_palette.ok() || !preserved_sprite_set.ok()) {
|
||||||
|
return absl::InternalError("Failed to read preserved data");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*preserved_graphics != *original_graphics ||
|
||||||
|
*preserved_palette != *original_palette ||
|
||||||
|
*preserved_sprite_set != *original_sprite_set) {
|
||||||
|
return absl::InternalError("Original data not preserved during upgrade");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify new v3 data is initialized
|
||||||
|
auto bg_colors = test_rom->ReadByte(0x140000);
|
||||||
|
auto subscreen_overlays = test_rom->ReadByte(0x140340);
|
||||||
|
auto animated_gfx = test_rom->ReadByte(0x1402A0);
|
||||||
|
auto custom_tiles = test_rom->ReadByte(0x140480);
|
||||||
|
|
||||||
|
if (!bg_colors.ok() || !subscreen_overlays.ok() ||
|
||||||
|
!animated_gfx.ok() || !custom_tiles.ok()) {
|
||||||
|
return absl::InternalError("Failed to read new v3 data");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*bg_colors != 0x00 || *subscreen_overlays != 0x00 ||
|
||||||
|
*animated_gfx != 0x00 || *custom_tiles != 0x00) {
|
||||||
|
return absl::InternalError("New v3 data not properly initialized");
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
});
|
||||||
|
|
||||||
|
if (test_status.ok()) {
|
||||||
|
result.status = TestStatus::kPassed;
|
||||||
|
result.error_message = "Data integrity test passed - original data preserved, new data initialized";
|
||||||
|
} else {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Data integrity test failed: " + test_status.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
result.status = TestStatus::kFailed;
|
||||||
|
result.error_message = "Data integrity test exception: " + std::string(e.what());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto end_time = std::chrono::steady_clock::now();
|
||||||
|
result.duration = std::chrono::duration_cast<std::chrono::milliseconds>(
|
||||||
|
end_time - start_time);
|
||||||
|
|
||||||
|
results.AddResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Configuration
|
||||||
|
bool test_vanilla_baseline_ = true;
|
||||||
|
bool test_v2_upgrade_ = true;
|
||||||
|
bool test_v3_upgrade_ = true;
|
||||||
|
bool test_address_validation_ = true;
|
||||||
|
bool test_feature_toggle_ = true;
|
||||||
|
bool test_data_integrity_ = true;
|
||||||
|
|
||||||
|
// Version data
|
||||||
|
std::map<std::string, std::pair<uint32_t, uint8_t>> vanilla_data_;
|
||||||
|
std::map<std::string, std::pair<uint32_t, uint8_t>> v2_data_;
|
||||||
|
std::map<std::string, std::pair<uint32_t, uint8_t>> v3_data_;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace test
|
||||||
|
} // namespace yaze
|
||||||
|
|
||||||
|
#endif // YAZE_APP_TEST_ZSCUSTOMOVERWORLD_TEST_SUITE_H
|
||||||
@@ -1,4 +1,3 @@
|
|||||||
|
|
||||||
set(YAZE_SRC_FILES "")
|
set(YAZE_SRC_FILES "")
|
||||||
foreach (file
|
foreach (file
|
||||||
app/rom.cc
|
app/rom.cc
|
||||||
@@ -12,46 +11,61 @@ foreach (file
|
|||||||
list(APPEND YAZE_SRC_FILES ${CMAKE_SOURCE_DIR}/src/${file})
|
list(APPEND YAZE_SRC_FILES ${CMAKE_SOURCE_DIR}/src/${file})
|
||||||
endforeach()
|
endforeach()
|
||||||
|
|
||||||
|
# Main test executable with enhanced argument handling for AI agents
|
||||||
add_executable(
|
add_executable(
|
||||||
yaze_test
|
yaze_test
|
||||||
yaze_test.cc
|
yaze_test.cc
|
||||||
rom_test.cc
|
|
||||||
test_editor.cc
|
test_editor.cc
|
||||||
hex_test.cc
|
test_editor.h
|
||||||
core/asar_wrapper_test.cc
|
testing.h
|
||||||
gfx/snes_tile_test.cc
|
test_utils.h
|
||||||
gfx/compression_test.cc
|
|
||||||
gfx/snes_palette_test.cc
|
# Unit Tests
|
||||||
zelda3/message_test.cc
|
unit/core/asar_wrapper_test.cc
|
||||||
zelda3/overworld_test.cc
|
unit/core/hex_test.cc
|
||||||
zelda3/overworld_integration_test.cc
|
unit/rom/rom_test.cc
|
||||||
zelda3/comprehensive_integration_test.cc
|
unit/gfx/snes_tile_test.cc
|
||||||
zelda3/dungeon_integration_test.cc
|
unit/gfx/compression_test.cc
|
||||||
zelda3/dungeon_object_renderer_integration_test.cc
|
unit/gfx/snes_palette_test.cc
|
||||||
zelda3/dungeon_object_renderer_mock_test.cc
|
unit/zelda3/message_test.cc
|
||||||
zelda3/dungeon_editor_system_integration_test.cc
|
unit/zelda3/overworld_test.cc
|
||||||
zelda3/sprite_builder_test.cc
|
unit/zelda3/object_parser_test.cc
|
||||||
zelda3/sprite_position_test.cc
|
unit/zelda3/object_parser_structs_test.cc
|
||||||
emu/cpu_test.cc
|
unit/zelda3/sprite_builder_test.cc
|
||||||
emu/ppu_test.cc
|
unit/zelda3/sprite_position_test.cc
|
||||||
emu/spc700_test.cc
|
unit/zelda3/test_dungeon_objects.cc
|
||||||
emu/audio/apu_test.cc
|
unit/zelda3/dungeon_component_unit_test.cc
|
||||||
emu/audio/ipl_handshake_test.cc
|
|
||||||
integration/dungeon_editor_test.cc
|
# Integration Tests
|
||||||
dungeon_component_unit_test.cc
|
|
||||||
integration/asar_integration_test.cc
|
integration/asar_integration_test.cc
|
||||||
integration/asar_rom_test.cc
|
integration/asar_rom_test.cc
|
||||||
editor/tile16_editor_test.cc
|
integration/dungeon_editor_test.cc
|
||||||
zelda3/object_parser_test.cc
|
integration/dungeon_editor_test.h
|
||||||
zelda3/object_parser_structs_test.cc
|
integration/editor/tile16_editor_test.cc
|
||||||
zelda3/test_dungeon_objects.cc
|
integration/editor/editor_integration_test.cc
|
||||||
|
integration/editor/editor_integration_test.h
|
||||||
|
|
||||||
|
# E2E Tests
|
||||||
|
e2e/rom_dependent/e2e_rom_test.cc
|
||||||
|
e2e/zscustomoverworld/zscustomoverworld_upgrade_test.cc
|
||||||
|
|
||||||
|
# Legacy Integration Tests (to be migrated)
|
||||||
|
unit/zelda3/comprehensive_integration_test.cc
|
||||||
|
unit/zelda3/overworld_integration_test.cc
|
||||||
|
unit/zelda3/dungeon_integration_test.cc
|
||||||
|
unit/zelda3/dungeon_editor_system_integration_test.cc
|
||||||
|
unit/zelda3/dungeon_object_renderer_integration_test.cc
|
||||||
|
unit/zelda3/dungeon_object_renderer_mock_test.cc
|
||||||
|
unit/zelda3/dungeon_object_rendering_tests.cc
|
||||||
|
unit/zelda3/dungeon_room_test.cc
|
||||||
)
|
)
|
||||||
|
|
||||||
# Add vanilla value extraction utility (only for local development with ROM access)
|
# Add vanilla value extraction utility (only for local development with ROM access)
|
||||||
if(NOT YAZE_MINIMAL_BUILD AND YAZE_ENABLE_ROM_TESTS)
|
if(NOT YAZE_MINIMAL_BUILD AND YAZE_ENABLE_ROM_TESTS)
|
||||||
add_executable(
|
add_executable(
|
||||||
extract_vanilla_values
|
extract_vanilla_values
|
||||||
zelda3/extract_vanilla_values.cc
|
unit/zelda3/extract_vanilla_values.cc
|
||||||
|
unit/zelda3/rom_patch_utility.cc
|
||||||
${YAZE_SRC_FILES}
|
${YAZE_SRC_FILES}
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -127,6 +141,7 @@ if(YAZE_ENABLE_UI_TESTS)
|
|||||||
target_link_libraries(yaze_test ${IMGUI_TEST_ENGINE_TARGET})
|
target_link_libraries(yaze_test ${IMGUI_TEST_ENGINE_TARGET})
|
||||||
target_compile_definitions(yaze_test PRIVATE ${IMGUI_TEST_ENGINE_DEFINITIONS})
|
target_compile_definitions(yaze_test PRIVATE ${IMGUI_TEST_ENGINE_DEFINITIONS})
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# ROM Testing Configuration
|
# ROM Testing Configuration
|
||||||
if(YAZE_ENABLE_ROM_TESTS)
|
if(YAZE_ENABLE_ROM_TESTS)
|
||||||
target_compile_definitions(yaze_test PRIVATE
|
target_compile_definitions(yaze_test PRIVATE
|
||||||
@@ -135,8 +150,6 @@ if(YAZE_ENABLE_ROM_TESTS)
|
|||||||
)
|
)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# ImGui Test Engine definitions are now handled conditionally above
|
|
||||||
|
|
||||||
# Platform-specific definitions
|
# Platform-specific definitions
|
||||||
if(UNIX AND NOT APPLE)
|
if(UNIX AND NOT APPLE)
|
||||||
target_compile_definitions(yaze_test PRIVATE "linux" "stricmp=strcasecmp")
|
target_compile_definitions(yaze_test PRIVATE "linux" "stricmp=strcasecmp")
|
||||||
@@ -155,5 +168,12 @@ if(YAZE_BUILD_TESTS)
|
|||||||
gtest_discover_tests(yaze_test)
|
gtest_discover_tests(yaze_test)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Add test labels using a simpler approach
|
# Test organization and labeling for CI/CD
|
||||||
# Note: Test names might have prefixes, we'll use regex patterns for CI
|
# Note: Test labeling is handled through the enhanced yaze_test executable
|
||||||
|
# which supports filtering by test categories using command line arguments:
|
||||||
|
# --unit, --integration, --e2e, --rom-dependent, --zscustomoverworld, etc.
|
||||||
|
#
|
||||||
|
# For CI/CD, use the test runner with appropriate filters:
|
||||||
|
# ./yaze_test --unit --verbose
|
||||||
|
# ./yaze_test --e2e --rom-path zelda3.sfc
|
||||||
|
# ./yaze_test --zscustomoverworld --verbose
|
||||||
195
test/README.md
Normal file
195
test/README.md
Normal file
@@ -0,0 +1,195 @@
|
|||||||
|
# YAZE Test Suite
|
||||||
|
|
||||||
|
This directory contains the comprehensive test suite for YAZE, organized for optimal AI agent testing and development workflow.
|
||||||
|
|
||||||
|
## Directory Structure
|
||||||
|
|
||||||
|
```
|
||||||
|
test/
|
||||||
|
├── unit/ # Unit tests for individual components
|
||||||
|
│ ├── core/ # Core functionality tests
|
||||||
|
│ ├── rom/ # ROM handling tests
|
||||||
|
│ ├── gfx/ # Graphics system tests
|
||||||
|
│ └── zelda3/ # Zelda 3 specific tests
|
||||||
|
├── integration/ # Integration tests
|
||||||
|
│ ├── editor/ # Editor integration tests
|
||||||
|
│ ├── asar_integration_test.cc
|
||||||
|
│ ├── asar_rom_test.cc
|
||||||
|
│ └── dungeon_editor_test.cc
|
||||||
|
├── e2e/ # End-to-end tests
|
||||||
|
│ ├── rom_dependent/ # ROM-dependent E2E tests
|
||||||
|
│ └── zscustomoverworld/ # ZSCustomOverworld upgrade tests
|
||||||
|
├── deprecated/ # Outdated tests (for cleanup)
|
||||||
|
│ └── emu/ # Deprecated emulator tests
|
||||||
|
├── mocks/ # Mock objects for testing
|
||||||
|
├── assets/ # Test assets and patches
|
||||||
|
└── yaze_test.cc # Enhanced test runner
|
||||||
|
```
|
||||||
|
|
||||||
|
## Test Categories
|
||||||
|
|
||||||
|
### Unit Tests (`unit/`)
|
||||||
|
- **Core**: ASAR wrapper, hex utilities, core functionality
|
||||||
|
- **ROM**: ROM loading, saving, validation
|
||||||
|
- **Graphics**: SNES tiles, palettes, compression
|
||||||
|
- **Zelda3**: Message system, overworld, objects, sprites
|
||||||
|
|
||||||
|
### Integration Tests (`integration/`)
|
||||||
|
- **Editor**: Tile editor, dungeon editor integration
|
||||||
|
- **ASAR**: ASAR integration and ROM patching
|
||||||
|
- **Dungeon**: Dungeon editor system integration
|
||||||
|
|
||||||
|
### End-to-End Tests (`e2e/`)
|
||||||
|
- **ROM Dependent**: Complete ROM editing workflow validation
|
||||||
|
- **ZSCustomOverworld**: Version upgrade testing (vanilla → v2 → v3)
|
||||||
|
|
||||||
|
## Enhanced Test Runner
|
||||||
|
|
||||||
|
The `yaze_test` executable now supports comprehensive argument handling for AI agents:
|
||||||
|
|
||||||
|
### Usage Examples
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Run all tests
|
||||||
|
./yaze_test
|
||||||
|
|
||||||
|
# Run specific test categories
|
||||||
|
./yaze_test --unit --verbose
|
||||||
|
./yaze_test --integration
|
||||||
|
./yaze_test --e2e --rom-path my_rom.sfc
|
||||||
|
./yaze_test --zscustomoverworld --verbose
|
||||||
|
|
||||||
|
# Run specific test patterns
|
||||||
|
./yaze_test RomTest.*
|
||||||
|
./yaze_test *ZSCustomOverworld*
|
||||||
|
|
||||||
|
# Skip ROM-dependent tests
|
||||||
|
./yaze_test --skip-rom-tests
|
||||||
|
|
||||||
|
# Enable UI tests
|
||||||
|
./yaze_test --enable-ui-tests
|
||||||
|
```
|
||||||
|
|
||||||
|
### Test Modes
|
||||||
|
|
||||||
|
- `--unit`: Unit tests only
|
||||||
|
- `--integration`: Integration tests only
|
||||||
|
- `--e2e`: End-to-end tests only
|
||||||
|
- `--rom-dependent`: ROM-dependent tests only
|
||||||
|
- `--zscustomoverworld`: ZSCustomOverworld tests only
|
||||||
|
- `--core`: Core functionality tests
|
||||||
|
- `--graphics`: Graphics tests
|
||||||
|
- `--editor`: Editor tests
|
||||||
|
- `--deprecated`: Deprecated tests (for cleanup)
|
||||||
|
|
||||||
|
### Options
|
||||||
|
|
||||||
|
- `--rom-path PATH`: Specify ROM path for testing
|
||||||
|
- `--skip-rom-tests`: Skip tests requiring ROM files
|
||||||
|
- `--enable-ui-tests`: Enable UI tests (requires display)
|
||||||
|
- `--verbose`: Enable verbose output
|
||||||
|
- `--help`: Show help message
|
||||||
|
|
||||||
|
## E2E ROM Testing
|
||||||
|
|
||||||
|
The E2E ROM test suite (`e2e/rom_dependent/e2e_rom_test.cc`) provides comprehensive validation of the complete ROM editing workflow:
|
||||||
|
|
||||||
|
1. **Load vanilla ROM**
|
||||||
|
2. **Apply various edits** (overworld, dungeon, graphics, etc.)
|
||||||
|
3. **Save changes**
|
||||||
|
4. **Reload ROM and verify edits persist**
|
||||||
|
5. **Verify no data corruption occurred**
|
||||||
|
|
||||||
|
### Test Cases
|
||||||
|
|
||||||
|
- `BasicROMLoadSave`: Basic ROM loading and saving
|
||||||
|
- `OverworldEditWorkflow`: Complete overworld editing workflow
|
||||||
|
- `DungeonEditWorkflow`: Complete dungeon editing workflow
|
||||||
|
- `TransactionSystem`: Multi-edit transaction validation
|
||||||
|
- `CorruptionDetection`: ROM corruption detection
|
||||||
|
- `LargeScaleEditing`: Large-scale editing without corruption
|
||||||
|
|
||||||
|
## ZSCustomOverworld Upgrade Testing
|
||||||
|
|
||||||
|
The ZSCustomOverworld test suite (`e2e/zscustomoverworld/zscustomoverworld_upgrade_test.cc`) validates version upgrades:
|
||||||
|
|
||||||
|
### Supported Upgrades
|
||||||
|
|
||||||
|
- **Vanilla → v2**: Basic upgrade with main palettes
|
||||||
|
- **v2 → v3**: Advanced upgrade with expanded features
|
||||||
|
- **Vanilla → v3**: Direct upgrade to latest version
|
||||||
|
|
||||||
|
### Test Cases
|
||||||
|
|
||||||
|
- `VanillaBaseline`: Validate vanilla ROM baseline
|
||||||
|
- `VanillaToV2Upgrade`: Test vanilla to v2 upgrade
|
||||||
|
- `V2ToV3Upgrade`: Test v2 to v3 upgrade
|
||||||
|
- `VanillaToV3Upgrade`: Test direct vanilla to v3 upgrade
|
||||||
|
- `AddressValidation`: Validate version-specific addresses
|
||||||
|
- `SaveCompatibility`: Test save compatibility between versions
|
||||||
|
- `FeatureToggle`: Test feature enablement/disablement
|
||||||
|
- `DataIntegrity`: Test data integrity during upgrades
|
||||||
|
|
||||||
|
### Version-Specific Features
|
||||||
|
|
||||||
|
#### Vanilla
|
||||||
|
- Basic overworld functionality
|
||||||
|
- Standard message IDs, area graphics, palettes
|
||||||
|
|
||||||
|
#### v2
|
||||||
|
- Main palettes support
|
||||||
|
- Expanded message ID table
|
||||||
|
|
||||||
|
#### v3
|
||||||
|
- Area-specific background colors
|
||||||
|
- Subscreen overlays
|
||||||
|
- Animated GFX
|
||||||
|
- Custom tile GFX groups
|
||||||
|
- Mosaic effects
|
||||||
|
|
||||||
|
## Environment Variables
|
||||||
|
|
||||||
|
- `YAZE_TEST_ROM_PATH`: Path to test ROM file
|
||||||
|
- `YAZE_SKIP_ROM_TESTS`: Skip ROM-dependent tests
|
||||||
|
- `YAZE_ENABLE_UI_TESTS`: Enable UI tests
|
||||||
|
- `YAZE_VERBOSE_TESTS`: Enable verbose test output
|
||||||
|
|
||||||
|
## CI/CD Integration
|
||||||
|
|
||||||
|
Tests are automatically labeled for CI/CD:
|
||||||
|
|
||||||
|
- `unit`: Fast unit tests
|
||||||
|
- `integration`: Medium-speed integration tests
|
||||||
|
- `e2e`: Slow end-to-end tests
|
||||||
|
- `rom`: ROM-dependent tests
|
||||||
|
- `zscustomoverworld`: ZSCustomOverworld specific tests
|
||||||
|
- `core`: Core functionality tests
|
||||||
|
- `graphics`: Graphics tests
|
||||||
|
- `editor`: Editor tests
|
||||||
|
- `deprecated`: Deprecated tests
|
||||||
|
|
||||||
|
## Deprecated Tests
|
||||||
|
|
||||||
|
The `deprecated/` directory contains outdated tests that no longer pass after the large refactor:
|
||||||
|
|
||||||
|
- **EMU tests**: CPU, PPU, SPC700, APU tests that are no longer compatible
|
||||||
|
- These tests are kept for reference but should not be run in CI/CD
|
||||||
|
|
||||||
|
## Best Practices
|
||||||
|
|
||||||
|
1. **Use appropriate test categories** for new tests
|
||||||
|
2. **Add comprehensive E2E tests** for new features
|
||||||
|
3. **Test upgrade paths** for ZSCustomOverworld features
|
||||||
|
4. **Validate data integrity** in all ROM operations
|
||||||
|
5. **Use descriptive test names** for AI agent clarity
|
||||||
|
6. **Include verbose output** for debugging
|
||||||
|
|
||||||
|
## AI Agent Testing
|
||||||
|
|
||||||
|
The enhanced test runner is specifically designed for AI agent testing:
|
||||||
|
|
||||||
|
- **Clear argument structure** for easy automation
|
||||||
|
- **Comprehensive help system** for understanding capabilities
|
||||||
|
- **Verbose output** for debugging and validation
|
||||||
|
- **Flexible test filtering** for targeted testing
|
||||||
|
- **Environment variable support** for configuration
|
||||||
241
test/e2e/rom_dependent/e2e_rom_test.cc
Normal file
241
test/e2e/rom_dependent/e2e_rom_test.cc
Normal file
@@ -0,0 +1,241 @@
|
|||||||
|
#include <gtest/gtest.h>
|
||||||
|
#include <filesystem>
|
||||||
|
#include <fstream>
|
||||||
|
#include <memory>
|
||||||
|
#include <vector>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
#include "app/rom.h"
|
||||||
|
#include "app/transaction.h"
|
||||||
|
#include "testing.h"
|
||||||
|
|
||||||
|
namespace yaze {
|
||||||
|
namespace test {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Comprehensive End-to-End ROM testing suite
|
||||||
|
*
|
||||||
|
* This test suite validates the complete ROM editing workflow:
|
||||||
|
* 1. Load vanilla ROM
|
||||||
|
* 2. Apply various edits (ROM data, graphics, etc.)
|
||||||
|
* 3. Save changes
|
||||||
|
* 4. Reload ROM and verify edits persist
|
||||||
|
* 5. Verify no data corruption occurred
|
||||||
|
*/
|
||||||
|
class E2ERomDependentTest : public ::testing::Test {
|
||||||
|
protected:
|
||||||
|
void SetUp() override {
|
||||||
|
// Skip tests if ROM is not available
|
||||||
|
if (getenv("YAZE_SKIP_ROM_TESTS")) {
|
||||||
|
GTEST_SKIP() << "ROM tests disabled";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get ROM path from environment or use default
|
||||||
|
const char* rom_path_env = getenv("YAZE_TEST_ROM_PATH");
|
||||||
|
vanilla_rom_path_ = rom_path_env ? rom_path_env : "zelda3.sfc";
|
||||||
|
|
||||||
|
if (!std::filesystem::exists(vanilla_rom_path_)) {
|
||||||
|
GTEST_SKIP() << "Test ROM not found: " << vanilla_rom_path_;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create test ROM copies
|
||||||
|
test_rom_path_ = "test_rom_edit.sfc";
|
||||||
|
backup_rom_path_ = "test_rom_backup.sfc";
|
||||||
|
|
||||||
|
// Copy vanilla ROM for testing
|
||||||
|
std::filesystem::copy_file(vanilla_rom_path_, test_rom_path_);
|
||||||
|
std::filesystem::copy_file(vanilla_rom_path_, backup_rom_path_);
|
||||||
|
}
|
||||||
|
|
||||||
|
void TearDown() override {
|
||||||
|
// Clean up test files
|
||||||
|
if (std::filesystem::exists(test_rom_path_)) {
|
||||||
|
std::filesystem::remove(test_rom_path_);
|
||||||
|
}
|
||||||
|
if (std::filesystem::exists(backup_rom_path_)) {
|
||||||
|
std::filesystem::remove(backup_rom_path_);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper to load ROM and verify basic integrity
|
||||||
|
static absl::Status LoadAndVerifyROM(const std::string& path, std::unique_ptr<Rom>& rom) {
|
||||||
|
rom = std::make_unique<Rom>();
|
||||||
|
RETURN_IF_ERROR(rom->LoadFromFile(path));
|
||||||
|
|
||||||
|
// Basic ROM integrity checks
|
||||||
|
EXPECT_EQ(rom->size(), 0x200000) << "ROM size should be 2MB";
|
||||||
|
EXPECT_NE(rom->data(), nullptr) << "ROM data should not be null";
|
||||||
|
|
||||||
|
// Check ROM header
|
||||||
|
EXPECT_EQ(rom->ReadByte(0x7FC0), 0x21) << "ROM should be LoROM format";
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper to verify ROM data integrity by comparing checksums
|
||||||
|
static bool VerifyROMIntegrity(const std::string& path1, const std::string& path2,
|
||||||
|
const std::vector<uint32_t>& exclude_ranges = {}) {
|
||||||
|
std::ifstream file1(path1, std::ios::binary);
|
||||||
|
std::ifstream file2(path2, std::ios::binary);
|
||||||
|
|
||||||
|
if (!file1.is_open() || !file2.is_open()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
file1.seekg(0, std::ios::end);
|
||||||
|
file2.seekg(0, std::ios::end);
|
||||||
|
|
||||||
|
size_t size1 = file1.tellg();
|
||||||
|
size_t size2 = file2.tellg();
|
||||||
|
|
||||||
|
if (size1 != size2) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
file1.seekg(0);
|
||||||
|
file2.seekg(0);
|
||||||
|
|
||||||
|
std::vector<char> buffer1(size1);
|
||||||
|
std::vector<char> buffer2(size2);
|
||||||
|
|
||||||
|
file1.read(buffer1.data(), size1);
|
||||||
|
file2.read(buffer2.data(), size2);
|
||||||
|
|
||||||
|
// Compare byte by byte, excluding specified ranges
|
||||||
|
for (size_t i = 0; i < size1; i++) {
|
||||||
|
bool in_exclude_range = false;
|
||||||
|
for (const auto& range : exclude_ranges) {
|
||||||
|
if (i >= (range & 0xFFFFFF) && i < ((range >> 24) & 0xFF)) {
|
||||||
|
in_exclude_range = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!in_exclude_range && buffer1[i] != buffer2[i]) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string vanilla_rom_path_;
|
||||||
|
std::string test_rom_path_;
|
||||||
|
std::string backup_rom_path_;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Test basic ROM loading and saving
|
||||||
|
TEST_F(E2ERomDependentTest, BasicROMLoadSave) {
|
||||||
|
std::unique_ptr<Rom> rom;
|
||||||
|
ASSERT_OK(LoadAndVerifyROM(vanilla_rom_path_, rom));
|
||||||
|
|
||||||
|
// Save ROM to test path
|
||||||
|
ASSERT_OK(rom->SaveToFile(Rom::SaveSettings{.filename = test_rom_path_}));
|
||||||
|
|
||||||
|
// Verify saved ROM matches original
|
||||||
|
EXPECT_TRUE(VerifyROMIntegrity(vanilla_rom_path_, test_rom_path_));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test ROM data editing workflow
|
||||||
|
TEST_F(E2ERomDependentTest, ROMDataEditWorkflow) {
|
||||||
|
std::unique_ptr<Rom> rom;
|
||||||
|
ASSERT_OK(LoadAndVerifyROM(vanilla_rom_path_, rom));
|
||||||
|
|
||||||
|
// Get initial state
|
||||||
|
auto initial_byte = rom->ReadByte(0x1000);
|
||||||
|
ASSERT_TRUE(initial_byte.ok());
|
||||||
|
|
||||||
|
// Make edits
|
||||||
|
ASSERT_OK(rom->WriteByte(0x1000, 0xAA));
|
||||||
|
ASSERT_OK(rom->WriteByte(0x2000, 0xBB));
|
||||||
|
ASSERT_OK(rom->WriteWord(0x3000, 0xCCDD));
|
||||||
|
|
||||||
|
// Save changes
|
||||||
|
ASSERT_OK(rom->SaveToFile(Rom::SaveSettings{.filename = test_rom_path_}));
|
||||||
|
|
||||||
|
// Reload and verify
|
||||||
|
std::unique_ptr<Rom> reloaded_rom;
|
||||||
|
ASSERT_OK(LoadAndVerifyROM(test_rom_path_, reloaded_rom));
|
||||||
|
|
||||||
|
EXPECT_EQ(reloaded_rom->ReadByte(0x1000), 0xAA);
|
||||||
|
EXPECT_EQ(reloaded_rom->ReadByte(0x2000), 0xBB);
|
||||||
|
EXPECT_EQ(reloaded_rom->ReadWord(0x3000), 0xCCDD);
|
||||||
|
|
||||||
|
// Verify other data wasn't corrupted
|
||||||
|
EXPECT_NE(reloaded_rom->ReadByte(0x1000), *initial_byte);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test transaction system with multiple edits
|
||||||
|
TEST_F(E2ERomDependentTest, TransactionSystem) {
|
||||||
|
std::unique_ptr<Rom> rom;
|
||||||
|
ASSERT_OK(LoadAndVerifyROM(vanilla_rom_path_, rom));
|
||||||
|
|
||||||
|
// Create transaction
|
||||||
|
auto transaction = std::make_unique<yaze::Transaction>(*rom);
|
||||||
|
|
||||||
|
// Make multiple edits in transaction
|
||||||
|
ASSERT_OK(transaction->WriteByte(0x1000, 0xAA));
|
||||||
|
ASSERT_OK(transaction->WriteByte(0x2000, 0xBB));
|
||||||
|
ASSERT_OK(transaction->WriteWord(0x3000, 0xCCDD));
|
||||||
|
|
||||||
|
// Commit transaction
|
||||||
|
ASSERT_OK(transaction->Commit());
|
||||||
|
|
||||||
|
// Save ROM
|
||||||
|
ASSERT_OK(rom->SaveToFile(Rom::SaveSettings{.filename = test_rom_path_}));
|
||||||
|
|
||||||
|
// Reload and verify all changes
|
||||||
|
std::unique_ptr<Rom> reloaded_rom;
|
||||||
|
ASSERT_OK(LoadAndVerifyROM(test_rom_path_, reloaded_rom));
|
||||||
|
|
||||||
|
EXPECT_EQ(reloaded_rom->ReadByte(0x1000), 0xAA);
|
||||||
|
EXPECT_EQ(reloaded_rom->ReadByte(0x2000), 0xBB);
|
||||||
|
EXPECT_EQ(reloaded_rom->ReadWord(0x3000), 0xCCDD);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test ROM corruption detection
|
||||||
|
TEST_F(E2ERomDependentTest, CorruptionDetection) {
|
||||||
|
std::unique_ptr<Rom> rom;
|
||||||
|
ASSERT_OK(LoadAndVerifyROM(vanilla_rom_path_, rom));
|
||||||
|
|
||||||
|
// Corrupt some data
|
||||||
|
ASSERT_OK(rom->WriteByte(0x1000, 0xFF)); // Corrupt some data
|
||||||
|
ASSERT_OK(rom->WriteByte(0x2000, 0xAA)); // Corrupt more data
|
||||||
|
|
||||||
|
// Save corrupted ROM
|
||||||
|
ASSERT_OK(rom->SaveToFile(Rom::SaveSettings{.filename = test_rom_path_}));
|
||||||
|
|
||||||
|
// Verify corruption is detected
|
||||||
|
std::unique_ptr<Rom> reloaded_rom;
|
||||||
|
ASSERT_OK(LoadAndVerifyROM(test_rom_path_, reloaded_rom));
|
||||||
|
|
||||||
|
EXPECT_EQ(reloaded_rom->ReadByte(0x1000), 0xFF);
|
||||||
|
EXPECT_EQ(reloaded_rom->ReadByte(0x2000), 0xAA);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test large-scale editing without corruption
|
||||||
|
TEST_F(E2ERomDependentTest, LargeScaleEditing) {
|
||||||
|
std::unique_ptr<Rom> rom;
|
||||||
|
ASSERT_OK(LoadAndVerifyROM(vanilla_rom_path_, rom));
|
||||||
|
|
||||||
|
// Edit multiple areas
|
||||||
|
for (int i = 0; i < 10; i++) {
|
||||||
|
ASSERT_OK(rom->WriteByte(0x1000 + i, i % 16));
|
||||||
|
ASSERT_OK(rom->WriteByte(0x2000 + i, (i + 1) % 16));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save and reload
|
||||||
|
ASSERT_OK(rom->SaveToFile(Rom::SaveSettings{.filename = test_rom_path_}));
|
||||||
|
|
||||||
|
std::unique_ptr<Rom> reloaded_rom;
|
||||||
|
ASSERT_OK(LoadAndVerifyROM(test_rom_path_, reloaded_rom));
|
||||||
|
|
||||||
|
// Verify all changes
|
||||||
|
for (int i = 0; i < 10; i++) {
|
||||||
|
EXPECT_EQ(reloaded_rom->ReadByte(0x1000 + i), i % 16);
|
||||||
|
EXPECT_EQ(reloaded_rom->ReadByte(0x2000 + i), (i + 1) % 16);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace test
|
||||||
|
} // namespace yaze
|
||||||
377
test/e2e/zscustomoverworld/zscustomoverworld_upgrade_test.cc
Normal file
377
test/e2e/zscustomoverworld/zscustomoverworld_upgrade_test.cc
Normal file
@@ -0,0 +1,377 @@
|
|||||||
|
#include <gtest/gtest.h>
|
||||||
|
#include <filesystem>
|
||||||
|
#include <memory>
|
||||||
|
#include <vector>
|
||||||
|
#include <string>
|
||||||
|
#include <map>
|
||||||
|
|
||||||
|
#include "app/rom.h"
|
||||||
|
#include "testing.h"
|
||||||
|
|
||||||
|
namespace yaze {
|
||||||
|
namespace test {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief ZSCustomOverworld upgrade testing suite
|
||||||
|
*
|
||||||
|
* This test suite validates ZSCustomOverworld version upgrades:
|
||||||
|
* 1. Vanilla -> v2 upgrade with proper address changes
|
||||||
|
* 2. v2 -> v3 upgrade with expanded features
|
||||||
|
* 3. Address validation for each version
|
||||||
|
* 4. Save compatibility between versions
|
||||||
|
* 5. Feature enablement/disablement
|
||||||
|
*/
|
||||||
|
class ZSCustomOverworldUpgradeTest : public ::testing::Test {
|
||||||
|
protected:
|
||||||
|
void SetUp() override {
|
||||||
|
// Skip tests if ROM is not available
|
||||||
|
if (getenv("YAZE_SKIP_ROM_TESTS")) {
|
||||||
|
GTEST_SKIP() << "ROM tests disabled";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get ROM path from environment or use default
|
||||||
|
const char* rom_path_env = getenv("YAZE_TEST_ROM_PATH");
|
||||||
|
vanilla_rom_path_ = rom_path_env ? rom_path_env : "zelda3.sfc";
|
||||||
|
|
||||||
|
if (!std::filesystem::exists(vanilla_rom_path_)) {
|
||||||
|
GTEST_SKIP() << "Test ROM not found: " << vanilla_rom_path_;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create test ROM copies for each version
|
||||||
|
vanilla_test_path_ = "test_vanilla.sfc";
|
||||||
|
v2_test_path_ = "test_v2.sfc";
|
||||||
|
v3_test_path_ = "test_v3.sfc";
|
||||||
|
|
||||||
|
// Copy vanilla ROM for testing
|
||||||
|
std::filesystem::copy_file(vanilla_rom_path_, vanilla_test_path_);
|
||||||
|
|
||||||
|
// Define version-specific addresses and features
|
||||||
|
InitializeVersionData();
|
||||||
|
}
|
||||||
|
|
||||||
|
void TearDown() override {
|
||||||
|
// Clean up test files
|
||||||
|
std::vector<std::string> test_files = {
|
||||||
|
vanilla_test_path_, v2_test_path_, v3_test_path_
|
||||||
|
};
|
||||||
|
|
||||||
|
for (const auto& file : test_files) {
|
||||||
|
if (std::filesystem::exists(file)) {
|
||||||
|
std::filesystem::remove(file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void InitializeVersionData() {
|
||||||
|
// Vanilla ROM addresses and values
|
||||||
|
vanilla_data_ = {
|
||||||
|
{"version_flag", {0x140145, 0xFF}}, // OverworldCustomASMHasBeenApplied
|
||||||
|
{"message_ids", {0x3F51D, 0x00}}, // Message ID table start
|
||||||
|
{"area_graphics", {0x7C9C, 0x00}}, // Area graphics table
|
||||||
|
{"area_palettes", {0x7D1C, 0x00}}, // Area palettes table
|
||||||
|
{"screen_sizes", {0x1788D, 0x01}}, // Screen sizes table
|
||||||
|
{"sprite_sets", {0x7A41, 0x00}}, // Sprite sets table
|
||||||
|
{"sprite_palettes", {0x7B41, 0x00}}, // Sprite palettes table
|
||||||
|
};
|
||||||
|
|
||||||
|
// v2 ROM addresses and values
|
||||||
|
v2_data_ = {
|
||||||
|
{"version_flag", {0x140145, 0x02}}, // v2 version
|
||||||
|
{"message_ids", {0x1417F8, 0x00}}, // Expanded message ID table
|
||||||
|
{"area_graphics", {0x7C9C, 0x00}}, // Same as vanilla
|
||||||
|
{"area_palettes", {0x7D1C, 0x00}}, // Same as vanilla
|
||||||
|
{"screen_sizes", {0x1788D, 0x01}}, // Same as vanilla
|
||||||
|
{"sprite_sets", {0x7A41, 0x00}}, // Same as vanilla
|
||||||
|
{"sprite_palettes", {0x7B41, 0x00}}, // Same as vanilla
|
||||||
|
{"main_palettes", {0x140160, 0x00}}, // New v2 feature
|
||||||
|
};
|
||||||
|
|
||||||
|
// v3 ROM addresses and values
|
||||||
|
v3_data_ = {
|
||||||
|
{"version_flag", {0x140145, 0x03}}, // v3 version
|
||||||
|
{"message_ids", {0x1417F8, 0x00}}, // Same as v2
|
||||||
|
{"area_graphics", {0x7C9C, 0x00}}, // Same as vanilla
|
||||||
|
{"area_palettes", {0x7D1C, 0x00}}, // Same as vanilla
|
||||||
|
{"screen_sizes", {0x1788D, 0x01}}, // Same as vanilla
|
||||||
|
{"sprite_sets", {0x7A41, 0x00}}, // Same as vanilla
|
||||||
|
{"sprite_palettes", {0x7B41, 0x00}}, // Same as vanilla
|
||||||
|
{"main_palettes", {0x140160, 0x00}}, // Same as v2
|
||||||
|
{"bg_colors", {0x140000, 0x00}}, // New v3 feature
|
||||||
|
{"subscreen_overlays", {0x140340, 0x00}}, // New v3 feature
|
||||||
|
{"animated_gfx", {0x1402A0, 0x00}}, // New v3 feature
|
||||||
|
{"custom_tiles", {0x140480, 0x00}}, // New v3 feature
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper to apply version-specific patches
|
||||||
|
absl::Status ApplyVersionPatch(Rom& rom, const std::string& version) {
|
||||||
|
const auto* data = &vanilla_data_;
|
||||||
|
if (version == "v2") {
|
||||||
|
data = &v2_data_;
|
||||||
|
} else if (version == "v3") {
|
||||||
|
data = &v3_data_;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Apply version-specific data
|
||||||
|
for (const auto& [key, value] : *data) {
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(value.first, value.second));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Apply version-specific features
|
||||||
|
if (version == "v2") {
|
||||||
|
// Enable v2 features
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x140146, 0x01)); // Enable main palettes
|
||||||
|
} else if (version == "v3") {
|
||||||
|
// Enable v3 features
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x140146, 0x01)); // Enable main palettes
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x140147, 0x01)); // Enable area-specific BG
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x140148, 0x01)); // Enable subscreen overlay
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x140149, 0x01)); // Enable animated GFX
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x14014A, 0x01)); // Enable custom tile GFX groups
|
||||||
|
RETURN_IF_ERROR(rom.WriteByte(0x14014B, 0x01)); // Enable mosaic
|
||||||
|
}
|
||||||
|
|
||||||
|
return absl::OkStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper to validate version-specific addresses
|
||||||
|
bool ValidateVersionAddresses(Rom& rom, const std::string& version) {
|
||||||
|
const auto* data = &vanilla_data_;
|
||||||
|
if (version == "v2") {
|
||||||
|
data = &v2_data_;
|
||||||
|
} else if (version == "v3") {
|
||||||
|
data = &v3_data_;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const auto& [key, value] : *data) {
|
||||||
|
auto byte_value = rom.ReadByte(value.first);
|
||||||
|
if (!byte_value.ok() || *byte_value != value.second) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string vanilla_rom_path_;
|
||||||
|
std::string vanilla_test_path_;
|
||||||
|
std::string v2_test_path_;
|
||||||
|
std::string v3_test_path_;
|
||||||
|
|
||||||
|
std::map<std::string, std::pair<uint32_t, uint8_t>> vanilla_data_;
|
||||||
|
std::map<std::string, std::pair<uint32_t, uint8_t>> v2_data_;
|
||||||
|
std::map<std::string, std::pair<uint32_t, uint8_t>> v3_data_;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Test vanilla ROM baseline
|
||||||
|
TEST_F(ZSCustomOverworldUpgradeTest, VanillaBaseline) {
|
||||||
|
std::unique_ptr<Rom> rom = std::make_unique<Rom>();
|
||||||
|
ASSERT_OK(rom->LoadFromFile(vanilla_test_path_));
|
||||||
|
|
||||||
|
// Validate vanilla addresses
|
||||||
|
EXPECT_TRUE(ValidateVersionAddresses(*rom, "vanilla"));
|
||||||
|
|
||||||
|
// Verify version flag
|
||||||
|
auto version_byte = rom->ReadByte(0x140145);
|
||||||
|
ASSERT_TRUE(version_byte.ok());
|
||||||
|
EXPECT_EQ(*version_byte, 0xFF);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test vanilla to v2 upgrade
|
||||||
|
TEST_F(ZSCustomOverworldUpgradeTest, VanillaToV2Upgrade) {
|
||||||
|
// Load vanilla ROM
|
||||||
|
std::unique_ptr<Rom> rom = std::make_unique<Rom>();
|
||||||
|
ASSERT_OK(rom->LoadFromFile(vanilla_test_path_));
|
||||||
|
|
||||||
|
// Apply v2 patch
|
||||||
|
ASSERT_OK(ApplyVersionPatch(*rom, "v2"));
|
||||||
|
|
||||||
|
// Validate v2 addresses
|
||||||
|
EXPECT_TRUE(ValidateVersionAddresses(*rom, "v2"));
|
||||||
|
|
||||||
|
// Save v2 ROM
|
||||||
|
ASSERT_OK(rom->SaveToFile(Rom::SaveSettings{.filename = v2_test_path_}));
|
||||||
|
|
||||||
|
// Reload and verify
|
||||||
|
std::unique_ptr<Rom> reloaded_rom = std::make_unique<Rom>();
|
||||||
|
ASSERT_OK(reloaded_rom->LoadFromFile(v2_test_path_));
|
||||||
|
|
||||||
|
EXPECT_TRUE(ValidateVersionAddresses(*reloaded_rom, "v2"));
|
||||||
|
auto version_byte = reloaded_rom->ReadByte(0x140145);
|
||||||
|
ASSERT_TRUE(version_byte.ok());
|
||||||
|
EXPECT_EQ(*version_byte, 0x02);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test v2 to v3 upgrade
|
||||||
|
TEST_F(ZSCustomOverworldUpgradeTest, V2ToV3Upgrade) {
|
||||||
|
// Load vanilla ROM
|
||||||
|
std::unique_ptr<Rom> rom = std::make_unique<Rom>();
|
||||||
|
ASSERT_OK(rom->LoadFromFile(vanilla_test_path_));
|
||||||
|
|
||||||
|
// Apply v2 patch first
|
||||||
|
ASSERT_OK(ApplyVersionPatch(*rom, "v2"));
|
||||||
|
|
||||||
|
// Apply v3 patch
|
||||||
|
ASSERT_OK(ApplyVersionPatch(*rom, "v3"));
|
||||||
|
|
||||||
|
// Validate v3 addresses
|
||||||
|
EXPECT_TRUE(ValidateVersionAddresses(*rom, "v3"));
|
||||||
|
|
||||||
|
// Save v3 ROM
|
||||||
|
ASSERT_OK(rom->SaveToFile(Rom::SaveSettings{.filename = v3_test_path_}));
|
||||||
|
|
||||||
|
// Reload and verify
|
||||||
|
std::unique_ptr<Rom> reloaded_rom = std::make_unique<Rom>();
|
||||||
|
ASSERT_OK(reloaded_rom->LoadFromFile(v3_test_path_));
|
||||||
|
|
||||||
|
EXPECT_TRUE(ValidateVersionAddresses(*reloaded_rom, "v3"));
|
||||||
|
auto version_byte = reloaded_rom->ReadByte(0x140145);
|
||||||
|
ASSERT_TRUE(version_byte.ok());
|
||||||
|
EXPECT_EQ(*version_byte, 0x03);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test direct vanilla to v3 upgrade
|
||||||
|
TEST_F(ZSCustomOverworldUpgradeTest, VanillaToV3Upgrade) {
|
||||||
|
// Load vanilla ROM
|
||||||
|
std::unique_ptr<Rom> rom = std::make_unique<Rom>();
|
||||||
|
ASSERT_OK(rom->LoadFromFile(vanilla_test_path_));
|
||||||
|
|
||||||
|
// Apply v3 patch directly
|
||||||
|
ASSERT_OK(ApplyVersionPatch(*rom, "v3"));
|
||||||
|
|
||||||
|
// Validate v3 addresses
|
||||||
|
EXPECT_TRUE(ValidateVersionAddresses(*rom, "v3"));
|
||||||
|
|
||||||
|
// Save v3 ROM
|
||||||
|
ASSERT_OK(rom->SaveToFile(Rom::SaveSettings{.filename = v3_test_path_}));
|
||||||
|
|
||||||
|
// Reload and verify
|
||||||
|
std::unique_ptr<Rom> reloaded_rom = std::make_unique<Rom>();
|
||||||
|
ASSERT_OK(reloaded_rom->LoadFromFile(v3_test_path_));
|
||||||
|
|
||||||
|
EXPECT_TRUE(ValidateVersionAddresses(*reloaded_rom, "v3"));
|
||||||
|
auto version_byte = reloaded_rom->ReadByte(0x140145);
|
||||||
|
ASSERT_TRUE(version_byte.ok());
|
||||||
|
EXPECT_EQ(*version_byte, 0x03);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test address validation for each version
|
||||||
|
TEST_F(ZSCustomOverworldUpgradeTest, AddressValidation) {
|
||||||
|
// Test vanilla addresses
|
||||||
|
std::unique_ptr<Rom> vanilla_rom = std::make_unique<Rom>();
|
||||||
|
ASSERT_OK(vanilla_rom->LoadFromFile(vanilla_test_path_));
|
||||||
|
EXPECT_TRUE(ValidateVersionAddresses(*vanilla_rom, "vanilla"));
|
||||||
|
|
||||||
|
// Test v2 addresses
|
||||||
|
ASSERT_OK(ApplyVersionPatch(*vanilla_rom, "v2"));
|
||||||
|
EXPECT_TRUE(ValidateVersionAddresses(*vanilla_rom, "v2"));
|
||||||
|
|
||||||
|
// Test v3 addresses
|
||||||
|
ASSERT_OK(ApplyVersionPatch(*vanilla_rom, "v3"));
|
||||||
|
EXPECT_TRUE(ValidateVersionAddresses(*vanilla_rom, "v3"));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test feature enablement/disablement
|
||||||
|
TEST_F(ZSCustomOverworldUpgradeTest, FeatureToggle) {
|
||||||
|
std::unique_ptr<Rom> rom = std::make_unique<Rom>();
|
||||||
|
ASSERT_OK(rom->LoadFromFile(vanilla_test_path_));
|
||||||
|
ASSERT_OK(ApplyVersionPatch(*rom, "v3"));
|
||||||
|
|
||||||
|
// Test feature flags
|
||||||
|
auto main_palettes = rom->ReadByte(0x140146);
|
||||||
|
auto area_bg = rom->ReadByte(0x140147);
|
||||||
|
auto subscreen_overlay = rom->ReadByte(0x140148);
|
||||||
|
auto animated_gfx = rom->ReadByte(0x140149);
|
||||||
|
auto custom_tiles = rom->ReadByte(0x14014A);
|
||||||
|
auto mosaic = rom->ReadByte(0x14014B);
|
||||||
|
|
||||||
|
ASSERT_TRUE(main_palettes.ok());
|
||||||
|
ASSERT_TRUE(area_bg.ok());
|
||||||
|
ASSERT_TRUE(subscreen_overlay.ok());
|
||||||
|
ASSERT_TRUE(animated_gfx.ok());
|
||||||
|
ASSERT_TRUE(custom_tiles.ok());
|
||||||
|
ASSERT_TRUE(mosaic.ok());
|
||||||
|
|
||||||
|
EXPECT_EQ(*main_palettes, 0x01); // Main palettes enabled
|
||||||
|
EXPECT_EQ(*area_bg, 0x01); // Area-specific BG enabled
|
||||||
|
EXPECT_EQ(*subscreen_overlay, 0x01); // Subscreen overlay enabled
|
||||||
|
EXPECT_EQ(*animated_gfx, 0x01); // Animated GFX enabled
|
||||||
|
EXPECT_EQ(*custom_tiles, 0x01); // Custom tile GFX groups enabled
|
||||||
|
EXPECT_EQ(*mosaic, 0x01); // Mosaic enabled
|
||||||
|
|
||||||
|
// Disable some features
|
||||||
|
ASSERT_OK(rom->WriteByte(0x140147, 0x00)); // Disable area-specific BG
|
||||||
|
ASSERT_OK(rom->WriteByte(0x140149, 0x00)); // Disable animated GFX
|
||||||
|
|
||||||
|
// Verify features are disabled
|
||||||
|
auto disabled_area_bg = rom->ReadByte(0x140147);
|
||||||
|
auto disabled_animated_gfx = rom->ReadByte(0x140149);
|
||||||
|
ASSERT_TRUE(disabled_area_bg.ok());
|
||||||
|
ASSERT_TRUE(disabled_animated_gfx.ok());
|
||||||
|
|
||||||
|
EXPECT_EQ(*disabled_area_bg, 0x00);
|
||||||
|
EXPECT_EQ(*disabled_animated_gfx, 0x00);
|
||||||
|
|
||||||
|
// Re-enable features
|
||||||
|
ASSERT_OK(rom->WriteByte(0x140147, 0x01));
|
||||||
|
ASSERT_OK(rom->WriteByte(0x140149, 0x01));
|
||||||
|
|
||||||
|
// Verify features are re-enabled
|
||||||
|
auto reenabled_area_bg = rom->ReadByte(0x140147);
|
||||||
|
auto reenabled_animated_gfx = rom->ReadByte(0x140149);
|
||||||
|
ASSERT_TRUE(reenabled_area_bg.ok());
|
||||||
|
ASSERT_TRUE(reenabled_animated_gfx.ok());
|
||||||
|
|
||||||
|
EXPECT_EQ(*reenabled_area_bg, 0x01);
|
||||||
|
EXPECT_EQ(*reenabled_animated_gfx, 0x01);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test data integrity during upgrades
|
||||||
|
TEST_F(ZSCustomOverworldUpgradeTest, DataIntegrity) {
|
||||||
|
std::unique_ptr<Rom> rom = std::make_unique<Rom>();
|
||||||
|
ASSERT_OK(rom->LoadFromFile(vanilla_test_path_));
|
||||||
|
|
||||||
|
// Store some original data
|
||||||
|
auto original_graphics = rom->ReadByte(0x7C9C);
|
||||||
|
auto original_palette = rom->ReadByte(0x7D1C);
|
||||||
|
auto original_sprite_set = rom->ReadByte(0x7A41);
|
||||||
|
|
||||||
|
ASSERT_TRUE(original_graphics.ok());
|
||||||
|
ASSERT_TRUE(original_palette.ok());
|
||||||
|
ASSERT_TRUE(original_sprite_set.ok());
|
||||||
|
|
||||||
|
// Upgrade to v3
|
||||||
|
ASSERT_OK(ApplyVersionPatch(*rom, "v3"));
|
||||||
|
|
||||||
|
// Verify original data is preserved
|
||||||
|
auto preserved_graphics = rom->ReadByte(0x7C9C);
|
||||||
|
auto preserved_palette = rom->ReadByte(0x7D1C);
|
||||||
|
auto preserved_sprite_set = rom->ReadByte(0x7A41);
|
||||||
|
|
||||||
|
ASSERT_TRUE(preserved_graphics.ok());
|
||||||
|
ASSERT_TRUE(preserved_palette.ok());
|
||||||
|
ASSERT_TRUE(preserved_sprite_set.ok());
|
||||||
|
|
||||||
|
EXPECT_EQ(*preserved_graphics, *original_graphics);
|
||||||
|
EXPECT_EQ(*preserved_palette, *original_palette);
|
||||||
|
EXPECT_EQ(*preserved_sprite_set, *original_sprite_set);
|
||||||
|
|
||||||
|
// Verify new v3 data is initialized
|
||||||
|
auto bg_colors = rom->ReadByte(0x140000);
|
||||||
|
auto subscreen_overlays = rom->ReadByte(0x140340);
|
||||||
|
auto animated_gfx = rom->ReadByte(0x1402A0);
|
||||||
|
auto custom_tiles = rom->ReadByte(0x140480);
|
||||||
|
|
||||||
|
ASSERT_TRUE(bg_colors.ok());
|
||||||
|
ASSERT_TRUE(subscreen_overlays.ok());
|
||||||
|
ASSERT_TRUE(animated_gfx.ok());
|
||||||
|
ASSERT_TRUE(custom_tiles.ok());
|
||||||
|
|
||||||
|
EXPECT_EQ(*bg_colors, 0x00); // BG colors
|
||||||
|
EXPECT_EQ(*subscreen_overlays, 0x00); // Subscreen overlays
|
||||||
|
EXPECT_EQ(*animated_gfx, 0x00); // Animated GFX
|
||||||
|
EXPECT_EQ(*custom_tiles, 0x00); // Custom tiles
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace test
|
||||||
|
} // namespace yaze
|
||||||
@@ -1,134 +0,0 @@
|
|||||||
#include "app/emu/audio/apu.h"
|
|
||||||
#include "app/emu/memory/memory.h"
|
|
||||||
|
|
||||||
#include <gmock/gmock-nice-strict.h>
|
|
||||||
#include <gmock/gmock.h>
|
|
||||||
#include <gtest/gtest.h>
|
|
||||||
|
|
||||||
namespace yaze {
|
|
||||||
namespace test {
|
|
||||||
|
|
||||||
using testing::_;
|
|
||||||
using testing::Return;
|
|
||||||
using yaze::emu::Apu;
|
|
||||||
using yaze::emu::MemoryImpl;
|
|
||||||
|
|
||||||
class ApuTest : public ::testing::Test {
|
|
||||||
protected:
|
|
||||||
void SetUp() override {
|
|
||||||
memory_ = std::make_unique<MemoryImpl>();
|
|
||||||
apu_ = std::make_unique<Apu>(*memory_);
|
|
||||||
apu_->Init();
|
|
||||||
}
|
|
||||||
|
|
||||||
std::unique_ptr<MemoryImpl> memory_;
|
|
||||||
std::unique_ptr<Apu> apu_;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Test the IPL ROM handshake sequence timing
|
|
||||||
TEST_F(ApuTest, IplRomHandshakeTiming) {
|
|
||||||
// 1. Initial state check
|
|
||||||
EXPECT_EQ(apu_->Read(0x00) & 0x80, 0); // Ready bit should be clear
|
|
||||||
|
|
||||||
// 2. Start handshake
|
|
||||||
apu_->Write(0x00, 0x80); // Set control register bit 7
|
|
||||||
|
|
||||||
// 3. Wait for APU ready signal with cycle counting
|
|
||||||
int cycles = 0;
|
|
||||||
const int max_cycles = 1000; // Maximum expected cycles for handshake
|
|
||||||
while (!(apu_->Read(0x00) & 0x80) && cycles < max_cycles) {
|
|
||||||
apu_->RunCycles(1);
|
|
||||||
cycles++;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 4. Verify timing constraints
|
|
||||||
EXPECT_LT(cycles, max_cycles); // Should complete within max cycles
|
|
||||||
EXPECT_GT(cycles, 0); // Should take some cycles
|
|
||||||
EXPECT_TRUE(apu_->Read(0x00) & 0x80); // Ready bit should be set
|
|
||||||
|
|
||||||
// 5. Verify handshake completion
|
|
||||||
EXPECT_EQ(apu_->GetStatus() & 0x80, 0x80); // Ready bit in status register
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test APU initialization sequence
|
|
||||||
TEST_F(ApuTest, ApuInitialization) {
|
|
||||||
// 1. Check initial state
|
|
||||||
EXPECT_EQ(apu_->GetStatus(), 0x00);
|
|
||||||
EXPECT_EQ(apu_->GetControl(), 0x00);
|
|
||||||
|
|
||||||
// 2. Initialize APU
|
|
||||||
apu_->Init();
|
|
||||||
|
|
||||||
// 3. Verify initialization
|
|
||||||
EXPECT_EQ(apu_->GetStatus(), 0x00);
|
|
||||||
EXPECT_EQ(apu_->GetControl(), 0x00);
|
|
||||||
|
|
||||||
// 4. Check DSP registers are initialized
|
|
||||||
for (int i = 0; i < 128; i++) {
|
|
||||||
EXPECT_EQ(apu_->Read(0x00 + i), 0x00);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test sample generation and timing
|
|
||||||
TEST_F(ApuTest, SampleGenerationTiming) {
|
|
||||||
// 1. Generate samples
|
|
||||||
const int sample_count = 1024;
|
|
||||||
std::vector<int16_t> samples(sample_count);
|
|
||||||
|
|
||||||
// 2. Measure timing
|
|
||||||
uint64_t start_cycles = apu_->GetCycles();
|
|
||||||
apu_->GetSamples(samples.data(), sample_count, false);
|
|
||||||
uint64_t end_cycles = apu_->GetCycles();
|
|
||||||
|
|
||||||
// 3. Verify timing
|
|
||||||
EXPECT_GT(end_cycles - start_cycles, 0);
|
|
||||||
|
|
||||||
// 4. Verify samples
|
|
||||||
bool has_non_zero = false;
|
|
||||||
for (int i = 0; i < sample_count; ++i) {
|
|
||||||
if (samples[i] != 0) {
|
|
||||||
has_non_zero = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
EXPECT_TRUE(has_non_zero);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test DSP register access timing
|
|
||||||
TEST_F(ApuTest, DspRegisterAccessTiming) {
|
|
||||||
// 1. Write to DSP registers
|
|
||||||
const uint8_t test_value = 0x42;
|
|
||||||
uint64_t start_cycles = apu_->GetCycles();
|
|
||||||
|
|
||||||
apu_->Write(0x00, 0x80); // Set control register
|
|
||||||
apu_->Write(0x01, test_value); // Write to DSP address
|
|
||||||
|
|
||||||
uint64_t end_cycles = apu_->GetCycles();
|
|
||||||
|
|
||||||
// 2. Verify timing
|
|
||||||
EXPECT_GT(end_cycles - start_cycles, 0);
|
|
||||||
|
|
||||||
// 3. Verify register access
|
|
||||||
EXPECT_EQ(apu_->Read(0x01), test_value);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test DMA transfer timing
|
|
||||||
TEST_F(ApuTest, DmaTransferTiming) {
|
|
||||||
// 1. Prepare DMA data
|
|
||||||
const uint8_t data[] = {0x01, 0x02, 0x03, 0x04};
|
|
||||||
|
|
||||||
// 2. Measure DMA timing
|
|
||||||
uint64_t start_cycles = apu_->GetCycles();
|
|
||||||
apu_->WriteDma(0x00, data, sizeof(data));
|
|
||||||
uint64_t end_cycles = apu_->GetCycles();
|
|
||||||
|
|
||||||
// 3. Verify timing
|
|
||||||
EXPECT_GT(end_cycles - start_cycles, 0);
|
|
||||||
|
|
||||||
// 4. Verify DMA transfer
|
|
||||||
EXPECT_EQ(apu_->Read(0x00), 0x01);
|
|
||||||
EXPECT_EQ(apu_->Read(0x01), 0x02);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace test
|
|
||||||
} // namespace yaze
|
|
||||||
@@ -1,122 +0,0 @@
|
|||||||
#include "app/emu/audio/apu.h"
|
|
||||||
#include "app/emu/memory/memory.h"
|
|
||||||
|
|
||||||
#include <gmock/gmock-nice-strict.h>
|
|
||||||
#include <gmock/gmock.h>
|
|
||||||
#include <gtest/gtest.h>
|
|
||||||
|
|
||||||
namespace yaze {
|
|
||||||
namespace test {
|
|
||||||
|
|
||||||
using testing::_;
|
|
||||||
using testing::Return;
|
|
||||||
using yaze::emu::Apu;
|
|
||||||
using yaze::emu::MemoryImpl;
|
|
||||||
|
|
||||||
class IplHandshakeTest : public ::testing::Test {
|
|
||||||
protected:
|
|
||||||
void SetUp() override {
|
|
||||||
memory_ = std::make_unique<MemoryImpl>();
|
|
||||||
apu_ = std::make_unique<Apu>(*memory_);
|
|
||||||
apu_->Init();
|
|
||||||
}
|
|
||||||
|
|
||||||
std::unique_ptr<MemoryImpl> memory_;
|
|
||||||
std::unique_ptr<Apu> apu_;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Test IPL ROM handshake timing with exact cycle counts
|
|
||||||
TEST_F(IplHandshakeTest, ExactCycleTiming) {
|
|
||||||
// 1. Initial state
|
|
||||||
EXPECT_EQ(apu_->Read(0x00) & 0x80, 0); // Ready bit should be clear
|
|
||||||
|
|
||||||
// 2. Start handshake
|
|
||||||
apu_->Write(0x00, 0x80); // Set control register bit 7
|
|
||||||
|
|
||||||
// 3. Run exact number of cycles for handshake
|
|
||||||
const int expected_cycles = 64; // Expected cycle count for handshake
|
|
||||||
apu_->RunCycles(expected_cycles);
|
|
||||||
|
|
||||||
// 4. Verify handshake completed
|
|
||||||
EXPECT_TRUE(apu_->Read(0x00) & 0x80); // Ready bit should be set
|
|
||||||
EXPECT_EQ(apu_->GetStatus() & 0x80, 0x80); // Ready bit in status register
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test IPL ROM handshake timing with cycle range
|
|
||||||
TEST_F(IplHandshakeTest, CycleRange) {
|
|
||||||
// 1. Initial state
|
|
||||||
EXPECT_EQ(apu_->Read(0x00) & 0x80, 0); // Ready bit should be clear
|
|
||||||
|
|
||||||
// 2. Start handshake
|
|
||||||
apu_->Write(0x00, 0x80); // Set control register bit 7
|
|
||||||
|
|
||||||
// 3. Wait for handshake with cycle counting
|
|
||||||
int cycles = 0;
|
|
||||||
const int min_cycles = 32; // Minimum expected cycles
|
|
||||||
const int max_cycles = 96; // Maximum expected cycles
|
|
||||||
|
|
||||||
while (!(apu_->Read(0x00) & 0x80) && cycles < max_cycles) {
|
|
||||||
apu_->RunCycles(1);
|
|
||||||
cycles++;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 4. Verify timing constraints
|
|
||||||
EXPECT_GE(cycles, min_cycles); // Should take at least min_cycles
|
|
||||||
EXPECT_LE(cycles, max_cycles); // Should complete within max_cycles
|
|
||||||
EXPECT_TRUE(apu_->Read(0x00) & 0x80); // Ready bit should be set
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test IPL ROM handshake with multiple attempts
|
|
||||||
TEST_F(IplHandshakeTest, MultipleAttempts) {
|
|
||||||
const int num_attempts = 10;
|
|
||||||
std::vector<int> cycle_counts;
|
|
||||||
|
|
||||||
for (int i = 0; i < num_attempts; i++) {
|
|
||||||
// Reset APU
|
|
||||||
apu_->Init();
|
|
||||||
|
|
||||||
// Start handshake
|
|
||||||
apu_->Write(0x00, 0x80);
|
|
||||||
|
|
||||||
// Count cycles until ready
|
|
||||||
int cycles = 0;
|
|
||||||
while (!(apu_->Read(0x00) & 0x80) && cycles < 1000) {
|
|
||||||
apu_->RunCycles(1);
|
|
||||||
cycles++;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Record cycle count
|
|
||||||
cycle_counts.push_back(cycles);
|
|
||||||
|
|
||||||
// Verify handshake completed
|
|
||||||
EXPECT_TRUE(apu_->Read(0x00) & 0x80);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify cycle count consistency
|
|
||||||
int min_cycles = *std::min_element(cycle_counts.begin(), cycle_counts.end());
|
|
||||||
int max_cycles = *std::max_element(cycle_counts.begin(), cycle_counts.end());
|
|
||||||
EXPECT_LE(max_cycles - min_cycles, 2); // Cycle count should be consistent
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test IPL ROM handshake with interrupts
|
|
||||||
TEST_F(IplHandshakeTest, WithInterrupts) {
|
|
||||||
// 1. Initial state
|
|
||||||
EXPECT_EQ(apu_->Read(0x00) & 0x80, 0);
|
|
||||||
|
|
||||||
// 2. Enable interrupts
|
|
||||||
apu_->Write(0x00, 0x80 | 0x40); // Set control register bits 7 and 6
|
|
||||||
|
|
||||||
// 3. Run cycles with interrupts
|
|
||||||
int cycles = 0;
|
|
||||||
while (!(apu_->Read(0x00) & 0x80) && cycles < 1000) {
|
|
||||||
apu_->RunCycles(1);
|
|
||||||
cycles++;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 4. Verify handshake completed
|
|
||||||
EXPECT_TRUE(apu_->Read(0x00) & 0x80);
|
|
||||||
EXPECT_EQ(apu_->GetStatus() & 0x80, 0x80);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace test
|
|
||||||
} // namespace yaze
|
|
||||||
4195
test/emu/cpu_test.cc
4195
test/emu/cpu_test.cc
File diff suppressed because it is too large
Load Diff
@@ -1,54 +0,0 @@
|
|||||||
#include "app/emu/video/ppu.h"
|
|
||||||
|
|
||||||
#include <gmock/gmock.h>
|
|
||||||
|
|
||||||
#include "mocks/mock_memory.h"
|
|
||||||
|
|
||||||
namespace yaze {
|
|
||||||
namespace test {
|
|
||||||
|
|
||||||
using yaze::emu::MockMemory;
|
|
||||||
using yaze::emu::BackgroundMode;
|
|
||||||
using yaze::emu::PpuInterface;
|
|
||||||
using yaze::emu::SpriteAttributes;
|
|
||||||
using yaze::emu::Tilemap;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Mock Ppu class for testing
|
|
||||||
*/
|
|
||||||
class MockPpu : public PpuInterface {
|
|
||||||
public:
|
|
||||||
MOCK_METHOD(void, Write, (uint16_t address, uint8_t data), (override));
|
|
||||||
MOCK_METHOD(uint8_t, Read, (uint16_t address), (const, override));
|
|
||||||
|
|
||||||
std::vector<uint8_t> internalFrameBuffer;
|
|
||||||
std::vector<uint8_t> vram;
|
|
||||||
std::vector<SpriteAttributes> sprites;
|
|
||||||
std::vector<Tilemap> tilemaps;
|
|
||||||
BackgroundMode bgMode;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \test Test fixture for PPU unit tests
|
|
||||||
*/
|
|
||||||
class PpuTest : public ::testing::Test {
|
|
||||||
protected:
|
|
||||||
MockMemory mock_memory;
|
|
||||||
MockPpu mock_ppu;
|
|
||||||
|
|
||||||
PpuTest() {}
|
|
||||||
|
|
||||||
void SetUp() override {
|
|
||||||
ON_CALL(mock_ppu, Write(::testing::_, ::testing::_))
|
|
||||||
.WillByDefault([this](uint16_t address, uint8_t data) {
|
|
||||||
mock_ppu.vram[address] = data;
|
|
||||||
});
|
|
||||||
|
|
||||||
ON_CALL(mock_ppu, Read(::testing::_))
|
|
||||||
.WillByDefault(
|
|
||||||
[this](uint16_t address) { return mock_ppu.vram[address]; });
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace test
|
|
||||||
} // namespace yaze
|
|
||||||
@@ -1,474 +0,0 @@
|
|||||||
#include "app/emu/audio/spc700.h"
|
|
||||||
|
|
||||||
#include <gmock/gmock-nice-strict.h>
|
|
||||||
#include <gmock/gmock.h>
|
|
||||||
#include <gtest/gtest.h>
|
|
||||||
|
|
||||||
namespace yaze {
|
|
||||||
namespace test {
|
|
||||||
|
|
||||||
using testing::_;
|
|
||||||
using testing::Return;
|
|
||||||
using yaze::emu::ApuCallbacks;
|
|
||||||
using yaze::emu::AudioRam;
|
|
||||||
using yaze::emu::Spc700;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief MockAudioRam is a mock class for the AudioRam class.
|
|
||||||
*/
|
|
||||||
class MockAudioRam : public AudioRam {
|
|
||||||
public:
|
|
||||||
MOCK_METHOD(void, reset, (), (override));
|
|
||||||
MOCK_METHOD(uint8_t, read, (uint16_t address), (const, override));
|
|
||||||
MOCK_METHOD(uint8_t&, mutable_read, (uint16_t address), (override));
|
|
||||||
MOCK_METHOD(void, write, (uint16_t address, uint8_t value), (override));
|
|
||||||
|
|
||||||
void SetupMemory(uint16_t address, const std::vector<uint8_t>& values) {
|
|
||||||
if (address > internal_audio_ram_.size()) {
|
|
||||||
internal_audio_ram_.resize(address + values.size());
|
|
||||||
}
|
|
||||||
int i = 0;
|
|
||||||
for (const auto& each : values) {
|
|
||||||
internal_audio_ram_[address + i] = each;
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void SetUp() {
|
|
||||||
// internal_audio_ram_.resize(0x10000); // 64 K (0x10000)
|
|
||||||
// std::fill(internal_audio_ram_.begin(), internal_audio_ram_.end(), 0);
|
|
||||||
ON_CALL(*this, read(_)).WillByDefault([this](uint16_t address) {
|
|
||||||
return internal_audio_ram_[address];
|
|
||||||
});
|
|
||||||
ON_CALL(*this, mutable_read(_))
|
|
||||||
.WillByDefault([this](uint16_t address) -> uint8_t& {
|
|
||||||
return internal_audio_ram_[address];
|
|
||||||
});
|
|
||||||
ON_CALL(*this, write(_, _))
|
|
||||||
.WillByDefault([this](uint16_t address, uint8_t value) {
|
|
||||||
internal_audio_ram_[address] = value;
|
|
||||||
});
|
|
||||||
ON_CALL(*this, reset()).WillByDefault([this]() {
|
|
||||||
std::fill(internal_audio_ram_.begin(), internal_audio_ram_.end(), 0);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<uint8_t> internal_audio_ram_ = std::vector<uint8_t>(0x10000, 0);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \test Spc700Test is a test fixture for the Spc700 class.
|
|
||||||
*/
|
|
||||||
class Spc700Test : public ::testing::Test {
|
|
||||||
public:
|
|
||||||
Spc700Test() = default;
|
|
||||||
void SetUp() override {
|
|
||||||
// Set up the mock
|
|
||||||
audioRAM.SetUp();
|
|
||||||
|
|
||||||
// Set the Spc700 to bank 01
|
|
||||||
spc700.PC = 0x0100;
|
|
||||||
}
|
|
||||||
|
|
||||||
testing::StrictMock<MockAudioRam> audioRAM;
|
|
||||||
ApuCallbacks callbacks_;
|
|
||||||
Spc700 spc700{callbacks_};
|
|
||||||
};
|
|
||||||
|
|
||||||
// ========================================================
|
|
||||||
// 8-bit Move Memory to Register
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, MOV_A_Immediate) {
|
|
||||||
// MOV A, imm
|
|
||||||
uint8_t opcode = 0xE8;
|
|
||||||
uint8_t immediate_value = 0x5A;
|
|
||||||
audioRAM.SetupMemory(0x0100, {opcode, immediate_value});
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_)).WillOnce(Return(immediate_value));
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, immediate_value);
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 0);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, MOV_A_X) {
|
|
||||||
// MOV A, X
|
|
||||||
uint8_t opcode = 0x7D;
|
|
||||||
spc700.X = 0x5A;
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, spc700.X);
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 0);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, MOV_A_Y) {
|
|
||||||
// MOV A, Y
|
|
||||||
uint8_t opcode = 0xDD;
|
|
||||||
spc700.Y = 0x5A;
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, spc700.Y);
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 0);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, MOV_A_dp) {
|
|
||||||
// MOV A, dp
|
|
||||||
uint8_t opcode = 0xE4;
|
|
||||||
uint8_t dp_value = 0x5A;
|
|
||||||
audioRAM.SetupMemory(0x005A, {0x42});
|
|
||||||
audioRAM.SetupMemory(0x0100, {opcode, dp_value});
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_))
|
|
||||||
.WillOnce(Return(dp_value))
|
|
||||||
.WillOnce(Return(0x42));
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, 0x42);
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 0);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, MOV_A_dp_plus_x) {
|
|
||||||
// MOV A, dp+X
|
|
||||||
uint8_t opcode = 0xF4;
|
|
||||||
uint8_t dp_value = 0x5A;
|
|
||||||
spc700.X = 0x01;
|
|
||||||
audioRAM.SetupMemory(0x005B, {0x42});
|
|
||||||
audioRAM.SetupMemory(0x0100, {opcode, dp_value});
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_))
|
|
||||||
.WillOnce(Return(dp_value + spc700.X))
|
|
||||||
.WillOnce(Return(0x42));
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, 0x42);
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 0);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, MOV_A_dp_indirect_plus_y) {
|
|
||||||
// MOV A, [dp]+Y
|
|
||||||
uint8_t opcode = 0xF7;
|
|
||||||
uint8_t dp_value = 0x5A;
|
|
||||||
spc700.Y = 0x01;
|
|
||||||
audioRAM.SetupMemory(0x005A, {0x00, 0x42});
|
|
||||||
audioRAM.SetupMemory(0x0100, {opcode, dp_value});
|
|
||||||
audioRAM.SetupMemory(0x4201, {0x69});
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_))
|
|
||||||
.WillOnce(Return(dp_value))
|
|
||||||
.WillOnce(Return(0x4200))
|
|
||||||
.WillOnce(Return(0x69));
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, 0x69);
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 0);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, MOV_A_dp_plus_x_indirect) {
|
|
||||||
// MOV A, [dp+X]
|
|
||||||
uint8_t opcode = 0xE7;
|
|
||||||
uint8_t dp_value = 0x5A;
|
|
||||||
spc700.X = 0x01;
|
|
||||||
audioRAM.SetupMemory(0x005B, {0x00, 0x42});
|
|
||||||
audioRAM.SetupMemory(0x0100, {opcode, dp_value});
|
|
||||||
audioRAM.SetupMemory(0x4200, {0x69});
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_))
|
|
||||||
.WillOnce(Return(dp_value + 1))
|
|
||||||
.WillOnce(Return(0x4200))
|
|
||||||
.WillOnce(Return(0x69));
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, 0x69);
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 0);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, MOV_A_abs) {
|
|
||||||
// MOV A, !abs
|
|
||||||
uint8_t opcode = 0xE5;
|
|
||||||
uint16_t abs_addr = 0x1234;
|
|
||||||
uint8_t abs_value = 0x5A;
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_))
|
|
||||||
.WillOnce(Return(abs_addr & 0xFF)) // Low byte
|
|
||||||
.WillOnce(Return(abs_addr >> 8)); // High byte
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(abs_addr)).WillOnce(Return(abs_value));
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, abs_value);
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 0);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
// ============================================================================
|
|
||||||
// 8-bit Move Register to Memory
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, MOV_Immediate) {
|
|
||||||
// MOV A, imm
|
|
||||||
uint8_t opcode = 0xE8;
|
|
||||||
uint8_t immediate_value = 0x5A;
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_)).WillOnce(Return(immediate_value));
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, immediate_value);
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 0);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
// ============================================================================
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, NOP_DoesNothing) {
|
|
||||||
// NOP opcode
|
|
||||||
uint8_t opcode = 0x00;
|
|
||||||
|
|
||||||
uint16_t initialPC = spc700.PC;
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
// PC should increment by 1, no other changes
|
|
||||||
EXPECT_EQ(spc700.PC, initialPC + 1);
|
|
||||||
// Add checks for other registers if needed
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, ADC_A_Immediate) {
|
|
||||||
// ADC A, #imm
|
|
||||||
uint8_t opcode = 0x88;
|
|
||||||
uint8_t immediate_value = 0x10;
|
|
||||||
spc700.A = 0x01;
|
|
||||||
spc700.PSW.C = 1; // Assume carry is set
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_)).WillOnce(Return(immediate_value));
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
// Verify A, and flags
|
|
||||||
EXPECT_EQ(spc700.A, 0x12); // 0x01 + 0x10 + 1 (carry)
|
|
||||||
// Check for other flags (Z, C, etc.) based on the result
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, BEQ_BranchesIfZeroFlagSet) {
|
|
||||||
// BEQ rel
|
|
||||||
uint8_t opcode = 0xF0;
|
|
||||||
int8_t offset = 0x05;
|
|
||||||
spc700.PSW.Z = 1; // Set Zero flag
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_)).WillOnce(Return(offset));
|
|
||||||
|
|
||||||
uint16_t initialPC = spc700.PC + 1;
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.PC, initialPC + offset);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, STA_Absolute) {
|
|
||||||
// STA !abs
|
|
||||||
uint8_t opcode = 0x85;
|
|
||||||
uint16_t abs_addr = 0x1234;
|
|
||||||
spc700.A = 0x80;
|
|
||||||
|
|
||||||
// Set up the mock to return the address for the absolute addressing
|
|
||||||
EXPECT_CALL(audioRAM, read(_))
|
|
||||||
.WillOnce(Return(abs_addr & 0xFF)) // Low byte
|
|
||||||
.WillOnce(Return(abs_addr >> 8)); // High byte
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, ExecuteADCWithImmediate) {
|
|
||||||
// ADC A, imm
|
|
||||||
uint8_t opcode = 0x88; // Replace with opcode for ADC A, imm
|
|
||||||
uint8_t immediate_value = 0x10;
|
|
||||||
spc700.A = 0x15;
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_)).WillOnce(Return(immediate_value));
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, 0x25); // 0x15 + 0x10
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 0);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
EXPECT_EQ(spc700.PSW.C, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, ExecuteBRA) {
|
|
||||||
// BRA
|
|
||||||
uint8_t opcode = 0x2F;
|
|
||||||
int8_t offset = 0x05;
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_)).WillOnce(Return(offset));
|
|
||||||
|
|
||||||
// rel() moves the PC forward one after read
|
|
||||||
uint16_t initialPC = spc700.PC + 1;
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.PC, initialPC + offset);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, ReadFromAudioRAM) {
|
|
||||||
uint16_t address = 0x1234;
|
|
||||||
uint8_t expected_value = 0x5A;
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(address)).WillOnce(Return(expected_value));
|
|
||||||
|
|
||||||
uint8_t value = spc700.read(address);
|
|
||||||
EXPECT_EQ(value, expected_value);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, WriteToAudioRAM) {
|
|
||||||
uint16_t address = 0x1234;
|
|
||||||
uint8_t value = 0x5A;
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, write(address, value));
|
|
||||||
|
|
||||||
spc700.write(address, value);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, ExecuteANDWithImmediate) {
|
|
||||||
// AND A, imm
|
|
||||||
uint8_t opcode = 0x28;
|
|
||||||
uint8_t immediate_value = 0x0F;
|
|
||||||
spc700.A = 0x5A; // 0101 1010
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_)).WillOnce(Return(immediate_value));
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, 0x0A); // 0101 1010 & 0000 1111 = 0000 1010
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 0);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, ExecuteORWithImmediate) {
|
|
||||||
// OR A, imm
|
|
||||||
uint8_t opcode = 0x08;
|
|
||||||
uint8_t immediate_value = 0x0F;
|
|
||||||
spc700.A = 0xA0; // 1010 0000
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_)).WillOnce(Return(immediate_value));
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, 0xAF); // 1010 0000 | 0000 1111 = 1010 1111
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 0);
|
|
||||||
// EXPECT_EQ(spc700.PSW.N, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, ExecuteEORWithImmediate) {
|
|
||||||
// EOR A, imm
|
|
||||||
uint8_t opcode = 0x48;
|
|
||||||
uint8_t immediate_value = 0x5A;
|
|
||||||
spc700.A = 0x5A; // 0101 1010
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_)).WillOnce(Return(immediate_value));
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, 0x00); // 0101 1010 ^ 0101 1010 = 0000 0000
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 1);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, ExecuteINC) {
|
|
||||||
// INC A
|
|
||||||
uint8_t opcode = 0xBC;
|
|
||||||
spc700.A = 0xFF;
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, 0x00);
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 1);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, ExecuteDEC) {
|
|
||||||
// DEC A
|
|
||||||
uint8_t opcode = 0x9C;
|
|
||||||
spc700.A = 0x01;
|
|
||||||
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.A, 0x00);
|
|
||||||
EXPECT_EQ(spc700.PSW.Z, 1);
|
|
||||||
EXPECT_EQ(spc700.PSW.N, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, ExecuteBNEWhenNotEqual) {
|
|
||||||
// BNE
|
|
||||||
uint8_t opcode = 0xD0;
|
|
||||||
int8_t offset = 0x05;
|
|
||||||
spc700.PSW.Z = 0;
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_)).WillOnce(Return(offset));
|
|
||||||
|
|
||||||
uint16_t initialPC = spc700.PC + 1;
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.PC, initialPC + offset);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, ExecuteBNEWhenEqual) {
|
|
||||||
// BNE
|
|
||||||
uint8_t opcode = 0xD0;
|
|
||||||
int8_t offset = 0x05;
|
|
||||||
spc700.PSW.Z = 1;
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_)).WillOnce(Return(offset));
|
|
||||||
|
|
||||||
uint16_t initialPC = spc700.PC;
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.PC, initialPC + 1); // +1 because of reading the offset
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, ExecuteBEQWhenEqual) {
|
|
||||||
// BEQ
|
|
||||||
uint8_t opcode = 0xF0;
|
|
||||||
int8_t offset = 0x05;
|
|
||||||
spc700.PSW.Z = 1;
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_)).WillOnce(Return(offset));
|
|
||||||
|
|
||||||
uint16_t initialPC = spc700.PC + 1;
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.PC, initialPC + offset);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, ExecuteBEQWhenNotEqual) {
|
|
||||||
// BEQ
|
|
||||||
uint8_t opcode = 0xF0;
|
|
||||||
int8_t offset = 0x05;
|
|
||||||
spc700.PSW.Z = 0;
|
|
||||||
|
|
||||||
EXPECT_CALL(audioRAM, read(_)).WillOnce(Return(offset));
|
|
||||||
|
|
||||||
uint16_t initialPC = spc700.PC;
|
|
||||||
spc700.ExecuteInstructions(opcode);
|
|
||||||
|
|
||||||
EXPECT_EQ(spc700.PC, initialPC + 1); // +1 because of reading the offset
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(Spc700Test, BootIplRomOk) {
|
|
||||||
// Boot the IPL ROM
|
|
||||||
// spc700.BootIplRom();
|
|
||||||
// EXPECT_EQ(spc700.PC, 0xFFC1 + 0x3F);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace test
|
|
||||||
} // namespace yaze
|
|
||||||
@@ -2,22 +2,207 @@
|
|||||||
|
|
||||||
#include <gtest/gtest.h>
|
#include <gtest/gtest.h>
|
||||||
#include <SDL.h>
|
#include <SDL.h>
|
||||||
|
#include <iostream>
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
#include "absl/debugging/failure_signal_handler.h"
|
#include "absl/debugging/failure_signal_handler.h"
|
||||||
#include "absl/debugging/symbolize.h"
|
#include "absl/debugging/symbolize.h"
|
||||||
#include "test_editor.h"
|
// #include "test_editor.h" // Not used in main
|
||||||
|
|
||||||
|
namespace yaze {
|
||||||
|
namespace test {
|
||||||
|
|
||||||
|
// Test execution modes for AI agents and developers
|
||||||
|
enum class TestMode {
|
||||||
|
kAll, // Run all tests (default)
|
||||||
|
kUnit, // Run only unit tests
|
||||||
|
kIntegration, // Run only integration tests
|
||||||
|
kE2E, // Run only end-to-end tests
|
||||||
|
kRomDependent, // Run ROM-dependent tests only
|
||||||
|
kZSCustomOverworld, // Run ZSCustomOverworld specific tests
|
||||||
|
kCore, // Run core functionality tests
|
||||||
|
kGraphics, // Run graphics-related tests
|
||||||
|
kEditor, // Run editor tests
|
||||||
|
kDeprecated, // Run deprecated tests (for cleanup)
|
||||||
|
kSpecific // Run specific test pattern
|
||||||
|
};
|
||||||
|
|
||||||
|
struct TestConfig {
|
||||||
|
TestMode mode = TestMode::kAll;
|
||||||
|
std::string test_pattern;
|
||||||
|
std::string rom_path = "zelda3.sfc";
|
||||||
|
bool verbose = false;
|
||||||
|
bool skip_rom_tests = false;
|
||||||
|
bool enable_ui_tests = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Parse command line arguments for better AI agent testing support
|
||||||
|
TestConfig ParseArguments(int argc, char* argv[]) {
|
||||||
|
TestConfig config;
|
||||||
|
|
||||||
|
for (int i = 1; i < argc; i++) {
|
||||||
|
std::string arg = argv[i];
|
||||||
|
|
||||||
|
if (arg == "--help" || arg == "-h") {
|
||||||
|
std::cout << "YAZE Test Runner - Enhanced for AI Agent Testing\n\n";
|
||||||
|
std::cout << "Usage: yaze_test [options] [test_pattern]\n\n";
|
||||||
|
std::cout << "Test Modes:\n";
|
||||||
|
std::cout << " --unit Run unit tests only\n";
|
||||||
|
std::cout << " --integration Run integration tests only\n";
|
||||||
|
std::cout << " --e2e Run end-to-end tests only\n";
|
||||||
|
std::cout << " --rom-dependent Run ROM-dependent tests only\n";
|
||||||
|
std::cout << " --zscustomoverworld Run ZSCustomOverworld tests only\n";
|
||||||
|
std::cout << " --core Run core functionality tests\n";
|
||||||
|
std::cout << " --graphics Run graphics tests\n";
|
||||||
|
std::cout << " --editor Run editor tests\n";
|
||||||
|
std::cout << " --deprecated Run deprecated tests\n\n";
|
||||||
|
std::cout << "Options:\n";
|
||||||
|
std::cout << " --rom-path PATH Specify ROM path for testing\n";
|
||||||
|
std::cout << " --skip-rom-tests Skip tests requiring ROM files\n";
|
||||||
|
std::cout << " --enable-ui-tests Enable UI tests (requires display)\n";
|
||||||
|
std::cout << " --verbose Enable verbose output\n";
|
||||||
|
std::cout << " --help Show this help message\n\n";
|
||||||
|
std::cout << "Examples:\n";
|
||||||
|
std::cout << " yaze_test --unit --verbose\n";
|
||||||
|
std::cout << " yaze_test --e2e --rom-path my_rom.sfc\n";
|
||||||
|
std::cout << " yaze_test --zscustomoverworld --verbose\n";
|
||||||
|
std::cout << " yaze_test RomTest.*\n";
|
||||||
|
exit(0);
|
||||||
|
} else if (arg == "--unit") {
|
||||||
|
config.mode = TestMode::kUnit;
|
||||||
|
} else if (arg == "--integration") {
|
||||||
|
config.mode = TestMode::kIntegration;
|
||||||
|
} else if (arg == "--e2e") {
|
||||||
|
config.mode = TestMode::kE2E;
|
||||||
|
} else if (arg == "--rom-dependent") {
|
||||||
|
config.mode = TestMode::kRomDependent;
|
||||||
|
} else if (arg == "--zscustomoverworld") {
|
||||||
|
config.mode = TestMode::kZSCustomOverworld;
|
||||||
|
} else if (arg == "--core") {
|
||||||
|
config.mode = TestMode::kCore;
|
||||||
|
} else if (arg == "--graphics") {
|
||||||
|
config.mode = TestMode::kGraphics;
|
||||||
|
} else if (arg == "--editor") {
|
||||||
|
config.mode = TestMode::kEditor;
|
||||||
|
} else if (arg == "--deprecated") {
|
||||||
|
config.mode = TestMode::kDeprecated;
|
||||||
|
} else if (arg == "--rom-path") {
|
||||||
|
if (i + 1 < argc) {
|
||||||
|
config.rom_path = argv[++i];
|
||||||
|
}
|
||||||
|
} else if (arg == "--skip-rom-tests") {
|
||||||
|
config.skip_rom_tests = true;
|
||||||
|
} else if (arg == "--enable-ui-tests") {
|
||||||
|
config.enable_ui_tests = true;
|
||||||
|
} else if (arg == "--verbose") {
|
||||||
|
config.verbose = true;
|
||||||
|
} else if (arg.find("--") != 0) {
|
||||||
|
// Test pattern (not a flag)
|
||||||
|
config.mode = TestMode::kSpecific;
|
||||||
|
config.test_pattern = arg;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return config;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set up test environment based on configuration
|
||||||
|
void SetupTestEnvironment(const TestConfig& config) {
|
||||||
|
// Set environment variables for tests
|
||||||
|
if (!config.rom_path.empty()) {
|
||||||
|
setenv("YAZE_TEST_ROM_PATH", config.rom_path.c_str(), 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (config.skip_rom_tests) {
|
||||||
|
setenv("YAZE_SKIP_ROM_TESTS", "1", 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (config.enable_ui_tests) {
|
||||||
|
setenv("YAZE_ENABLE_UI_TESTS", "1", 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (config.verbose) {
|
||||||
|
setenv("YAZE_VERBOSE_TESTS", "1", 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Configure Google Test filters based on test mode
|
||||||
|
void ConfigureTestFilters(const TestConfig& config) {
|
||||||
|
std::vector<std::string> filters;
|
||||||
|
|
||||||
|
switch (config.mode) {
|
||||||
|
case TestMode::kUnit:
|
||||||
|
filters.push_back("UnitTest.*");
|
||||||
|
break;
|
||||||
|
case TestMode::kIntegration:
|
||||||
|
filters.push_back("IntegrationTest.*");
|
||||||
|
break;
|
||||||
|
case TestMode::kE2E:
|
||||||
|
filters.push_back("E2ETest.*");
|
||||||
|
break;
|
||||||
|
case TestMode::kRomDependent:
|
||||||
|
filters.push_back("*RomDependent*");
|
||||||
|
break;
|
||||||
|
case TestMode::kZSCustomOverworld:
|
||||||
|
filters.push_back("*ZSCustomOverworld*");
|
||||||
|
break;
|
||||||
|
case TestMode::kCore:
|
||||||
|
filters.push_back("*Core*");
|
||||||
|
filters.push_back("*Asar*");
|
||||||
|
filters.push_back("*Rom*");
|
||||||
|
break;
|
||||||
|
case TestMode::kGraphics:
|
||||||
|
filters.push_back("*Graphics*");
|
||||||
|
filters.push_back("*Gfx*");
|
||||||
|
filters.push_back("*Palette*");
|
||||||
|
filters.push_back("*Tile*");
|
||||||
|
break;
|
||||||
|
case TestMode::kEditor:
|
||||||
|
filters.push_back("*Editor*");
|
||||||
|
break;
|
||||||
|
case TestMode::kDeprecated:
|
||||||
|
filters.push_back("*Deprecated*");
|
||||||
|
break;
|
||||||
|
case TestMode::kSpecific:
|
||||||
|
if (!config.test_pattern.empty()) {
|
||||||
|
filters.push_back(config.test_pattern);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case TestMode::kAll:
|
||||||
|
default:
|
||||||
|
// No filters - run all tests
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!filters.empty()) {
|
||||||
|
std::string filter_string;
|
||||||
|
for (size_t i = 0; i < filters.size(); i++) {
|
||||||
|
if (i > 0) filter_string += ":";
|
||||||
|
filter_string += filters[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
::testing::GTEST_FLAG(filter) = filter_string;
|
||||||
|
|
||||||
|
if (config.verbose) {
|
||||||
|
std::cout << "Test filter: " << filter_string << std::endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace test
|
||||||
|
} // namespace yaze
|
||||||
|
|
||||||
int main(int argc, char* argv[]) {
|
int main(int argc, char* argv[]) {
|
||||||
absl::InitializeSymbolizer(argv[0]);
|
absl::InitializeSymbolizer(argv[0]);
|
||||||
|
|
||||||
// Configure failure signal handler to be less aggressive for testing
|
// Configure failure signal handler to be less aggressive for testing
|
||||||
// This prevents false positives during SDL/graphics cleanup in tests
|
|
||||||
absl::FailureSignalHandlerOptions options;
|
absl::FailureSignalHandlerOptions options;
|
||||||
options.symbolize_stacktrace = true;
|
options.symbolize_stacktrace = true;
|
||||||
options.use_alternate_stack = false; // Avoid conflicts with normal stack during cleanup
|
options.use_alternate_stack = false;
|
||||||
options.alarm_on_failure_secs = false; // Don't set alarms that can trigger on natural leaks
|
options.alarm_on_failure_secs = false;
|
||||||
options.call_previous_handler = true; // Allow system handlers to also run
|
options.call_previous_handler = true;
|
||||||
options.writerfn = nullptr; // Use default writer to avoid custom handling issues
|
options.writerfn = nullptr;
|
||||||
absl::InstallFailureSignalHandler(options);
|
absl::InstallFailureSignalHandler(options);
|
||||||
|
|
||||||
// Initialize SDL to prevent crashes in graphics components
|
// Initialize SDL to prevent crashes in graphics components
|
||||||
@@ -26,20 +211,23 @@ int main(int argc, char* argv[]) {
|
|||||||
// Continue anyway for tests that don't need graphics
|
// Continue anyway for tests that don't need graphics
|
||||||
}
|
}
|
||||||
|
|
||||||
if (argc > 1 && std::string(argv[1]) == "integration") {
|
// Parse command line arguments
|
||||||
return yaze::test::RunIntegrationTest();
|
auto config = yaze::test::ParseArguments(argc, argv);
|
||||||
} else if (argc > 1 && std::string(argv[1]) == "room_object") {
|
|
||||||
::testing::InitGoogleTest(&argc, argv);
|
// Set up test environment
|
||||||
if (!RUN_ALL_TESTS()) {
|
yaze::test::SetupTestEnvironment(config);
|
||||||
return yaze::test::RunIntegrationTest();
|
|
||||||
}
|
// Configure test filters
|
||||||
}
|
yaze::test::ConfigureTestFilters(config);
|
||||||
|
|
||||||
|
// Initialize Google Test
|
||||||
::testing::InitGoogleTest(&argc, argv);
|
::testing::InitGoogleTest(&argc, argv);
|
||||||
|
|
||||||
|
// Run tests
|
||||||
int result = RUN_ALL_TESTS();
|
int result = RUN_ALL_TESTS();
|
||||||
|
|
||||||
// Cleanup SDL
|
// Cleanup SDL
|
||||||
SDL_Quit();
|
SDL_Quit();
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user