Use Slot in EXI devices
This commit is contained in:
parent
3f0b23ed2b
commit
777bb4d82c
|
@ -156,12 +156,13 @@ bool BootCore(std::unique_ptr<BootParameters> boot, const WindowSystemInfo& wsi)
|
||||||
// Movie settings
|
// Movie settings
|
||||||
if (Movie::IsPlayingInput() && Movie::IsConfigSaved())
|
if (Movie::IsPlayingInput() && Movie::IsConfigSaved())
|
||||||
{
|
{
|
||||||
for (int i = 0; i < 2; ++i)
|
for (ExpansionInterface::Slot slot : ExpansionInterface::MEMCARD_SLOTS)
|
||||||
{
|
{
|
||||||
if (Movie::IsUsingMemcard(i) && Movie::IsStartingFromClearSave() && !StartUp.bWii)
|
if (Movie::IsUsingMemcard(slot) && Movie::IsStartingFromClearSave() && !StartUp.bWii)
|
||||||
{
|
{
|
||||||
const auto raw_path =
|
const auto raw_path =
|
||||||
File::GetUserPath(D_GCUSER_IDX) + fmt::format("Movie{}.raw", (i == 0) ? 'A' : 'B');
|
File::GetUserPath(D_GCUSER_IDX) +
|
||||||
|
fmt::format("Movie{}.raw", slot == ExpansionInterface::Slot::A ? 'A' : 'B');
|
||||||
if (File::Exists(raw_path))
|
if (File::Exists(raw_path))
|
||||||
File::Delete(raw_path);
|
File::Delete(raw_path);
|
||||||
|
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
#include <fmt/format.h>
|
#include <fmt/format.h>
|
||||||
|
|
||||||
#include "AudioCommon/AudioCommon.h"
|
#include "AudioCommon/AudioCommon.h"
|
||||||
|
#include "Common/Assert.h"
|
||||||
#include "Common/CommonPaths.h"
|
#include "Common/CommonPaths.h"
|
||||||
#include "Common/Config/Config.h"
|
#include "Common/Config/Config.h"
|
||||||
#include "Common/EnumMap.h"
|
#include "Common/EnumMap.h"
|
||||||
|
@ -50,12 +51,42 @@ const Info<bool> MAIN_AUDIO_STRETCH{{System::Main, "Core", "AudioStretch"}, fals
|
||||||
const Info<int> MAIN_AUDIO_STRETCH_LATENCY{{System::Main, "Core", "AudioStretchMaxLatency"}, 80};
|
const Info<int> MAIN_AUDIO_STRETCH_LATENCY{{System::Main, "Core", "AudioStretchMaxLatency"}, 80};
|
||||||
const Info<std::string> MAIN_MEMCARD_A_PATH{{System::Main, "Core", "MemcardAPath"}, ""};
|
const Info<std::string> MAIN_MEMCARD_A_PATH{{System::Main, "Core", "MemcardAPath"}, ""};
|
||||||
const Info<std::string> MAIN_MEMCARD_B_PATH{{System::Main, "Core", "MemcardBPath"}, ""};
|
const Info<std::string> MAIN_MEMCARD_B_PATH{{System::Main, "Core", "MemcardBPath"}, ""};
|
||||||
|
const Info<std::string>& GetInfoForMemcardPath(ExpansionInterface::Slot slot)
|
||||||
|
{
|
||||||
|
ASSERT(ExpansionInterface::IsMemcardSlot(slot));
|
||||||
|
static constexpr Common::EnumMap<const Info<std::string>*, ExpansionInterface::MAX_MEMCARD_SLOT>
|
||||||
|
infos{
|
||||||
|
&MAIN_MEMCARD_A_PATH,
|
||||||
|
&MAIN_MEMCARD_B_PATH,
|
||||||
|
};
|
||||||
|
return *infos[slot];
|
||||||
|
}
|
||||||
const Info<std::string> MAIN_AGP_CART_A_PATH{{System::Main, "Core", "AgpCartAPath"}, ""};
|
const Info<std::string> MAIN_AGP_CART_A_PATH{{System::Main, "Core", "AgpCartAPath"}, ""};
|
||||||
const Info<std::string> MAIN_AGP_CART_B_PATH{{System::Main, "Core", "AgpCartBPath"}, ""};
|
const Info<std::string> MAIN_AGP_CART_B_PATH{{System::Main, "Core", "AgpCartBPath"}, ""};
|
||||||
|
const Info<std::string>& GetInfoForAGPCartPath(ExpansionInterface::Slot slot)
|
||||||
|
{
|
||||||
|
ASSERT(ExpansionInterface::IsMemcardSlot(slot));
|
||||||
|
static constexpr Common::EnumMap<const Info<std::string>*, ExpansionInterface::MAX_MEMCARD_SLOT>
|
||||||
|
infos{
|
||||||
|
&MAIN_AGP_CART_A_PATH,
|
||||||
|
&MAIN_AGP_CART_B_PATH,
|
||||||
|
};
|
||||||
|
return *infos[slot];
|
||||||
|
}
|
||||||
const Info<std::string> MAIN_GCI_FOLDER_A_PATH_OVERRIDE{
|
const Info<std::string> MAIN_GCI_FOLDER_A_PATH_OVERRIDE{
|
||||||
{System::Main, "Core", "GCIFolderAPathOverride"}, ""};
|
{System::Main, "Core", "GCIFolderAPathOverride"}, ""};
|
||||||
const Info<std::string> MAIN_GCI_FOLDER_B_PATH_OVERRIDE{
|
const Info<std::string> MAIN_GCI_FOLDER_B_PATH_OVERRIDE{
|
||||||
{System::Main, "Core", "GCIFolderBPathOverride"}, ""};
|
{System::Main, "Core", "GCIFolderBPathOverride"}, ""};
|
||||||
|
const Info<std::string>& GetInfoForGCIPathOverride(ExpansionInterface::Slot slot)
|
||||||
|
{
|
||||||
|
ASSERT(ExpansionInterface::IsMemcardSlot(slot));
|
||||||
|
static constexpr Common::EnumMap<const Info<std::string>*, ExpansionInterface::MAX_MEMCARD_SLOT>
|
||||||
|
infos{
|
||||||
|
&MAIN_GCI_FOLDER_A_PATH_OVERRIDE,
|
||||||
|
&MAIN_GCI_FOLDER_B_PATH_OVERRIDE,
|
||||||
|
};
|
||||||
|
return *infos[slot];
|
||||||
|
}
|
||||||
|
|
||||||
const Info<ExpansionInterface::EXIDeviceType> MAIN_SLOT_A{
|
const Info<ExpansionInterface::EXIDeviceType> MAIN_SLOT_A{
|
||||||
{System::Main, "Core", "SlotA"}, ExpansionInterface::EXIDeviceType::MemoryCardFolder};
|
{System::Main, "Core", "SlotA"}, ExpansionInterface::EXIDeviceType::MemoryCardFolder};
|
||||||
|
|
|
@ -66,10 +66,13 @@ extern const Info<bool> MAIN_AUDIO_STRETCH;
|
||||||
extern const Info<int> MAIN_AUDIO_STRETCH_LATENCY;
|
extern const Info<int> MAIN_AUDIO_STRETCH_LATENCY;
|
||||||
extern const Info<std::string> MAIN_MEMCARD_A_PATH;
|
extern const Info<std::string> MAIN_MEMCARD_A_PATH;
|
||||||
extern const Info<std::string> MAIN_MEMCARD_B_PATH;
|
extern const Info<std::string> MAIN_MEMCARD_B_PATH;
|
||||||
|
const Info<std::string>& GetInfoForMemcardPath(ExpansionInterface::Slot slot);
|
||||||
extern const Info<std::string> MAIN_AGP_CART_A_PATH;
|
extern const Info<std::string> MAIN_AGP_CART_A_PATH;
|
||||||
extern const Info<std::string> MAIN_AGP_CART_B_PATH;
|
extern const Info<std::string> MAIN_AGP_CART_B_PATH;
|
||||||
|
const Info<std::string>& GetInfoForAGPCartPath(ExpansionInterface::Slot slot);
|
||||||
extern const Info<std::string> MAIN_GCI_FOLDER_A_PATH_OVERRIDE;
|
extern const Info<std::string> MAIN_GCI_FOLDER_A_PATH_OVERRIDE;
|
||||||
extern const Info<std::string> MAIN_GCI_FOLDER_B_PATH_OVERRIDE;
|
extern const Info<std::string> MAIN_GCI_FOLDER_B_PATH_OVERRIDE;
|
||||||
|
const Info<std::string>& GetInfoForGCIPathOverride(ExpansionInterface::Slot slot);
|
||||||
extern const Info<ExpansionInterface::EXIDeviceType> MAIN_SLOT_A;
|
extern const Info<ExpansionInterface::EXIDeviceType> MAIN_SLOT_A;
|
||||||
extern const Info<ExpansionInterface::EXIDeviceType> MAIN_SLOT_B;
|
extern const Info<ExpansionInterface::EXIDeviceType> MAIN_SLOT_B;
|
||||||
extern const Info<ExpansionInterface::EXIDeviceType> MAIN_SERIAL_PORT_1;
|
extern const Info<ExpansionInterface::EXIDeviceType> MAIN_SERIAL_PORT_1;
|
||||||
|
|
|
@ -39,15 +39,14 @@ static void UpdateInterruptsCallback(u64 userdata, s64 cycles_late);
|
||||||
|
|
||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
void AddMemoryCards(int i)
|
void AddMemoryCard(Slot slot)
|
||||||
{
|
{
|
||||||
EXIDeviceType memorycard_device;
|
EXIDeviceType memorycard_device;
|
||||||
if (Movie::IsPlayingInput() && Movie::IsConfigSaved())
|
if (Movie::IsPlayingInput() && Movie::IsConfigSaved())
|
||||||
{
|
{
|
||||||
if (Movie::IsUsingMemcard(i))
|
if (Movie::IsUsingMemcard(slot))
|
||||||
{
|
{
|
||||||
if (Config::Get(Config::GetInfoForEXIDevice(static_cast<Slot>(i))) ==
|
if (Config::Get(Config::GetInfoForEXIDevice(slot)) == EXIDeviceType::MemoryCardFolder)
|
||||||
EXIDeviceType::MemoryCardFolder)
|
|
||||||
memorycard_device = EXIDeviceType::MemoryCardFolder;
|
memorycard_device = EXIDeviceType::MemoryCardFolder;
|
||||||
else
|
else
|
||||||
memorycard_device = EXIDeviceType::MemoryCard;
|
memorycard_device = EXIDeviceType::MemoryCard;
|
||||||
|
@ -59,10 +58,10 @@ void AddMemoryCards(int i)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
memorycard_device = Config::Get(Config::GetInfoForEXIDevice(static_cast<Slot>(i)));
|
memorycard_device = Config::Get(Config::GetInfoForEXIDevice(slot));
|
||||||
}
|
}
|
||||||
|
|
||||||
g_Channels[i]->AddDevice(memorycard_device, 0);
|
g_Channels[SlotToEXIChannel(slot)]->AddDevice(memorycard_device, SlotToEXIDevice(slot));
|
||||||
}
|
}
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
|
@ -132,7 +131,7 @@ void Init()
|
||||||
}
|
}
|
||||||
|
|
||||||
for (Slot slot : MEMCARD_SLOTS)
|
for (Slot slot : MEMCARD_SLOTS)
|
||||||
AddMemoryCards(static_cast<int>(slot));
|
AddMemoryCard(slot);
|
||||||
|
|
||||||
g_Channels[0]->AddDevice(EXIDeviceType::MaskROM, 1);
|
g_Channels[0]->AddDevice(EXIDeviceType::MaskROM, 1);
|
||||||
g_Channels[SlotToEXIChannel(Slot::SP1)]->AddDevice(Config::Get(Config::MAIN_SERIAL_PORT_1),
|
g_Channels[SlotToEXIChannel(Slot::SP1)]->AddDevice(Config::Get(Config::MAIN_SERIAL_PORT_1),
|
||||||
|
@ -186,7 +185,12 @@ static void ChangeDeviceCallback(u64 userdata, s64 cyclesLate)
|
||||||
g_Channels.at(channel)->AddDevice(static_cast<EXIDeviceType>(type), num);
|
g_Channels.at(channel)->AddDevice(static_cast<EXIDeviceType>(type), num);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChangeDevice(const u8 channel, const EXIDeviceType device_type, const u8 device_num,
|
void ChangeDevice(Slot slot, EXIDeviceType device_type, CoreTiming::FromThread from_thread)
|
||||||
|
{
|
||||||
|
ChangeDevice(SlotToEXIChannel(slot), SlotToEXIDevice(slot), device_type, from_thread);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ChangeDevice(u8 channel, u8 device_num, EXIDeviceType device_type,
|
||||||
CoreTiming::FromThread from_thread)
|
CoreTiming::FromThread from_thread)
|
||||||
{
|
{
|
||||||
// Let the hardware see no device for 1 second
|
// Let the hardware see no device for 1 second
|
||||||
|
@ -203,15 +207,9 @@ CEXIChannel* GetChannel(u32 index)
|
||||||
return g_Channels.at(index).get();
|
return g_Channels.at(index).get();
|
||||||
}
|
}
|
||||||
|
|
||||||
IEXIDevice* FindDevice(EXIDeviceType device_type, int customIndex)
|
IEXIDevice* GetDevice(Slot slot)
|
||||||
{
|
{
|
||||||
for (auto& channel : g_Channels)
|
return g_Channels.at(SlotToEXIChannel(slot))->GetDevice(1 << SlotToEXIDevice(slot));
|
||||||
{
|
|
||||||
IEXIDevice* device = channel->FindDevice(device_type, customIndex);
|
|
||||||
if (device)
|
|
||||||
return device;
|
|
||||||
}
|
|
||||||
return nullptr;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void UpdateInterrupts()
|
void UpdateInterrupts()
|
||||||
|
|
|
@ -40,6 +40,11 @@ enum class Slot : int
|
||||||
static constexpr auto SLOTS = {Slot::A, Slot::B, Slot::SP1};
|
static constexpr auto SLOTS = {Slot::A, Slot::B, Slot::SP1};
|
||||||
static constexpr auto MAX_SLOT = Slot::SP1;
|
static constexpr auto MAX_SLOT = Slot::SP1;
|
||||||
static constexpr auto MEMCARD_SLOTS = {Slot::A, Slot::B};
|
static constexpr auto MEMCARD_SLOTS = {Slot::A, Slot::B};
|
||||||
|
static constexpr auto MAX_MEMCARD_SLOT = Slot::B;
|
||||||
|
constexpr bool IsMemcardSlot(Slot slot)
|
||||||
|
{
|
||||||
|
return slot == Slot::A || slot == Slot::B;
|
||||||
|
}
|
||||||
|
|
||||||
u8 SlotToEXIChannel(Slot slot);
|
u8 SlotToEXIChannel(Slot slot);
|
||||||
u8 SlotToEXIDevice(Slot slot);
|
u8 SlotToEXIDevice(Slot slot);
|
||||||
|
@ -54,12 +59,13 @@ void RegisterMMIO(MMIO::Mapping* mmio, u32 base);
|
||||||
void UpdateInterrupts();
|
void UpdateInterrupts();
|
||||||
void ScheduleUpdateInterrupts(CoreTiming::FromThread from, int cycles_late);
|
void ScheduleUpdateInterrupts(CoreTiming::FromThread from, int cycles_late);
|
||||||
|
|
||||||
void ChangeDevice(const u8 channel, const EXIDeviceType device_type, const u8 device_num,
|
void ChangeDevice(Slot slot, EXIDeviceType device_type,
|
||||||
|
CoreTiming::FromThread from_thread = CoreTiming::FromThread::NON_CPU);
|
||||||
|
void ChangeDevice(u8 channel, u8 device_num, EXIDeviceType device_type,
|
||||||
CoreTiming::FromThread from_thread = CoreTiming::FromThread::NON_CPU);
|
CoreTiming::FromThread from_thread = CoreTiming::FromThread::NON_CPU);
|
||||||
|
|
||||||
CEXIChannel* GetChannel(u32 index);
|
CEXIChannel* GetChannel(u32 index);
|
||||||
|
IEXIDevice* GetDevice(Slot slot);
|
||||||
IEXIDevice* FindDevice(EXIDeviceType device_type, int customIndex = -1);
|
|
||||||
|
|
||||||
} // namespace ExpansionInterface
|
} // namespace ExpansionInterface
|
||||||
|
|
||||||
|
|
|
@ -278,7 +278,7 @@ void CEXIChannel::DoState(PointerWrap& p)
|
||||||
// the new device type are identical in this case. I assume there is a reason we have this
|
// the new device type are identical in this case. I assume there is a reason we have this
|
||||||
// grace period when switching in the GUI.
|
// grace period when switching in the GUI.
|
||||||
AddDevice(EXIDeviceType::None, device_index);
|
AddDevice(EXIDeviceType::None, device_index);
|
||||||
ExpansionInterface::ChangeDevice(m_channel_id, EXIDeviceType::MemoryCardFolder, device_index,
|
ExpansionInterface::ChangeDevice(m_channel_id, device_index, EXIDeviceType::MemoryCardFolder,
|
||||||
CoreTiming::FromThread::CPU);
|
CoreTiming::FromThread::CPU);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -294,15 +294,4 @@ void CEXIChannel::SetEXIINT(bool exiint)
|
||||||
{
|
{
|
||||||
m_status.EXIINT = !!exiint;
|
m_status.EXIINT = !!exiint;
|
||||||
}
|
}
|
||||||
|
|
||||||
IEXIDevice* CEXIChannel::FindDevice(EXIDeviceType device_type, int custom_index)
|
|
||||||
{
|
|
||||||
for (auto& sup : m_devices)
|
|
||||||
{
|
|
||||||
IEXIDevice* device = sup->FindDevice(device_type, custom_index);
|
|
||||||
if (device)
|
|
||||||
return device;
|
|
||||||
}
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
} // namespace ExpansionInterface
|
} // namespace ExpansionInterface
|
||||||
|
|
|
@ -30,7 +30,6 @@ public:
|
||||||
|
|
||||||
// get device
|
// get device
|
||||||
IEXIDevice* GetDevice(u8 chip_select);
|
IEXIDevice* GetDevice(u8 chip_select);
|
||||||
IEXIDevice* FindDevice(EXIDeviceType device_type, int custom_index = -1);
|
|
||||||
|
|
||||||
void RegisterMMIO(MMIO::Mapping* mmio, u32 base);
|
void RegisterMMIO(MMIO::Mapping* mmio, u32 base);
|
||||||
|
|
||||||
|
|
|
@ -64,11 +64,6 @@ void IEXIDevice::DMARead(u32 address, u32 size)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
IEXIDevice* IEXIDevice::FindDevice(EXIDeviceType device_type, int custom_index)
|
|
||||||
{
|
|
||||||
return (device_type == m_device_type) ? this : nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IEXIDevice::UseDelayedTransferCompletion() const
|
bool IEXIDevice::UseDelayedTransferCompletion() const
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
|
@ -105,6 +100,9 @@ std::unique_ptr<IEXIDevice> EXIDevice_Create(const EXIDeviceType device_type, co
|
||||||
const Memcard::HeaderData& memcard_header_data)
|
const Memcard::HeaderData& memcard_header_data)
|
||||||
{
|
{
|
||||||
std::unique_ptr<IEXIDevice> result;
|
std::unique_ptr<IEXIDevice> result;
|
||||||
|
// XXX This computation isn't necessarily right (it holds for A/B, but not SP1)
|
||||||
|
// However, the devices that care about slots currently only go in A/B.
|
||||||
|
const Slot slot = static_cast<Slot>(channel_num);
|
||||||
|
|
||||||
switch (device_type)
|
switch (device_type)
|
||||||
{
|
{
|
||||||
|
@ -116,7 +114,7 @@ std::unique_ptr<IEXIDevice> EXIDevice_Create(const EXIDeviceType device_type, co
|
||||||
case EXIDeviceType::MemoryCardFolder:
|
case EXIDeviceType::MemoryCardFolder:
|
||||||
{
|
{
|
||||||
bool gci_folder = (device_type == EXIDeviceType::MemoryCardFolder);
|
bool gci_folder = (device_type == EXIDeviceType::MemoryCardFolder);
|
||||||
result = std::make_unique<CEXIMemoryCard>(channel_num, gci_folder, memcard_header_data);
|
result = std::make_unique<CEXIMemoryCard>(slot, gci_folder, memcard_header_data);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case EXIDeviceType::MaskROM:
|
case EXIDeviceType::MaskROM:
|
||||||
|
@ -150,7 +148,7 @@ std::unique_ptr<IEXIDevice> EXIDevice_Create(const EXIDeviceType device_type, co
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case EXIDeviceType::AGP:
|
case EXIDeviceType::AGP:
|
||||||
result = std::make_unique<CEXIAgp>(channel_num);
|
result = std::make_unique<CEXIAgp>(slot);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case EXIDeviceType::AMBaseboard:
|
case EXIDeviceType::AMBaseboard:
|
||||||
|
|
|
@ -54,8 +54,6 @@ public:
|
||||||
virtual void DMAWrite(u32 address, u32 size);
|
virtual void DMAWrite(u32 address, u32 size);
|
||||||
virtual void DMARead(u32 address, u32 size);
|
virtual void DMARead(u32 address, u32 size);
|
||||||
|
|
||||||
virtual IEXIDevice* FindDevice(EXIDeviceType device_type, int custom_index = -1);
|
|
||||||
|
|
||||||
virtual bool UseDelayedTransferCompletion() const;
|
virtual bool UseDelayedTransferCompletion() const;
|
||||||
virtual bool IsPresent() const;
|
virtual bool IsPresent() const;
|
||||||
virtual void SetCS(int cs);
|
virtual void SetCS(int cs);
|
||||||
|
|
|
@ -8,18 +8,21 @@
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
|
#include "Common/Assert.h"
|
||||||
#include "Common/ChunkFile.h"
|
#include "Common/ChunkFile.h"
|
||||||
#include "Common/CommonTypes.h"
|
#include "Common/CommonTypes.h"
|
||||||
#include "Common/IOFile.h"
|
#include "Common/IOFile.h"
|
||||||
#include "Common/Logging/Log.h"
|
#include "Common/Logging/Log.h"
|
||||||
#include "Common/StringUtil.h"
|
#include "Common/StringUtil.h"
|
||||||
#include "Core/Config/MainSettings.h"
|
#include "Core/Config/MainSettings.h"
|
||||||
|
#include "Core/HW/EXI/EXI.h"
|
||||||
|
|
||||||
namespace ExpansionInterface
|
namespace ExpansionInterface
|
||||||
{
|
{
|
||||||
CEXIAgp::CEXIAgp(int index)
|
CEXIAgp::CEXIAgp(Slot slot)
|
||||||
{
|
{
|
||||||
m_slot = index;
|
ASSERT(IsMemcardSlot(slot));
|
||||||
|
m_slot = slot;
|
||||||
|
|
||||||
// Create the ROM
|
// Create the ROM
|
||||||
m_rom_size = 0;
|
m_rom_size = 0;
|
||||||
|
@ -35,9 +38,7 @@ CEXIAgp::~CEXIAgp()
|
||||||
std::string filename;
|
std::string filename;
|
||||||
std::string ext;
|
std::string ext;
|
||||||
std::string gbapath;
|
std::string gbapath;
|
||||||
SplitPath(m_slot == 0 ? Config::Get(Config::MAIN_AGP_CART_A_PATH) :
|
SplitPath(Config::Get(Config::GetInfoForAGPCartPath(m_slot)), &path, &filename, &ext);
|
||||||
Config::Get(Config::MAIN_AGP_CART_B_PATH),
|
|
||||||
&path, &filename, &ext);
|
|
||||||
gbapath = path + filename;
|
gbapath = path + filename;
|
||||||
|
|
||||||
SaveFileFromEEPROM(gbapath + ".sav");
|
SaveFileFromEEPROM(gbapath + ".sav");
|
||||||
|
@ -75,9 +76,7 @@ void CEXIAgp::LoadRom()
|
||||||
std::string path;
|
std::string path;
|
||||||
std::string filename;
|
std::string filename;
|
||||||
std::string ext;
|
std::string ext;
|
||||||
SplitPath(m_slot == 0 ? Config::Get(Config::MAIN_AGP_CART_A_PATH) :
|
SplitPath(Config::Get(Config::GetInfoForAGPCartPath(m_slot)), &path, &filename, &ext);
|
||||||
Config::Get(Config::MAIN_AGP_CART_B_PATH),
|
|
||||||
&path, &filename, &ext);
|
|
||||||
const std::string gbapath = path + filename;
|
const std::string gbapath = path + filename;
|
||||||
LoadFileToROM(gbapath + ext);
|
LoadFileToROM(gbapath + ext);
|
||||||
INFO_LOG_FMT(EXPANSIONINTERFACE, "Loaded GBA rom: {} card: {}", gbapath, m_slot);
|
INFO_LOG_FMT(EXPANSIONINTERFACE, "Loaded GBA rom: {} card: {}", gbapath, m_slot);
|
||||||
|
|
|
@ -12,10 +12,12 @@ class PointerWrap;
|
||||||
|
|
||||||
namespace ExpansionInterface
|
namespace ExpansionInterface
|
||||||
{
|
{
|
||||||
|
enum class Slot : int;
|
||||||
|
|
||||||
class CEXIAgp : public IEXIDevice
|
class CEXIAgp : public IEXIDevice
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
CEXIAgp(const int index);
|
CEXIAgp(const Slot slot);
|
||||||
virtual ~CEXIAgp() override;
|
virtual ~CEXIAgp() override;
|
||||||
bool IsPresent() const override { return true; }
|
bool IsPresent() const override { return true; }
|
||||||
void ImmWrite(u32 _uData, u32 _uSize) override;
|
void ImmWrite(u32 _uData, u32 _uSize) override;
|
||||||
|
@ -31,7 +33,7 @@ private:
|
||||||
EE_READ_TRUE = 0xB,
|
EE_READ_TRUE = 0xB,
|
||||||
};
|
};
|
||||||
|
|
||||||
int m_slot;
|
Slot m_slot;
|
||||||
|
|
||||||
//! ROM
|
//! ROM
|
||||||
u32 m_rom_size = 0;
|
u32 m_rom_size = 0;
|
||||||
|
|
|
@ -16,6 +16,7 @@
|
||||||
#include "Common/CommonPaths.h"
|
#include "Common/CommonPaths.h"
|
||||||
#include "Common/CommonTypes.h"
|
#include "Common/CommonTypes.h"
|
||||||
#include "Common/Config/Config.h"
|
#include "Common/Config/Config.h"
|
||||||
|
#include "Common/EnumMap.h"
|
||||||
#include "Common/FileUtil.h"
|
#include "Common/FileUtil.h"
|
||||||
#include "Common/IniFile.h"
|
#include "Common/IniFile.h"
|
||||||
#include "Common/Logging/Log.h"
|
#include "Common/Logging/Log.h"
|
||||||
|
@ -48,25 +49,24 @@ namespace ExpansionInterface
|
||||||
static const u32 MC_TRANSFER_RATE_READ = 512 * 1024;
|
static const u32 MC_TRANSFER_RATE_READ = 512 * 1024;
|
||||||
static const auto MC_TRANSFER_RATE_WRITE = static_cast<u32>(96.125f * 1024.0f);
|
static const auto MC_TRANSFER_RATE_WRITE = static_cast<u32>(96.125f * 1024.0f);
|
||||||
|
|
||||||
static std::array<CoreTiming::EventType*, 2> s_et_cmd_done;
|
static Common::EnumMap<CoreTiming::EventType*, MAX_MEMCARD_SLOT> s_et_cmd_done;
|
||||||
static std::array<CoreTiming::EventType*, 2> s_et_transfer_complete;
|
static Common::EnumMap<CoreTiming::EventType*, MAX_MEMCARD_SLOT> s_et_transfer_complete;
|
||||||
|
static Common::EnumMap<char, MAX_MEMCARD_SLOT> s_card_short_names{'A', 'B'};
|
||||||
|
|
||||||
// Takes care of the nasty recovery of the 'this' pointer from card_index,
|
// Takes care of the nasty recovery of the 'this' pointer from card_slot,
|
||||||
// stored in the userdata parameter of the CoreTiming event.
|
// stored in the userdata parameter of the CoreTiming event.
|
||||||
void CEXIMemoryCard::EventCompleteFindInstance(u64 userdata,
|
void CEXIMemoryCard::EventCompleteFindInstance(u64 userdata,
|
||||||
std::function<void(CEXIMemoryCard*)> callback)
|
std::function<void(CEXIMemoryCard*)> callback)
|
||||||
{
|
{
|
||||||
int card_index = (int)userdata;
|
Slot card_slot = static_cast<Slot>(userdata);
|
||||||
auto* self = static_cast<CEXIMemoryCard*>(
|
IEXIDevice* self = ExpansionInterface::GetDevice(card_slot);
|
||||||
ExpansionInterface::FindDevice(EXIDeviceType::MemoryCard, card_index));
|
if (self != nullptr)
|
||||||
if (self == nullptr)
|
|
||||||
{
|
{
|
||||||
self = static_cast<CEXIMemoryCard*>(
|
if (self->m_device_type == EXIDeviceType::MemoryCard ||
|
||||||
ExpansionInterface::FindDevice(EXIDeviceType::MemoryCardFolder, card_index));
|
self->m_device_type == EXIDeviceType::MemoryCardFolder)
|
||||||
}
|
{
|
||||||
if (self)
|
callback(static_cast<CEXIMemoryCard*>(self));
|
||||||
{
|
}
|
||||||
callback(self);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -83,19 +83,15 @@ void CEXIMemoryCard::TransferCompleteCallback(u64 userdata, s64)
|
||||||
|
|
||||||
void CEXIMemoryCard::Init()
|
void CEXIMemoryCard::Init()
|
||||||
{
|
{
|
||||||
static constexpr char DONE_PREFIX[] = "memcardDone";
|
|
||||||
static constexpr char TRANSFER_COMPLETE_PREFIX[] = "memcardTransferComplete";
|
|
||||||
|
|
||||||
static_assert(s_et_cmd_done.size() == s_et_transfer_complete.size(), "Event array size differs");
|
static_assert(s_et_cmd_done.size() == s_et_transfer_complete.size(), "Event array size differs");
|
||||||
for (unsigned int i = 0; i < s_et_cmd_done.size(); ++i)
|
static_assert(s_et_cmd_done.size() == MEMCARD_SLOTS.size(), "Event array size differs");
|
||||||
|
for (Slot slot : MEMCARD_SLOTS)
|
||||||
{
|
{
|
||||||
std::string name = DONE_PREFIX;
|
s_et_cmd_done[slot] = CoreTiming::RegisterEvent(
|
||||||
name += static_cast<char>('A' + i);
|
fmt::format("memcardDone{}", s_card_short_names[slot]), CmdDoneCallback);
|
||||||
s_et_cmd_done[i] = CoreTiming::RegisterEvent(name, CmdDoneCallback);
|
s_et_transfer_complete[slot] = CoreTiming::RegisterEvent(
|
||||||
|
fmt::format("memcardTransferComplete{}", s_card_short_names[slot]),
|
||||||
name = TRANSFER_COMPLETE_PREFIX;
|
TransferCompleteCallback);
|
||||||
name += static_cast<char>('A' + i);
|
|
||||||
s_et_transfer_complete[i] = CoreTiming::RegisterEvent(name, TransferCompleteCallback);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -105,12 +101,12 @@ void CEXIMemoryCard::Shutdown()
|
||||||
s_et_transfer_complete.fill(nullptr);
|
s_et_transfer_complete.fill(nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
CEXIMemoryCard::CEXIMemoryCard(const int index, bool gci_folder,
|
CEXIMemoryCard::CEXIMemoryCard(const Slot slot, bool gci_folder,
|
||||||
const Memcard::HeaderData& header_data)
|
const Memcard::HeaderData& header_data)
|
||||||
: m_card_index(index)
|
: m_card_slot(slot)
|
||||||
{
|
{
|
||||||
ASSERT_MSG(EXPANSIONINTERFACE, static_cast<std::size_t>(index) < s_et_cmd_done.size(),
|
ASSERT_MSG(EXPANSIONINTERFACE, IsMemcardSlot(slot), "Trying to create invalid memory card in {}.",
|
||||||
"Trying to create invalid memory card index {}.", index);
|
slot);
|
||||||
|
|
||||||
// NOTE: When loading a save state, DMA completion callbacks (s_et_transfer_complete) and such
|
// NOTE: When loading a save state, DMA completion callbacks (s_et_transfer_complete) and such
|
||||||
// may have been restored, we need to anticipate those arriving.
|
// may have been restored, we need to anticipate those arriving.
|
||||||
|
@ -145,15 +141,13 @@ CEXIMemoryCard::CEXIMemoryCard(const int index, bool gci_folder,
|
||||||
m_memory_card_size = m_memory_card->GetCardId() * SIZE_TO_Mb;
|
m_memory_card_size = m_memory_card->GetCardId() * SIZE_TO_Mb;
|
||||||
std::array<u8, 20> header{};
|
std::array<u8, 20> header{};
|
||||||
m_memory_card->Read(0, static_cast<s32>(header.size()), header.data());
|
m_memory_card->Read(0, static_cast<s32>(header.size()), header.data());
|
||||||
SetCardFlashID(header.data(), m_card_index);
|
SetCardFlashID(header.data(), m_card_slot);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::pair<std::string /* path */, bool /* migrate */>
|
std::pair<std::string /* path */, bool /* migrate */>
|
||||||
CEXIMemoryCard::GetGCIFolderPath(int card_index, AllowMovieFolder allow_movie_folder)
|
CEXIMemoryCard::GetGCIFolderPath(Slot card_slot, AllowMovieFolder allow_movie_folder)
|
||||||
{
|
{
|
||||||
std::string path_override =
|
std::string path_override = Config::Get(Config::GetInfoForGCIPathOverride(card_slot));
|
||||||
Config::Get(card_index == 0 ? Config::MAIN_GCI_FOLDER_A_PATH_OVERRIDE :
|
|
||||||
Config::MAIN_GCI_FOLDER_B_PATH_OVERRIDE);
|
|
||||||
|
|
||||||
if (!path_override.empty())
|
if (!path_override.empty())
|
||||||
return {std::move(path_override), false};
|
return {std::move(path_override), false};
|
||||||
|
@ -162,7 +156,7 @@ CEXIMemoryCard::GetGCIFolderPath(int card_index, AllowMovieFolder allow_movie_fo
|
||||||
|
|
||||||
const bool use_movie_folder = allow_movie_folder == AllowMovieFolder::Yes &&
|
const bool use_movie_folder = allow_movie_folder == AllowMovieFolder::Yes &&
|
||||||
Movie::IsPlayingInput() && Movie::IsConfigSaved() &&
|
Movie::IsPlayingInput() && Movie::IsConfigSaved() &&
|
||||||
Movie::IsUsingMemcard(card_index) &&
|
Movie::IsUsingMemcard(card_slot) &&
|
||||||
Movie::IsStartingFromClearSave();
|
Movie::IsStartingFromClearSave();
|
||||||
|
|
||||||
if (use_movie_folder)
|
if (use_movie_folder)
|
||||||
|
@ -170,7 +164,7 @@ CEXIMemoryCard::GetGCIFolderPath(int card_index, AllowMovieFolder allow_movie_fo
|
||||||
|
|
||||||
const DiscIO::Region region = SConfig::ToGameCubeRegion(SConfig::GetInstance().m_region);
|
const DiscIO::Region region = SConfig::ToGameCubeRegion(SConfig::GetInstance().m_region);
|
||||||
path = path + SConfig::GetDirectoryForRegion(region) + DIR_SEP +
|
path = path + SConfig::GetDirectoryForRegion(region) + DIR_SEP +
|
||||||
fmt::format("Card {}", char('A' + card_index));
|
fmt::format("Card {}", s_card_short_names[card_slot]);
|
||||||
return {std::move(path), !use_movie_folder};
|
return {std::move(path), !use_movie_folder};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -186,7 +180,7 @@ void CEXIMemoryCard::SetupGciFolder(const Memcard::HeaderData& header_data)
|
||||||
|
|
||||||
// TODO(C++20): Use structured bindings when we can use C++20 and refer to structured bindings
|
// TODO(C++20): Use structured bindings when we can use C++20 and refer to structured bindings
|
||||||
// in lambda captures
|
// in lambda captures
|
||||||
const auto folder_path_pair = GetGCIFolderPath(m_card_index, AllowMovieFolder::Yes);
|
const auto folder_path_pair = GetGCIFolderPath(m_card_slot, AllowMovieFolder::Yes);
|
||||||
const std::string& dir_path = folder_path_pair.first;
|
const std::string& dir_path = folder_path_pair.first;
|
||||||
const bool migrate = folder_path_pair.second;
|
const bool migrate = folder_path_pair.second;
|
||||||
|
|
||||||
|
@ -194,7 +188,7 @@ void CEXIMemoryCard::SetupGciFolder(const Memcard::HeaderData& header_data)
|
||||||
if (!file_info.Exists())
|
if (!file_info.Exists())
|
||||||
{
|
{
|
||||||
if (migrate) // first use of memcard folder, migrate automatically
|
if (migrate) // first use of memcard folder, migrate automatically
|
||||||
MigrateFromMemcardFile(dir_path + DIR_SEP, m_card_index);
|
MigrateFromMemcardFile(dir_path + DIR_SEP, m_card_slot);
|
||||||
else
|
else
|
||||||
File::CreateFullPath(dir_path + DIR_SEP);
|
File::CreateFullPath(dir_path + DIR_SEP);
|
||||||
}
|
}
|
||||||
|
@ -204,7 +198,7 @@ void CEXIMemoryCard::SetupGciFolder(const Memcard::HeaderData& header_data)
|
||||||
{
|
{
|
||||||
PanicAlertFmtT("{0} was not a directory, moved to *.original", dir_path);
|
PanicAlertFmtT("{0} was not a directory, moved to *.original", dir_path);
|
||||||
if (migrate)
|
if (migrate)
|
||||||
MigrateFromMemcardFile(dir_path + DIR_SEP, m_card_index);
|
MigrateFromMemcardFile(dir_path + DIR_SEP, m_card_slot);
|
||||||
else
|
else
|
||||||
File::CreateFullPath(dir_path + DIR_SEP);
|
File::CreateFullPath(dir_path + DIR_SEP);
|
||||||
}
|
}
|
||||||
|
@ -218,22 +212,23 @@ void CEXIMemoryCard::SetupGciFolder(const Memcard::HeaderData& header_data)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
m_memory_card = std::make_unique<GCMemcardDirectory>(dir_path + DIR_SEP, m_card_index,
|
m_memory_card = std::make_unique<GCMemcardDirectory>(dir_path + DIR_SEP, m_card_slot, header_data,
|
||||||
header_data, current_game_id);
|
current_game_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CEXIMemoryCard::SetupRawMemcard(u16 size_mb)
|
void CEXIMemoryCard::SetupRawMemcard(u16 size_mb)
|
||||||
{
|
{
|
||||||
const bool is_slot_a = m_card_index == 0;
|
std::string filename = Config::Get(Config::GetInfoForMemcardPath(m_card_slot));
|
||||||
std::string filename = is_slot_a ? Config::Get(Config::MAIN_MEMCARD_A_PATH) :
|
if (Movie::IsPlayingInput() && Movie::IsConfigSaved() && Movie::IsUsingMemcard(m_card_slot) &&
|
||||||
Config::Get(Config::MAIN_MEMCARD_B_PATH);
|
|
||||||
if (Movie::IsPlayingInput() && Movie::IsConfigSaved() && Movie::IsUsingMemcard(m_card_index) &&
|
|
||||||
Movie::IsStartingFromClearSave())
|
Movie::IsStartingFromClearSave())
|
||||||
filename = File::GetUserPath(D_GCUSER_IDX) + fmt::format("Movie{}.raw", is_slot_a ? 'A' : 'B');
|
{
|
||||||
|
filename = File::GetUserPath(D_GCUSER_IDX) +
|
||||||
|
fmt::format("Movie{}.raw", s_card_short_names[m_card_slot]);
|
||||||
|
}
|
||||||
|
|
||||||
const std::string region_dir =
|
const std::string region_dir =
|
||||||
SConfig::GetDirectoryForRegion(SConfig::ToGameCubeRegion(SConfig::GetInstance().m_region));
|
SConfig::GetDirectoryForRegion(SConfig::ToGameCubeRegion(SConfig::GetInstance().m_region));
|
||||||
MemoryCard::CheckPath(filename, region_dir, is_slot_a);
|
MemoryCard::CheckPath(filename, region_dir, m_card_slot);
|
||||||
|
|
||||||
if (size_mb < Memcard::MBIT_SIZE_MEMORY_CARD_2043)
|
if (size_mb < Memcard::MBIT_SIZE_MEMORY_CARD_2043)
|
||||||
{
|
{
|
||||||
|
@ -241,13 +236,13 @@ void CEXIMemoryCard::SetupRawMemcard(u16 size_mb)
|
||||||
fmt::format(".{}", Memcard::MbitToFreeBlocks(size_mb)));
|
fmt::format(".{}", Memcard::MbitToFreeBlocks(size_mb)));
|
||||||
}
|
}
|
||||||
|
|
||||||
m_memory_card = std::make_unique<MemoryCard>(filename, m_card_index, size_mb);
|
m_memory_card = std::make_unique<MemoryCard>(filename, m_card_slot, size_mb);
|
||||||
}
|
}
|
||||||
|
|
||||||
CEXIMemoryCard::~CEXIMemoryCard()
|
CEXIMemoryCard::~CEXIMemoryCard()
|
||||||
{
|
{
|
||||||
CoreTiming::RemoveEvent(s_et_cmd_done[m_card_index]);
|
CoreTiming::RemoveEvent(s_et_cmd_done[m_card_slot]);
|
||||||
CoreTiming::RemoveEvent(s_et_transfer_complete[m_card_index]);
|
CoreTiming::RemoveEvent(s_et_transfer_complete[m_card_slot]);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CEXIMemoryCard::UseDelayedTransferCompletion() const
|
bool CEXIMemoryCard::UseDelayedTransferCompletion() const
|
||||||
|
@ -272,13 +267,14 @@ void CEXIMemoryCard::CmdDone()
|
||||||
void CEXIMemoryCard::TransferComplete()
|
void CEXIMemoryCard::TransferComplete()
|
||||||
{
|
{
|
||||||
// Transfer complete, send interrupt
|
// Transfer complete, send interrupt
|
||||||
ExpansionInterface::GetChannel(m_card_index)->SendTransferComplete();
|
ExpansionInterface::GetChannel(ExpansionInterface::SlotToEXIChannel(m_card_slot))
|
||||||
|
->SendTransferComplete();
|
||||||
}
|
}
|
||||||
|
|
||||||
void CEXIMemoryCard::CmdDoneLater(u64 cycles)
|
void CEXIMemoryCard::CmdDoneLater(u64 cycles)
|
||||||
{
|
{
|
||||||
CoreTiming::RemoveEvent(s_et_cmd_done[m_card_index]);
|
CoreTiming::RemoveEvent(s_et_cmd_done[m_card_slot]);
|
||||||
CoreTiming::ScheduleEvent(cycles, s_et_cmd_done[m_card_index], m_card_index);
|
CoreTiming::ScheduleEvent(cycles, s_et_cmd_done[m_card_slot], static_cast<u64>(m_card_slot));
|
||||||
}
|
}
|
||||||
|
|
||||||
void CEXIMemoryCard::SetCS(int cs)
|
void CEXIMemoryCard::SetCS(int cs)
|
||||||
|
@ -522,19 +518,10 @@ void CEXIMemoryCard::DoState(PointerWrap& p)
|
||||||
p.Do(m_programming_buffer);
|
p.Do(m_programming_buffer);
|
||||||
p.Do(m_address);
|
p.Do(m_address);
|
||||||
m_memory_card->DoState(p);
|
m_memory_card->DoState(p);
|
||||||
p.Do(m_card_index);
|
p.Do(m_card_slot);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
IEXIDevice* CEXIMemoryCard::FindDevice(EXIDeviceType device_type, int custom_index)
|
|
||||||
{
|
|
||||||
if (device_type != m_device_type)
|
|
||||||
return nullptr;
|
|
||||||
if (custom_index != m_card_index)
|
|
||||||
return nullptr;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
// DMA reads are preceded by all of the necessary setup via IMMRead
|
// DMA reads are preceded by all of the necessary setup via IMMRead
|
||||||
// read all at once instead of single byte at a time as done by IEXIDevice::DMARead
|
// read all at once instead of single byte at a time as done by IEXIDevice::DMARead
|
||||||
void CEXIMemoryCard::DMARead(u32 addr, u32 size)
|
void CEXIMemoryCard::DMARead(u32 addr, u32 size)
|
||||||
|
@ -548,7 +535,7 @@ void CEXIMemoryCard::DMARead(u32 addr, u32 size)
|
||||||
|
|
||||||
// Schedule transfer complete later based on read speed
|
// Schedule transfer complete later based on read speed
|
||||||
CoreTiming::ScheduleEvent(size * (SystemTimers::GetTicksPerSecond() / MC_TRANSFER_RATE_READ),
|
CoreTiming::ScheduleEvent(size * (SystemTimers::GetTicksPerSecond() / MC_TRANSFER_RATE_READ),
|
||||||
s_et_transfer_complete[m_card_index], m_card_index);
|
s_et_transfer_complete[m_card_slot], static_cast<u64>(m_card_slot));
|
||||||
}
|
}
|
||||||
|
|
||||||
// DMA write are preceded by all of the necessary setup via IMMWrite
|
// DMA write are preceded by all of the necessary setup via IMMWrite
|
||||||
|
@ -564,6 +551,6 @@ void CEXIMemoryCard::DMAWrite(u32 addr, u32 size)
|
||||||
|
|
||||||
// Schedule transfer complete later based on write speed
|
// Schedule transfer complete later based on write speed
|
||||||
CoreTiming::ScheduleEvent(size * (SystemTimers::GetTicksPerSecond() / MC_TRANSFER_RATE_WRITE),
|
CoreTiming::ScheduleEvent(size * (SystemTimers::GetTicksPerSecond() / MC_TRANSFER_RATE_WRITE),
|
||||||
s_et_transfer_complete[m_card_index], m_card_index);
|
s_et_transfer_complete[m_card_slot], static_cast<u64>(m_card_slot));
|
||||||
}
|
}
|
||||||
} // namespace ExpansionInterface
|
} // namespace ExpansionInterface
|
||||||
|
|
|
@ -21,6 +21,8 @@ struct HeaderData;
|
||||||
|
|
||||||
namespace ExpansionInterface
|
namespace ExpansionInterface
|
||||||
{
|
{
|
||||||
|
enum class Slot : int;
|
||||||
|
|
||||||
enum class AllowMovieFolder
|
enum class AllowMovieFolder
|
||||||
{
|
{
|
||||||
Yes,
|
Yes,
|
||||||
|
@ -30,14 +32,13 @@ enum class AllowMovieFolder
|
||||||
class CEXIMemoryCard : public IEXIDevice
|
class CEXIMemoryCard : public IEXIDevice
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
CEXIMemoryCard(int index, bool gci_folder, const Memcard::HeaderData& header_data);
|
CEXIMemoryCard(Slot slot, bool gci_folder, const Memcard::HeaderData& header_data);
|
||||||
~CEXIMemoryCard() override;
|
~CEXIMemoryCard() override;
|
||||||
void SetCS(int cs) override;
|
void SetCS(int cs) override;
|
||||||
bool IsInterruptSet() override;
|
bool IsInterruptSet() override;
|
||||||
bool UseDelayedTransferCompletion() const override;
|
bool UseDelayedTransferCompletion() const override;
|
||||||
bool IsPresent() const override;
|
bool IsPresent() const override;
|
||||||
void DoState(PointerWrap& p) override;
|
void DoState(PointerWrap& p) override;
|
||||||
IEXIDevice* FindDevice(EXIDeviceType device_type, int custom_index) override;
|
|
||||||
void DMARead(u32 addr, u32 size) override;
|
void DMARead(u32 addr, u32 size) override;
|
||||||
void DMAWrite(u32 addr, u32 size) override;
|
void DMAWrite(u32 addr, u32 size) override;
|
||||||
|
|
||||||
|
@ -48,7 +49,7 @@ public:
|
||||||
static void Shutdown();
|
static void Shutdown();
|
||||||
|
|
||||||
static std::pair<std::string /* path */, bool /* migrate */>
|
static std::pair<std::string /* path */, bool /* migrate */>
|
||||||
GetGCIFolderPath(int card_index, AllowMovieFolder allow_movie_folder);
|
GetGCIFolderPath(Slot card_slot, AllowMovieFolder allow_movie_folder);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void SetupGciFolder(const Memcard::HeaderData& header_data);
|
void SetupGciFolder(const Memcard::HeaderData& header_data);
|
||||||
|
@ -90,7 +91,7 @@ private:
|
||||||
ChipErase = 0xF4,
|
ChipErase = 0xF4,
|
||||||
};
|
};
|
||||||
|
|
||||||
int m_card_index;
|
Slot m_card_slot;
|
||||||
//! memory card state
|
//! memory card state
|
||||||
|
|
||||||
// STATE_TO_SAVE
|
// STATE_TO_SAVE
|
||||||
|
|
|
@ -12,8 +12,9 @@ class PointerWrap;
|
||||||
class MemoryCardBase
|
class MemoryCardBase
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
explicit MemoryCardBase(int card_index = 0, int size_mbits = Memcard::MBIT_SIZE_MEMORY_CARD_2043)
|
explicit MemoryCardBase(ExpansionInterface::Slot card_slot,
|
||||||
: m_card_index(card_index), m_nintendo_card_id(size_mbits)
|
int size_mbits = Memcard::MBIT_SIZE_MEMORY_CARD_2043)
|
||||||
|
: m_card_slot(card_slot), m_nintendo_card_id(size_mbits)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
virtual ~MemoryCardBase() = default;
|
virtual ~MemoryCardBase() = default;
|
||||||
|
@ -25,6 +26,6 @@ public:
|
||||||
u32 GetCardId() const { return m_nintendo_card_id; }
|
u32 GetCardId() const { return m_nintendo_card_id; }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
int m_card_index;
|
ExpansionInterface::Slot m_card_slot;
|
||||||
u16 m_nintendo_card_id;
|
u16 m_nintendo_card_id;
|
||||||
};
|
};
|
||||||
|
|
|
@ -152,7 +152,7 @@ std::vector<std::string> GCMemcardDirectory::GetFileNamesForGameID(const std::st
|
||||||
return filenames;
|
return filenames;
|
||||||
}
|
}
|
||||||
|
|
||||||
GCMemcardDirectory::GCMemcardDirectory(const std::string& directory, int slot,
|
GCMemcardDirectory::GCMemcardDirectory(const std::string& directory, ExpansionInterface::Slot slot,
|
||||||
const Memcard::HeaderData& header_data, u32 game_id)
|
const Memcard::HeaderData& header_data, u32 game_id)
|
||||||
: MemoryCardBase(slot, header_data.m_size_mb), m_game_id(game_id), m_last_block(-1),
|
: MemoryCardBase(slot, header_data.m_size_mb), m_game_id(game_id), m_last_block(-1),
|
||||||
m_hdr(header_data), m_bat1(header_data.m_size_mb), m_saves(0), m_save_directory(directory),
|
m_hdr(header_data), m_bat1(header_data.m_size_mb), m_saves(0), m_save_directory(directory),
|
||||||
|
@ -240,7 +240,7 @@ void GCMemcardDirectory::FlushThread()
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Common::SetCurrentThreadName(fmt::format("Memcard {} flushing thread", m_card_index).c_str());
|
Common::SetCurrentThreadName(fmt::format("Memcard {} flushing thread", m_card_slot).c_str());
|
||||||
|
|
||||||
constexpr std::chrono::seconds flush_interval{1};
|
constexpr std::chrono::seconds flush_interval{1};
|
||||||
while (true)
|
while (true)
|
||||||
|
@ -705,11 +705,10 @@ void GCMemcardDirectory::DoState(PointerWrap& p)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void MigrateFromMemcardFile(const std::string& directory_name, int card_index)
|
void MigrateFromMemcardFile(const std::string& directory_name, ExpansionInterface::Slot card_slot)
|
||||||
{
|
{
|
||||||
File::CreateFullPath(directory_name);
|
File::CreateFullPath(directory_name);
|
||||||
std::string ini_memcard = (card_index == 0) ? Config::Get(Config::MAIN_MEMCARD_A_PATH) :
|
std::string ini_memcard = Config::Get(Config::GetInfoForMemcardPath(card_slot));
|
||||||
Config::Get(Config::MAIN_MEMCARD_B_PATH);
|
|
||||||
if (File::Exists(ini_memcard))
|
if (File::Exists(ini_memcard))
|
||||||
{
|
{
|
||||||
auto [error_code, memcard] = Memcard::GCMemcard::Open(ini_memcard.c_str());
|
auto [error_code, memcard] = Memcard::GCMemcard::Open(ini_memcard.c_str());
|
||||||
|
|
|
@ -15,13 +15,13 @@
|
||||||
|
|
||||||
// Uncomment this to write the system data of the memorycard from directory to disc
|
// Uncomment this to write the system data of the memorycard from directory to disc
|
||||||
//#define _WRITE_MC_HEADER 1
|
//#define _WRITE_MC_HEADER 1
|
||||||
void MigrateFromMemcardFile(const std::string& directory_name, int card_index);
|
void MigrateFromMemcardFile(const std::string& directory_name, ExpansionInterface::Slot card_slot);
|
||||||
|
|
||||||
class GCMemcardDirectory : public MemoryCardBase
|
class GCMemcardDirectory : public MemoryCardBase
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
GCMemcardDirectory(const std::string& directory, int slot, const Memcard::HeaderData& header_data,
|
GCMemcardDirectory(const std::string& directory, ExpansionInterface::Slot slot,
|
||||||
u32 game_id);
|
const Memcard::HeaderData& header_data, u32 game_id);
|
||||||
~GCMemcardDirectory();
|
~GCMemcardDirectory();
|
||||||
|
|
||||||
GCMemcardDirectory(const GCMemcardDirectory&) = delete;
|
GCMemcardDirectory(const GCMemcardDirectory&) = delete;
|
||||||
|
|
|
@ -26,6 +26,7 @@
|
||||||
#include "Core/Config/SessionSettings.h"
|
#include "Core/Config/SessionSettings.h"
|
||||||
#include "Core/ConfigManager.h"
|
#include "Core/ConfigManager.h"
|
||||||
#include "Core/Core.h"
|
#include "Core/Core.h"
|
||||||
|
#include "Core/HW/EXI/EXI.h"
|
||||||
#include "Core/HW/EXI/EXI_DeviceIPL.h"
|
#include "Core/HW/EXI/EXI_DeviceIPL.h"
|
||||||
#include "Core/HW/GCMemcard/GCMemcard.h"
|
#include "Core/HW/GCMemcard/GCMemcard.h"
|
||||||
#include "Core/HW/Sram.h"
|
#include "Core/HW/Sram.h"
|
||||||
|
@ -33,8 +34,9 @@
|
||||||
#define SIZE_TO_Mb (1024 * 8 * 16)
|
#define SIZE_TO_Mb (1024 * 8 * 16)
|
||||||
#define MC_HDR_SIZE 0xA000
|
#define MC_HDR_SIZE 0xA000
|
||||||
|
|
||||||
MemoryCard::MemoryCard(const std::string& filename, int card_index, u16 size_mbits)
|
MemoryCard::MemoryCard(const std::string& filename, ExpansionInterface::Slot card_slot,
|
||||||
: MemoryCardBase(card_index, size_mbits), m_filename(filename)
|
u16 size_mbits)
|
||||||
|
: MemoryCardBase(card_slot, size_mbits), m_filename(filename)
|
||||||
{
|
{
|
||||||
File::IOFile file(m_filename, "rb");
|
File::IOFile file(m_filename, "rb");
|
||||||
if (file)
|
if (file)
|
||||||
|
@ -88,13 +90,15 @@ MemoryCard::~MemoryCard()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void MemoryCard::CheckPath(std::string& memcardPath, const std::string& gameRegion, bool isSlotA)
|
void MemoryCard::CheckPath(std::string& memcardPath, const std::string& gameRegion,
|
||||||
|
ExpansionInterface::Slot card_slot)
|
||||||
{
|
{
|
||||||
|
bool is_slot_a = card_slot == ExpansionInterface::Slot::A;
|
||||||
std::string ext("." + gameRegion + ".raw");
|
std::string ext("." + gameRegion + ".raw");
|
||||||
if (memcardPath.empty())
|
if (memcardPath.empty())
|
||||||
{
|
{
|
||||||
// Use default memcard path if there is no user defined name
|
// Use default memcard path if there is no user defined name
|
||||||
std::string defaultFilename = isSlotA ? GC_MEMCARDA : GC_MEMCARDB;
|
std::string defaultFilename = is_slot_a ? GC_MEMCARDA : GC_MEMCARDB;
|
||||||
memcardPath = File::GetUserPath(D_GCUSER_IDX) + defaultFilename + ext;
|
memcardPath = File::GetUserPath(D_GCUSER_IDX) + defaultFilename + ext;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -118,7 +122,7 @@ void MemoryCard::CheckPath(std::string& memcardPath, const std::string& gameRegi
|
||||||
"Slot {1} path was changed to\n"
|
"Slot {1} path was changed to\n"
|
||||||
"{2}\n"
|
"{2}\n"
|
||||||
"Would you like to copy the old file to this new location?\n",
|
"Would you like to copy the old file to this new location?\n",
|
||||||
isSlotA ? 'A' : 'B', isSlotA ? 'A' : 'B', filename))
|
is_slot_a ? 'A' : 'B', is_slot_a ? 'A' : 'B', filename))
|
||||||
{
|
{
|
||||||
if (!File::Copy(oldFilename, filename))
|
if (!File::Copy(oldFilename, filename))
|
||||||
PanicAlertFmtT("Copy failed");
|
PanicAlertFmtT("Copy failed");
|
||||||
|
@ -142,7 +146,7 @@ void MemoryCard::FlushThread()
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Common::SetCurrentThreadName(fmt::format("Memcard {} flushing thread", m_card_index).c_str());
|
Common::SetCurrentThreadName(fmt::format("Memcard {} flushing thread", m_card_slot).c_str());
|
||||||
|
|
||||||
const auto flush_interval = std::chrono::seconds(15);
|
const auto flush_interval = std::chrono::seconds(15);
|
||||||
|
|
||||||
|
@ -199,9 +203,10 @@ void MemoryCard::FlushThread()
|
||||||
if (do_exit)
|
if (do_exit)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
Core::DisplayMessage(
|
Core::DisplayMessage(fmt::format("Wrote memory card {} contents to {}",
|
||||||
fmt::format("Wrote memory card {} contents to {}", m_card_index ? 'B' : 'A', m_filename),
|
m_card_slot == ExpansionInterface::Slot::A ? 'A' : 'B',
|
||||||
4000);
|
m_filename),
|
||||||
|
4000);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -265,7 +270,7 @@ void MemoryCard::ClearAll()
|
||||||
|
|
||||||
void MemoryCard::DoState(PointerWrap& p)
|
void MemoryCard::DoState(PointerWrap& p)
|
||||||
{
|
{
|
||||||
p.Do(m_card_index);
|
p.Do(m_card_slot);
|
||||||
p.Do(m_memory_card_size);
|
p.Do(m_memory_card_size);
|
||||||
p.DoArray(&m_memcard_data[0], m_memory_card_size);
|
p.DoArray(&m_memcard_data[0], m_memory_card_size);
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,10 +17,11 @@ class PointerWrap;
|
||||||
class MemoryCard : public MemoryCardBase
|
class MemoryCard : public MemoryCardBase
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
MemoryCard(const std::string& filename, int card_index,
|
MemoryCard(const std::string& filename, ExpansionInterface::Slot card_slot,
|
||||||
u16 size_mbits = Memcard::MBIT_SIZE_MEMORY_CARD_2043);
|
u16 size_mbits = Memcard::MBIT_SIZE_MEMORY_CARD_2043);
|
||||||
~MemoryCard();
|
~MemoryCard();
|
||||||
static void CheckPath(std::string& memcardPath, const std::string& gameRegion, bool isSlotA);
|
static void CheckPath(std::string& memcardPath, const std::string& gameRegion,
|
||||||
|
ExpansionInterface::Slot slot);
|
||||||
void FlushThread();
|
void FlushThread();
|
||||||
void MakeDirty();
|
void MakeDirty();
|
||||||
|
|
||||||
|
|
|
@ -6,8 +6,11 @@
|
||||||
#include "Common/CommonTypes.h"
|
#include "Common/CommonTypes.h"
|
||||||
#include "Common/IOFile.h"
|
#include "Common/IOFile.h"
|
||||||
#include "Common/Logging/Log.h"
|
#include "Common/Logging/Log.h"
|
||||||
|
#include "Common/MsgHandler.h"
|
||||||
#include "Common/Swap.h"
|
#include "Common/Swap.h"
|
||||||
|
|
||||||
#include "Core/ConfigManager.h"
|
#include "Core/ConfigManager.h"
|
||||||
|
#include "Core/HW/EXI/EXI.h"
|
||||||
|
|
||||||
// English
|
// English
|
||||||
// This is just a template. Most/all fields are updated with sane(r) values at runtime.
|
// This is just a template. Most/all fields are updated with sane(r) values at runtime.
|
||||||
|
@ -72,8 +75,22 @@ void InitSRAM()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void SetCardFlashID(const u8* buffer, u8 card_index)
|
void SetCardFlashID(const u8* buffer, ExpansionInterface::Slot card_slot)
|
||||||
{
|
{
|
||||||
|
u8 card_index;
|
||||||
|
switch (card_slot)
|
||||||
|
{
|
||||||
|
case ExpansionInterface::Slot::A:
|
||||||
|
card_index = 0;
|
||||||
|
break;
|
||||||
|
case ExpansionInterface::Slot::B:
|
||||||
|
card_index = 1;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
PanicAlertFmt("Invalid memcard slot {}", card_slot);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
u64 rand = Common::swap64(&buffer[12]);
|
u64 rand = Common::swap64(&buffer[12]);
|
||||||
u8 csum = 0;
|
u8 csum = 0;
|
||||||
for (int i = 0; i < 12; i++)
|
for (int i = 0; i < 12; i++)
|
||||||
|
|
|
@ -34,9 +34,15 @@ distribution.
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <array>
|
#include <array>
|
||||||
|
|
||||||
#include "Common/CommonTypes.h"
|
#include "Common/CommonTypes.h"
|
||||||
#include "Common/Swap.h"
|
#include "Common/Swap.h"
|
||||||
|
|
||||||
|
namespace ExpansionInterface
|
||||||
|
{
|
||||||
|
enum class Slot : int;
|
||||||
|
};
|
||||||
|
|
||||||
using CardFlashId = std::array<u8, 12>;
|
using CardFlashId = std::array<u8, 12>;
|
||||||
|
|
||||||
#pragma pack(push, 1)
|
#pragma pack(push, 1)
|
||||||
|
@ -128,7 +134,7 @@ static_assert(sizeof(Sram) == 0x44);
|
||||||
#pragma pack(pop)
|
#pragma pack(pop)
|
||||||
|
|
||||||
void InitSRAM();
|
void InitSRAM();
|
||||||
void SetCardFlashID(const u8* buffer, u8 card_index);
|
void SetCardFlashID(const u8* buffer, ExpansionInterface::Slot card_slot);
|
||||||
void FixSRAMChecksums();
|
void FixSRAMChecksums();
|
||||||
|
|
||||||
extern Sram g_SRAM;
|
extern Sram g_SRAM;
|
||||||
|
|
|
@ -43,6 +43,7 @@
|
||||||
#include "Core/DSP/DSPCore.h"
|
#include "Core/DSP/DSPCore.h"
|
||||||
#include "Core/HW/CPU.h"
|
#include "Core/HW/CPU.h"
|
||||||
#include "Core/HW/DVD/DVDInterface.h"
|
#include "Core/HW/DVD/DVDInterface.h"
|
||||||
|
#include "Core/HW/EXI/EXI.h"
|
||||||
#include "Core/HW/EXI/EXI_DeviceIPL.h"
|
#include "Core/HW/EXI/EXI_DeviceIPL.h"
|
||||||
#include "Core/HW/EXI/EXI_DeviceMemoryCard.h"
|
#include "Core/HW/EXI/EXI_DeviceMemoryCard.h"
|
||||||
#include "Core/HW/ProcessorInterface.h"
|
#include "Core/HW/ProcessorInterface.h"
|
||||||
|
@ -433,9 +434,17 @@ bool IsStartingFromClearSave()
|
||||||
return s_bClearSave;
|
return s_bClearSave;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsUsingMemcard(int memcard)
|
bool IsUsingMemcard(ExpansionInterface::Slot slot)
|
||||||
{
|
{
|
||||||
return (s_memcards & (1 << memcard)) != 0;
|
switch (slot)
|
||||||
|
{
|
||||||
|
case ExpansionInterface::Slot::A:
|
||||||
|
return (s_memcards & 1) != 0;
|
||||||
|
case ExpansionInterface::Slot::B:
|
||||||
|
return (s_memcards & 2) != 0;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsNetPlayRecording()
|
bool IsNetPlayRecording()
|
||||||
|
@ -1456,9 +1465,9 @@ void GetSettings()
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
const auto gci_folder_has_saves = [](int card_index) {
|
const auto gci_folder_has_saves = [](ExpansionInterface::Slot card_slot) {
|
||||||
const auto [path, migrate] = ExpansionInterface::CEXIMemoryCard::GetGCIFolderPath(
|
const auto [path, migrate] = ExpansionInterface::CEXIMemoryCard::GetGCIFolderPath(
|
||||||
card_index, ExpansionInterface::AllowMovieFolder::No);
|
card_slot, ExpansionInterface::AllowMovieFolder::No);
|
||||||
const u64 number_of_saves = File::ScanDirectoryTree(path, false).size;
|
const u64 number_of_saves = File::ScanDirectoryTree(path, false).size;
|
||||||
return number_of_saves > 0;
|
return number_of_saves > 0;
|
||||||
};
|
};
|
||||||
|
@ -1466,8 +1475,8 @@ void GetSettings()
|
||||||
s_bClearSave =
|
s_bClearSave =
|
||||||
!(slot_a_has_raw_memcard && File::Exists(Config::Get(Config::MAIN_MEMCARD_A_PATH))) &&
|
!(slot_a_has_raw_memcard && File::Exists(Config::Get(Config::MAIN_MEMCARD_A_PATH))) &&
|
||||||
!(slot_b_has_raw_memcard && File::Exists(Config::Get(Config::MAIN_MEMCARD_B_PATH))) &&
|
!(slot_b_has_raw_memcard && File::Exists(Config::Get(Config::MAIN_MEMCARD_B_PATH))) &&
|
||||||
!(slot_a_has_gci_folder && gci_folder_has_saves(0)) &&
|
!(slot_a_has_gci_folder && gci_folder_has_saves(ExpansionInterface::Slot::A)) &&
|
||||||
!(slot_b_has_gci_folder && gci_folder_has_saves(1));
|
!(slot_b_has_gci_folder && gci_folder_has_saves(ExpansionInterface::Slot::B));
|
||||||
}
|
}
|
||||||
s_memcards |= (slot_a_has_raw_memcard || slot_a_has_gci_folder) << 0;
|
s_memcards |= (slot_a_has_raw_memcard || slot_a_has_gci_folder) << 0;
|
||||||
s_memcards |= (slot_b_has_raw_memcard || slot_b_has_gci_folder) << 1;
|
s_memcards |= (slot_b_has_raw_memcard || slot_b_has_gci_folder) << 1;
|
||||||
|
|
|
@ -17,6 +17,11 @@ struct BootParameters;
|
||||||
struct GCPadStatus;
|
struct GCPadStatus;
|
||||||
class PointerWrap;
|
class PointerWrap;
|
||||||
|
|
||||||
|
namespace ExpansionInterface
|
||||||
|
{
|
||||||
|
enum class Slot : int;
|
||||||
|
}
|
||||||
|
|
||||||
namespace WiimoteCommon
|
namespace WiimoteCommon
|
||||||
{
|
{
|
||||||
class DataReportBuilder;
|
class DataReportBuilder;
|
||||||
|
@ -166,7 +171,7 @@ void SetReset(bool reset);
|
||||||
|
|
||||||
bool IsConfigSaved();
|
bool IsConfigSaved();
|
||||||
bool IsStartingFromClearSave();
|
bool IsStartingFromClearSave();
|
||||||
bool IsUsingMemcard(int memcard);
|
bool IsUsingMemcard(ExpansionInterface::Slot slot);
|
||||||
void SetGraphicsConfig();
|
void SetGraphicsConfig();
|
||||||
bool IsNetPlayRecording();
|
bool IsNetPlayRecording();
|
||||||
|
|
||||||
|
|
|
@ -1670,10 +1670,9 @@ bool NetPlayServer::SyncSaveData()
|
||||||
|
|
||||||
if (m_settings.m_EXIDevice[slot] == ExpansionInterface::EXIDeviceType::MemoryCard)
|
if (m_settings.m_EXIDevice[slot] == ExpansionInterface::EXIDeviceType::MemoryCard)
|
||||||
{
|
{
|
||||||
std::string path = is_slot_a ? Config::Get(Config::MAIN_MEMCARD_A_PATH) :
|
std::string path = Config::Get(Config::GetInfoForMemcardPath(slot));
|
||||||
Config::Get(Config::MAIN_MEMCARD_B_PATH);
|
|
||||||
|
|
||||||
MemoryCard::CheckPath(path, region, is_slot_a);
|
MemoryCard::CheckPath(path, region, slot);
|
||||||
|
|
||||||
int size_override;
|
int size_override;
|
||||||
IniFile gameIni = SConfig::LoadGameIni(game->GetGameID(), game->GetRevision());
|
IniFile gameIni = SConfig::LoadGameIni(game->GetGameID(), game->GetRevision());
|
||||||
|
|
|
@ -677,9 +677,7 @@ void GameList::OpenGCSaveFolder()
|
||||||
SConfig::GetDirectoryForRegion(game->GetRegion()),
|
SConfig::GetDirectoryForRegion(game->GetRegion()),
|
||||||
slot == Slot::A ? "Card A" : "Card B");
|
slot == Slot::A ? "Card A" : "Card B");
|
||||||
|
|
||||||
std::string override_path = slot == Slot::A ?
|
std::string override_path = Config::Get(Config::GetInfoForGCIPathOverride(slot));
|
||||||
Config::Get(Config::MAIN_GCI_FOLDER_A_PATH_OVERRIDE) :
|
|
||||||
Config::Get(Config::MAIN_GCI_FOLDER_B_PATH_OVERRIDE);
|
|
||||||
|
|
||||||
if (!override_path.empty())
|
if (!override_path.empty())
|
||||||
path = override_path;
|
path = override_path;
|
||||||
|
@ -697,8 +695,7 @@ void GameList::OpenGCSaveFolder()
|
||||||
}
|
}
|
||||||
case ExpansionInterface::EXIDeviceType::MemoryCard:
|
case ExpansionInterface::EXIDeviceType::MemoryCard:
|
||||||
{
|
{
|
||||||
std::string memcard_path = slot == Slot::A ? Config::Get(Config::MAIN_MEMCARD_A_PATH) :
|
std::string memcard_path = Config::Get(Config::GetInfoForMemcardPath(slot));
|
||||||
Config::Get(Config::MAIN_MEMCARD_B_PATH);
|
|
||||||
|
|
||||||
std::string memcard_dir;
|
std::string memcard_dir;
|
||||||
|
|
||||||
|
|
|
@ -387,11 +387,11 @@ void GameCubePane::OnConfigPressed(int slot)
|
||||||
ExpansionInterface::ChangeDevice(
|
ExpansionInterface::ChangeDevice(
|
||||||
// SlotB is on channel 1, slotA and SP1 are on 0
|
// SlotB is on channel 1, slotA and SP1 are on 0
|
||||||
slot,
|
slot,
|
||||||
|
// SP1 is device 2, slots are device 0
|
||||||
|
0,
|
||||||
// The device enum to change to
|
// The device enum to change to
|
||||||
memcard ? ExpansionInterface::EXIDeviceType::MemoryCard :
|
memcard ? ExpansionInterface::EXIDeviceType::MemoryCard :
|
||||||
ExpansionInterface::EXIDeviceType::AGP,
|
ExpansionInterface::EXIDeviceType::AGP);
|
||||||
// SP1 is device 2, slots are device 0
|
|
||||||
0);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -501,10 +501,10 @@ void GameCubePane::SaveSettings()
|
||||||
ExpansionInterface::ChangeDevice(
|
ExpansionInterface::ChangeDevice(
|
||||||
// SlotB is on channel 1, slotA and SP1 are on 0
|
// SlotB is on channel 1, slotA and SP1 are on 0
|
||||||
(i == 1) ? 1 : 0,
|
(i == 1) ? 1 : 0,
|
||||||
// The device enum to change to
|
|
||||||
dev,
|
|
||||||
// SP1 is device 2, slots are device 0
|
// SP1 is device 2, slots are device 0
|
||||||
(i == 2) ? 2 : 0);
|
(i == 2) ? 2 : 0,
|
||||||
|
// The device enum to change to
|
||||||
|
dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
Config::SetBaseOrCurrent(Config::GetInfoForEXIDevice(static_cast<ExpansionInterface::Slot>(i)),
|
Config::SetBaseOrCurrent(Config::GetInfoForEXIDevice(static_cast<ExpansionInterface::Slot>(i)),
|
||||||
|
|
Loading…
Reference in New Issue