mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-24 23:11:14 +01:00
Use Core timing instead of thread
Skylander code tidy ups Convert c array to std::array and fix comments Formatting fixes/review changes Variable comment Migrate portal to System Impl and code tidy ups Use struct Restore review changes Minor fix to schedule transfer method Change descriptors to hex and fix comments
This commit is contained in:
parent
f76a6789a0
commit
18fd0d7dcd
@ -399,8 +399,6 @@ add_library(core
|
||||
IOS/USB/Bluetooth/WiimoteHIDAttr.h
|
||||
IOS/USB/Common.cpp
|
||||
IOS/USB/Common.h
|
||||
IOS/USB/EmulatedUSBDevice.cpp
|
||||
IOS/USB/EmulatedUSBDevice.h
|
||||
IOS/USB/Emulated/Skylander.cpp
|
||||
IOS/USB/Emulated/Skylander.h
|
||||
IOS/USB/Host.cpp
|
||||
|
@ -556,11 +556,6 @@ void SetUSBDeviceWhitelist(const std::set<std::pair<u16, u16>>& devices)
|
||||
const Info<bool> MAIN_EMULATE_SKYLANDER_PORTAL{
|
||||
{System::Main, "EmulatedUSBDevices", "EmulateSkylanderPortal"}, false};
|
||||
|
||||
bool EmulateSkylanderPortal()
|
||||
{
|
||||
return Config::Get(Config::MAIN_EMULATE_SKYLANDER_PORTAL);
|
||||
}
|
||||
|
||||
// The reason we need this function is because some memory card code
|
||||
// expects to get a non-NTSC-K region even if we're emulating an NTSC-K Wii.
|
||||
DiscIO::Region ToGameCubeRegion(DiscIO::Region region)
|
||||
|
@ -345,7 +345,6 @@ void SetUSBDeviceWhitelist(const std::set<std::pair<u16, u16>>& devices);
|
||||
// Main.EmulatedUSBDevices
|
||||
|
||||
extern const Info<bool> MAIN_EMULATE_SKYLANDER_PORTAL;
|
||||
bool EmulateSkylanderPortal();
|
||||
|
||||
// GameCube path utility functions
|
||||
|
||||
|
@ -29,7 +29,7 @@ bool IsSettingSaveable(const Config::Location& config_location)
|
||||
for (const std::string_view section :
|
||||
{"NetPlay", "General", "GBA", "Display", "Network", "Analytics", "AndroidOverlayButtons",
|
||||
"DSP", "GameList", "FifoPlayer", "AutoUpdate", "Movie", "Input", "Debug",
|
||||
"BluetoothPassthrough", "USBPassthrough", "Interface"})
|
||||
"BluetoothPassthrough", "USBPassthrough", "Interface", "EmulatedUSBDevices"})
|
||||
{
|
||||
if (config_location.section == section)
|
||||
return true;
|
||||
|
@ -38,6 +38,13 @@ void TransferCommand::OnTransferComplete(s32 return_value) const
|
||||
m_ios.EnqueueIPCReply(ios_request, return_value, 0, CoreTiming::FromThread::NON_CPU);
|
||||
}
|
||||
|
||||
void TransferCommand::ScheduleTransferCompletion(s32 return_value, u32 expected_time_us) const
|
||||
{
|
||||
auto ticks = SystemTimers::GetTicksPerSecond();
|
||||
s64 cycles_in_future = static_cast<s64>((static_cast<u64>(ticks) * expected_time_us) / 1'000'000);
|
||||
m_ios.EnqueueIPCReply(ios_request, return_value, cycles_in_future, CoreTiming::FromThread::ANY);
|
||||
}
|
||||
|
||||
void IsoMessage::SetPacketReturnValue(const size_t packet_num, const u16 return_value) const
|
||||
{
|
||||
auto& system = Core::System::GetInstance();
|
||||
|
@ -109,13 +109,10 @@ struct TransferCommand
|
||||
// Called after a transfer has completed to reply to the IPC request.
|
||||
// This can be overridden for additional processing before replying.
|
||||
virtual void OnTransferComplete(s32 return_value) const;
|
||||
void ScheduleTransferCompletion(s32 return_value, u32 expected_time_us) const;
|
||||
std::unique_ptr<u8[]> MakeBuffer(size_t size) const;
|
||||
void FillBuffer(const u8* src, size_t size) const;
|
||||
|
||||
// Fake Transfers
|
||||
u64 expected_time;
|
||||
u32 expected_count;
|
||||
|
||||
private:
|
||||
Kernel& m_ios;
|
||||
};
|
||||
|
@ -1,4 +1,4 @@
|
||||
// Copyright 2017 Dolphin Emulator Project
|
||||
// Copyright 2022 Dolphin Emulator Project
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
||||
#include "Core/IOS/USB/Emulated/Skylander.h"
|
||||
@ -16,44 +16,41 @@
|
||||
|
||||
namespace IOS::HLE::USB
|
||||
{
|
||||
SkylanderPortal g_skyportal;
|
||||
|
||||
SkylanderUSB::SkylanderUSB(Kernel& ios, const std::string& device_name)
|
||||
: EmulatedUSBDevice(ios, device_name)
|
||||
SkylanderUSB::SkylanderUSB(Kernel& ios, const std::string& device_name) : m_ios(ios)
|
||||
{
|
||||
m_vid = 0x1430;
|
||||
m_pid = 0x150;
|
||||
m_id = (static_cast<u64>(m_vid) << 32 | static_cast<u64>(m_pid) << 16 | static_cast<u64>(9) << 8 |
|
||||
static_cast<u64>(1));
|
||||
deviceDesc = DeviceDescriptor{18, 1, 512, 0, 0, 0, 64, 5168, 336, 256, 1, 2, 0, 1};
|
||||
configDesc.emplace_back(ConfigDescriptor{9, 2, 41, 1, 1, 0, 128, 250});
|
||||
interfaceDesc.emplace_back(InterfaceDescriptor{9, 4, 0, 0, 2, 3, 0, 0, 0});
|
||||
endpointDesc.emplace_back(EndpointDescriptor{7, 5, 129, 3, 64, 1});
|
||||
endpointDesc.emplace_back(EndpointDescriptor{7, 5, 2, 3, 64, 1});
|
||||
m_device_descriptor = DeviceDescriptor{0x12, 0x1, 0x200, 0x0, 0x0, 0x0, 0x40,
|
||||
0x1430, 0x150, 0x100, 0x1, 0x2, 0x0, 0x1};
|
||||
m_config_descriptor.emplace_back(ConfigDescriptor{0x9, 0x2, 0x29, 0x1, 0x1, 0x0, 0x80, 0xFA});
|
||||
m_interface_descriptor.emplace_back(
|
||||
InterfaceDescriptor{0x9, 0x4, 0x0, 0x0, 0x2, 0x3, 0x0, 0x0, 0x0});
|
||||
m_endpoint_descriptor.emplace_back(EndpointDescriptor{0x7, 0x5, 0x81, 0x3, 0x40, 0x1});
|
||||
m_endpoint_descriptor.emplace_back(EndpointDescriptor{0x7, 0x5, 0x2, 0x3, 0x40, 0x1});
|
||||
}
|
||||
|
||||
SkylanderUSB::~SkylanderUSB()
|
||||
{
|
||||
}
|
||||
SkylanderUSB::~SkylanderUSB() = default;
|
||||
|
||||
DeviceDescriptor SkylanderUSB::GetDeviceDescriptor() const
|
||||
{
|
||||
return deviceDesc;
|
||||
return m_device_descriptor;
|
||||
}
|
||||
|
||||
std::vector<ConfigDescriptor> SkylanderUSB::GetConfigurations() const
|
||||
{
|
||||
return configDesc;
|
||||
return m_config_descriptor;
|
||||
}
|
||||
|
||||
std::vector<InterfaceDescriptor> SkylanderUSB::GetInterfaces(u8 config) const
|
||||
{
|
||||
return interfaceDesc;
|
||||
return m_interface_descriptor;
|
||||
}
|
||||
|
||||
std::vector<EndpointDescriptor> SkylanderUSB::GetEndpoints(u8 config, u8 interface, u8 alt) const
|
||||
{
|
||||
return endpointDesc;
|
||||
return m_endpoint_descriptor;
|
||||
}
|
||||
|
||||
bool SkylanderUSB::Attach()
|
||||
@ -64,11 +61,6 @@ bool SkylanderUSB::Attach()
|
||||
}
|
||||
DEBUG_LOG_FMT(IOS_USB, "[{:04x}:{:04x}] Opening device", m_vid, m_pid);
|
||||
m_device_attached = true;
|
||||
if (!m_has_initialised && !Core::WantsDeterminism())
|
||||
{
|
||||
GetTransferThread().Start();
|
||||
m_has_initialised = true;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
@ -81,11 +73,7 @@ int SkylanderUSB::CancelTransfer(const u8 endpoint)
|
||||
{
|
||||
INFO_LOG_FMT(IOS_USB, "[{:04x}:{:04x} {}] Cancelling transfers (endpoint {:#x})", m_vid, m_pid,
|
||||
m_active_interface, endpoint);
|
||||
if (GetTransferThread().GetTransfers())
|
||||
{
|
||||
return IPC_ENOENT;
|
||||
}
|
||||
GetTransferThread().ClearTransfers();
|
||||
|
||||
return IPC_SUCCESS;
|
||||
}
|
||||
|
||||
@ -121,17 +109,23 @@ int SkylanderUSB::SubmitTransfer(std::unique_ptr<CtrlMessage> cmd)
|
||||
m_vid, m_pid, m_active_interface, cmd->request_type, cmd->request, cmd->value,
|
||||
cmd->index, cmd->length);
|
||||
|
||||
cmd->expected_time = Common::Timer::NowUs() + 100;
|
||||
auto& system = Core::System::GetInstance();
|
||||
auto& memory = system.GetMemory();
|
||||
u8* buf = memory.GetPointerForRange(cmd->data_address, cmd->length);
|
||||
std::array<u8, 64> q_result = {};
|
||||
std::array<u8, 64> q_data = {};
|
||||
// Control transfers are instantaneous
|
||||
switch (cmd->request_type)
|
||||
if ((cmd->length == 0 || buf == nullptr) && cmd->request == 0x09)
|
||||
{
|
||||
// HID host to device type
|
||||
case 0x21:
|
||||
ERROR_LOG_FMT(IOS_USB, "Skylander command invalid");
|
||||
return IPC_EINVAL;
|
||||
}
|
||||
std::array<u8, 64> result = {};
|
||||
std::array<u8, 64> data = {};
|
||||
s32 expected_count = 0;
|
||||
u64 expected_time_us = 100;
|
||||
// Control transfers are instantaneous
|
||||
u8 request_type = cmd->request_type;
|
||||
if (request_type == 0x21)
|
||||
{
|
||||
// HID host to device type
|
||||
switch (cmd->request)
|
||||
{
|
||||
case 0x09:
|
||||
@ -145,7 +139,7 @@ int SkylanderUSB::SubmitTransfer(std::unique_ptr<CtrlMessage> cmd)
|
||||
// Response { 'A', (00 | 01),
|
||||
// ff, 77, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00,
|
||||
// 00, 00, 00, 00, 00, 00, 00, 00 }
|
||||
// The 3rd byte of the command is whether to activate (0x01) or deactivate (0x00) the
|
||||
// The 2nd byte of the command is whether to activate (0x01) or deactivate (0x00) the
|
||||
// portal. The response echos back the activation byte as the 2nd byte of the response. The
|
||||
// 3rd and 4th bytes of the response appear to vary from wired to wireless. On wired
|
||||
// portals, the bytes appear to always be ff 77. On wireless portals, during activation the
|
||||
@ -154,15 +148,15 @@ int SkylanderUSB::SubmitTransfer(std::unique_ptr<CtrlMessage> cmd)
|
||||
// for wireless portals.
|
||||
|
||||
// Wii U Wireless: 41 01 f4 00 41 00 ed 00 41 01 f4 00 41 00 eb 00 41 01 f3 00 41 00 ed 00
|
||||
if (cmd->length == 2 || cmd->length == 32)
|
||||
if (cmd->length == 2)
|
||||
{
|
||||
q_data = {buf[0], buf[1]};
|
||||
q_result = {0x41, buf[1], 0xFF, 0x77, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
||||
q_queries.push(q_result);
|
||||
cmd->expected_count = 10;
|
||||
g_skyportal.Activate();
|
||||
data = {buf[0], buf[1]};
|
||||
result = {0x41, buf[1], 0xFF, 0x77, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
||||
m_queries.push(result);
|
||||
expected_count = 10;
|
||||
system.GetSkylanderPortal().Activate();
|
||||
}
|
||||
break;
|
||||
}
|
||||
@ -179,39 +173,39 @@ int SkylanderUSB::SubmitTransfer(std::unique_ptr<CtrlMessage> cmd)
|
||||
// This command should set the color of the LED in the portal, however this appears
|
||||
// deprecated in most of the recent portals. On portals that do not have LEDs, this command
|
||||
// is silently ignored and do not require a response.
|
||||
if (cmd->length == 4 || cmd->length == 32)
|
||||
if (cmd->length == 4)
|
||||
{
|
||||
g_skyportal.SetLEDs(0x01, buf[1], buf[2], buf[3]);
|
||||
q_data = {0x43, buf[1], buf[2], buf[3]};
|
||||
cmd->expected_count = 12;
|
||||
system.GetSkylanderPortal().SetLEDs(0x01, buf[1], buf[2], buf[3]);
|
||||
data = {0x43, buf[1], buf[2], buf[3]};
|
||||
expected_count = 12;
|
||||
}
|
||||
break;
|
||||
}
|
||||
case 'J':
|
||||
{
|
||||
// Sided color
|
||||
// buf[1] is the side
|
||||
// The 2nd byte is the side
|
||||
// 0x00: right
|
||||
// 0x01: left and right
|
||||
// 0x02: left
|
||||
|
||||
// buf[2], buf[3] and buf[4] are red, green and blue
|
||||
// The 3rd, 4th and 5th bytes are red, green and blue
|
||||
|
||||
// buf[5] is unknown. Observed values are 0x00, 0x0D and 0xF4
|
||||
// The 6th byte is unknown. Observed values are 0x00, 0x0D and 0xF4
|
||||
|
||||
// buf[6] is the fade duration. Exact value-time corrolation unknown. Observed values are
|
||||
// 0x00, 0x01 and 0x07. Custom commands show that the higher this value the longer the
|
||||
// The 7th byte is the fade duration. Exact value-time corrolation unknown. Observed values
|
||||
// are 0x00, 0x01 and 0x07. Custom commands show that the higher this value the longer the
|
||||
// duration.
|
||||
|
||||
// Empty J response is send after the fade is completed. Immeditately sending it is fine
|
||||
// as long as we don't show the fade happening
|
||||
if (cmd->length == 7)
|
||||
{
|
||||
q_data = {buf[0], buf[1], buf[2], buf[3], buf[4], buf[5], buf[6]};
|
||||
cmd->expected_count = 15;
|
||||
q_result = {buf[0]};
|
||||
q_queries.push(q_result);
|
||||
g_skyportal.SetLEDs(buf[1], buf[2], buf[3], buf[4]);
|
||||
data = {buf[0], buf[1], buf[2], buf[3], buf[4], buf[5], buf[6]};
|
||||
expected_count = 15;
|
||||
result = {buf[0]};
|
||||
m_queries.push(result);
|
||||
system.GetSkylanderPortal().SetLEDs(buf[1], buf[2], buf[3], buf[4]);
|
||||
}
|
||||
break;
|
||||
}
|
||||
@ -220,40 +214,38 @@ int SkylanderUSB::SubmitTransfer(std::unique_ptr<CtrlMessage> cmd)
|
||||
// Light
|
||||
// This command is used while playing audio through the portal
|
||||
|
||||
// buf[1] is the position
|
||||
// The 2nd bytes is the position
|
||||
// 0x00: right
|
||||
// 0x01: trap led
|
||||
// 0x02: left
|
||||
|
||||
// buf[2], buf[3] and buf[4] are red, green and blue
|
||||
// The 3rd, 4th and 5th bytes are red, green and blue
|
||||
// the trap led is white-only
|
||||
// increasing or decreasing the values results in a birghter or dimmer light
|
||||
|
||||
// buf[5] is unknown.
|
||||
// A range of values have been observed
|
||||
// increasing or decreasing the values results in a brighter or dimmer light
|
||||
if (cmd->length == 5)
|
||||
{
|
||||
q_data = {buf[0], buf[1], buf[2], buf[3], buf[4]};
|
||||
cmd->expected_count = 13;
|
||||
data = {buf[0], buf[1], buf[2], buf[3], buf[4]};
|
||||
expected_count = 13;
|
||||
|
||||
u8 side = buf[1];
|
||||
if (side == 0x02)
|
||||
{
|
||||
side = 0x04;
|
||||
}
|
||||
g_skyportal.SetLEDs(side, buf[2], buf[3], buf[4]);
|
||||
system.GetSkylanderPortal().SetLEDs(side, buf[2], buf[3], buf[4]);
|
||||
}
|
||||
break;
|
||||
}
|
||||
case 'M':
|
||||
{
|
||||
// Audio Firmware version
|
||||
// Respond with version obtained from Trap Team wired portal
|
||||
if (cmd->length == 2)
|
||||
{
|
||||
q_data = {buf[0], buf[1]};
|
||||
cmd->expected_count = 10;
|
||||
q_result = {buf[0], buf[1], 0x00, 0x19};
|
||||
q_queries.push(q_result);
|
||||
data = {buf[0], buf[1]};
|
||||
expected_count = 10;
|
||||
result = {buf[0], buf[1], 0x00, 0x19};
|
||||
m_queries.push(result);
|
||||
}
|
||||
break;
|
||||
}
|
||||
@ -263,25 +255,28 @@ int SkylanderUSB::SubmitTransfer(std::unique_ptr<CtrlMessage> cmd)
|
||||
// Response { 'Q', 10, 00, 00, 00, 00,
|
||||
// 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00,
|
||||
// 00, 00, 00, 00 }
|
||||
// In the command the 3rd byte indicates which Skylander to query data
|
||||
// In the command the 2nd byte indicates which Skylander to query data
|
||||
// from. Index starts at 0x10 for the 1st Skylander (as reported in the Status command.) The
|
||||
// 16th Skylander indexed would be 0x20.
|
||||
// 16th Skylander indexed would be 0x20. The 3rd byte indicate which block to read from.
|
||||
|
||||
// A response with the 2nd byte of 0x01 indicates an error in the read. Otherwise, the
|
||||
// response indicates the Skylander's index in the 2nd byte, the block read in the 3rd byte,
|
||||
// data (16 bytes) is contained in bytes 4-20.
|
||||
// data (16 bytes) is contained in bytes 4-19.
|
||||
|
||||
// A Skylander has 64 blocks of data indexed from 0x00 to 0x3f. SwapForce characters have 2
|
||||
// character indexes, these may not be sequential.
|
||||
case 'Q':
|
||||
{
|
||||
// Queries a block
|
||||
const u8 sky_num = buf[1] & 0xF;
|
||||
const u8 block = buf[2];
|
||||
g_skyportal.QueryBlock(sky_num, block, q_result.data());
|
||||
q_queries.push(q_result);
|
||||
q_data = {buf[0], buf[1], buf[2]};
|
||||
cmd->expected_count = 11;
|
||||
if (cmd->length == 3)
|
||||
{
|
||||
const u8 sky_num = buf[1] & 0xF;
|
||||
const u8 block = buf[2];
|
||||
system.GetSkylanderPortal().QueryBlock(sky_num, block, result.data());
|
||||
m_queries.push(result);
|
||||
data = {buf[0], buf[1], buf[2]};
|
||||
expected_count = 11;
|
||||
}
|
||||
break;
|
||||
}
|
||||
case 'R':
|
||||
@ -294,14 +289,14 @@ int SkylanderUSB::SubmitTransfer(std::unique_ptr<CtrlMessage> cmd)
|
||||
// 00, 00, 00, 00 }
|
||||
// The 4 byte sequence after the R (0x52) is unknown, but appears consistent based on device
|
||||
// type.
|
||||
if (cmd->length == 2 || cmd->length == 32)
|
||||
if (cmd->length == 2)
|
||||
{
|
||||
q_data = {0x52, 0x00};
|
||||
q_result = {0x52, 0x02, 0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
||||
q_queries.push(q_result);
|
||||
cmd->expected_count = 10;
|
||||
data = {0x52, 0x00};
|
||||
result = {0x52, 0x02, 0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
||||
m_queries.push(result);
|
||||
expected_count = 10;
|
||||
}
|
||||
break;
|
||||
}
|
||||
@ -336,14 +331,20 @@ int SkylanderUSB::SubmitTransfer(std::unique_ptr<CtrlMessage> cmd)
|
||||
// been activated: {01} when active and {00} when deactivated.
|
||||
case 'S':
|
||||
{
|
||||
q_data = {buf[0]};
|
||||
cmd->expected_count = 9;
|
||||
if (cmd->length == 1)
|
||||
{
|
||||
data = {buf[0]};
|
||||
expected_count = 9;
|
||||
}
|
||||
break;
|
||||
}
|
||||
case 'V':
|
||||
{
|
||||
q_data = {buf[0], buf[1], buf[2], buf[3]};
|
||||
cmd->expected_count = 12;
|
||||
if (cmd->length == 4)
|
||||
{
|
||||
data = {buf[0], buf[1], buf[2], buf[3]};
|
||||
expected_count = 12;
|
||||
}
|
||||
break;
|
||||
}
|
||||
// Write
|
||||
@ -352,13 +353,13 @@ int SkylanderUSB::SubmitTransfer(std::unique_ptr<CtrlMessage> cmd)
|
||||
// Response { 'W', 00, 00, 00, 00, 00,
|
||||
// 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00,
|
||||
// 00, 00, 00, 00 }
|
||||
// In the command the 3rd byte indicates which Skylander to query data from. Index starts at
|
||||
// In the command the 2nd byte indicates which Skylander to query data from. Index starts at
|
||||
// 0x10 for the 1st Skylander (as reported in the Status command.) The 16th Skylander
|
||||
// indexed would be 0x20.
|
||||
|
||||
// 4th byte is the block to write to.
|
||||
// 3rd byte is the block to write to.
|
||||
|
||||
// Bytes 5 - 20 ({ 01, 02, 03, 04, 05, 06, 07, 08, 09, 0a, 0b, 0c, 0d, 0e, 0f }) are the
|
||||
// Bytes 4 - 19 ({ 01, 02, 03, 04, 05, 06, 07, 08, 09, 0a, 0b, 0c, 0d, 0e, 0f }) are the
|
||||
// data to write.
|
||||
|
||||
// The response does not appear to return the id of the Skylander being written, the 2nd
|
||||
@ -367,14 +368,17 @@ int SkylanderUSB::SubmitTransfer(std::unique_ptr<CtrlMessage> cmd)
|
||||
|
||||
case 'W':
|
||||
{
|
||||
const u8 sky_num = buf[1] & 0xF;
|
||||
const u8 block = buf[2];
|
||||
g_skyportal.WriteBlock(sky_num, block, &buf[3], q_result.data());
|
||||
q_queries.push(q_result);
|
||||
q_data = {buf[0], buf[1], buf[2], buf[3], buf[4], buf[5], buf[6],
|
||||
if (cmd->length == 19)
|
||||
{
|
||||
const u8 sky_num = buf[1] & 0xF;
|
||||
const u8 block = buf[2];
|
||||
system.GetSkylanderPortal().WriteBlock(sky_num, block, &buf[3], result.data());
|
||||
m_queries.push(result);
|
||||
data = {buf[0], buf[1], buf[2], buf[3], buf[4], buf[5], buf[6],
|
||||
buf[7], buf[8], buf[9], buf[10], buf[11], buf[12], buf[13],
|
||||
buf[14], buf[15], buf[16], buf[17], buf[18]};
|
||||
cmd->expected_count = 19;
|
||||
expected_count = 27;
|
||||
}
|
||||
break;
|
||||
}
|
||||
default:
|
||||
@ -383,24 +387,22 @@ int SkylanderUSB::SubmitTransfer(std::unique_ptr<CtrlMessage> cmd)
|
||||
}
|
||||
break;
|
||||
case 0x0A:
|
||||
cmd->expected_count = 8;
|
||||
expected_count = 8;
|
||||
break;
|
||||
case 0x0B:
|
||||
cmd->expected_count = 8;
|
||||
expected_count = 8;
|
||||
break;
|
||||
default:
|
||||
ERROR_LOG_FMT(IOS_USB, "Unhandled Request {}", cmd->request);
|
||||
break;
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
cmd->expected_time = Common::Timer::NowUs() + 100;
|
||||
GetTransferThread().AddTransfer(std::move(cmd), q_data);
|
||||
if (expected_count == 0)
|
||||
return IPC_EINVAL;
|
||||
ScheduleTransfer(std::move(cmd), data, expected_count, expected_time_us);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int SkylanderUSB::SubmitTransfer(std::unique_ptr<BulkMessage> cmd)
|
||||
{
|
||||
DEBUG_LOG_FMT(IOS_USB, "[{:04x}:{:04x} {}] Bulk: length={} endpoint={}", m_vid, m_pid,
|
||||
@ -420,38 +422,46 @@ int SkylanderUSB::SubmitTransfer(std::unique_ptr<IntrMessage> cmd)
|
||||
auto& system = Core::System::GetInstance();
|
||||
auto& memory = system.GetMemory();
|
||||
u8* buf = memory.GetPointerForRange(cmd->data_address, cmd->length);
|
||||
std::array<u8, 64> q_result = {};
|
||||
if (cmd->length == 0 || buf == nullptr)
|
||||
{
|
||||
ERROR_LOG_FMT(IOS_USB, "Skylander command invalid");
|
||||
return IPC_EINVAL;
|
||||
}
|
||||
std::array<u8, 64> result = {};
|
||||
s32 expected_count;
|
||||
u64 expected_time_us;
|
||||
// Audio requests are 64 bytes long, are the only Interrupt requests longer than 32 bytes,
|
||||
// echo the request as the response and respond after 1ms
|
||||
if (cmd->length > 32)
|
||||
if (cmd->length > 32 && cmd->length <= 64)
|
||||
{
|
||||
std::array<u8, 64> q_audio_result = {};
|
||||
u8* audio_buf = q_audio_result.data();
|
||||
std::array<u8, 64> audio_result = {};
|
||||
u8* audio_buf = audio_result.data();
|
||||
memcpy(audio_buf, buf, cmd->length);
|
||||
cmd->expected_time = Common::Timer::NowUs() + 1000;
|
||||
cmd->expected_count = cmd->length;
|
||||
GetTransferThread().AddTransfer(std::move(cmd), q_audio_result);
|
||||
expected_time_us = 1000;
|
||||
expected_count = cmd->length;
|
||||
ScheduleTransfer(std::move(cmd), audio_result, expected_count, expected_time_us);
|
||||
return 0;
|
||||
}
|
||||
// If some data was requested from the Control Message, then the Interrupt message needs to
|
||||
// respond with that data. Check if the queries queue is empty
|
||||
if (!q_queries.empty())
|
||||
if (!m_queries.empty())
|
||||
{
|
||||
q_result = q_queries.front();
|
||||
q_queries.pop();
|
||||
result = m_queries.front();
|
||||
m_queries.pop();
|
||||
// This needs to happen after ~22 milliseconds
|
||||
cmd->expected_time = Common::Timer::NowUs() + 22000;
|
||||
expected_time_us = 22000;
|
||||
}
|
||||
// If there is no relevant data to respond with, respond with the currentstatus of the Portal
|
||||
else
|
||||
{
|
||||
q_result = g_skyportal.GetStatus();
|
||||
cmd->expected_time = Common::Timer::NowUs() + 2000;
|
||||
result = system.GetSkylanderPortal().GetStatus();
|
||||
expected_time_us = 2000;
|
||||
}
|
||||
cmd->expected_count = 32;
|
||||
GetTransferThread().AddTransfer(std::move(cmd), q_result);
|
||||
expected_count = 32;
|
||||
ScheduleTransfer(std::move(cmd), result, expected_count, expected_time_us);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int SkylanderUSB::SubmitTransfer(std::unique_ptr<IsoMessage> cmd)
|
||||
{
|
||||
DEBUG_LOG_FMT(IOS_USB, "[{:04x}:{:04x} {}] Isochronous: length={} endpoint={} num_packets={}",
|
||||
@ -459,23 +469,27 @@ int SkylanderUSB::SubmitTransfer(std::unique_ptr<IsoMessage> cmd)
|
||||
return 0;
|
||||
}
|
||||
|
||||
void Skylander::save()
|
||||
void SkylanderUSB::ScheduleTransfer(std::unique_ptr<TransferCommand> command,
|
||||
const std::array<u8, 64>& data, s32 expected_count,
|
||||
u64 expected_time_us)
|
||||
{
|
||||
command->FillBuffer(data.data(), expected_count);
|
||||
command->ScheduleTransferCompletion(expected_count, expected_time_us);
|
||||
}
|
||||
|
||||
void Skylander::Save()
|
||||
{
|
||||
if (!sky_file)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
{
|
||||
sky_file.Seek(0, File::SeekOrigin::Begin);
|
||||
sky_file.WriteBytes(data.data(), 0x40 * 0x10);
|
||||
}
|
||||
sky_file.Seek(0, File::SeekOrigin::Begin);
|
||||
sky_file.WriteBytes(data.data(), 0x40 * 0x10);
|
||||
}
|
||||
|
||||
void SkylanderPortal::Activate()
|
||||
{
|
||||
std::lock_guard lock(sky_mutex);
|
||||
if (activated)
|
||||
if (m_activated)
|
||||
{
|
||||
// If the portal was already active no change is needed
|
||||
return;
|
||||
@ -486,12 +500,12 @@ void SkylanderPortal::Activate()
|
||||
{
|
||||
if (s.status & 1)
|
||||
{
|
||||
s.queued_status.push(3);
|
||||
s.queued_status.push(1);
|
||||
s.queued_status.push(Skylander::ADDED);
|
||||
s.queued_status.push(Skylander::READY);
|
||||
}
|
||||
}
|
||||
|
||||
activated = true;
|
||||
m_activated = true;
|
||||
}
|
||||
|
||||
void SkylanderPortal::Deactivate()
|
||||
@ -510,32 +524,32 @@ void SkylanderPortal::Deactivate()
|
||||
s.status &= 1;
|
||||
}
|
||||
|
||||
activated = false;
|
||||
m_activated = false;
|
||||
}
|
||||
|
||||
bool SkylanderPortal::IsActivated()
|
||||
{
|
||||
std::lock_guard lock(sky_mutex);
|
||||
|
||||
return activated;
|
||||
return m_activated;
|
||||
}
|
||||
|
||||
void SkylanderPortal::UpdateStatus()
|
||||
{
|
||||
std::lock_guard lock(sky_mutex);
|
||||
|
||||
if (!status_updated)
|
||||
if (!m_status_updated)
|
||||
{
|
||||
for (auto& s : skylanders)
|
||||
{
|
||||
if (s.status & 1)
|
||||
{
|
||||
s.queued_status.push(0);
|
||||
s.queued_status.push(3);
|
||||
s.queued_status.push(1);
|
||||
s.queued_status.push(Skylander::REMOVED);
|
||||
s.queued_status.push(Skylander::ADDED);
|
||||
s.queued_status.push(Skylander::READY);
|
||||
}
|
||||
}
|
||||
status_updated = true;
|
||||
m_status_updated = true;
|
||||
}
|
||||
}
|
||||
|
||||
@ -549,31 +563,31 @@ void SkylanderPortal::SetLEDs(u8 side, u8 red, u8 green, u8 blue)
|
||||
std::lock_guard lock(sky_mutex);
|
||||
if (side == 0x00)
|
||||
{
|
||||
this->color_right.r = red;
|
||||
this->color_right.g = green;
|
||||
this->color_right.b = blue;
|
||||
m_color_right.red = red;
|
||||
m_color_right.green = green;
|
||||
m_color_right.blue = blue;
|
||||
}
|
||||
else if (side == 0x01)
|
||||
{
|
||||
this->color_right.r = red;
|
||||
this->color_right.g = green;
|
||||
this->color_right.b = blue;
|
||||
m_color_right.red = red;
|
||||
m_color_right.green = green;
|
||||
m_color_right.blue = blue;
|
||||
|
||||
this->color_left.r = red;
|
||||
this->color_left.g = green;
|
||||
this->color_left.b = blue;
|
||||
m_color_left.red = red;
|
||||
m_color_left.green = green;
|
||||
m_color_left.blue = blue;
|
||||
}
|
||||
else if (side == 0x02)
|
||||
{
|
||||
this->color_left.r = red;
|
||||
this->color_left.g = green;
|
||||
this->color_left.b = blue;
|
||||
m_color_left.red = red;
|
||||
m_color_left.green = green;
|
||||
m_color_left.blue = blue;
|
||||
}
|
||||
else if (side == 0x03)
|
||||
{
|
||||
this->color_trap.r = red;
|
||||
this->color_trap.g = green;
|
||||
this->color_trap.b = blue;
|
||||
m_color_trap.red = red;
|
||||
m_color_trap.green = green;
|
||||
m_color_trap.blue = blue;
|
||||
}
|
||||
}
|
||||
|
||||
@ -584,7 +598,7 @@ std::array<u8, 64> SkylanderPortal::GetStatus()
|
||||
u32 status = 0;
|
||||
u8 active = 0x00;
|
||||
|
||||
if (activated)
|
||||
if (m_activated)
|
||||
{
|
||||
active = 0x01;
|
||||
}
|
||||
@ -602,14 +616,14 @@ std::array<u8, 64> SkylanderPortal::GetStatus()
|
||||
status |= s.status;
|
||||
}
|
||||
|
||||
std::array<u8, 64> q_result = {0x53, 0x00, 0x00, 0x00, 0x00, interrupt_counter++,
|
||||
active, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00};
|
||||
memcpy(&q_result.data()[1], &status, sizeof(status));
|
||||
return q_result;
|
||||
std::array<u8, 64> result = {0x53, 0x00, 0x00, 0x00, 0x00, m_interrupt_counter++,
|
||||
active, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||
0x00, 0x00};
|
||||
memcpy(&result[1], &status, sizeof(status));
|
||||
return result;
|
||||
}
|
||||
|
||||
void SkylanderPortal::QueryBlock(u8 sky_num, u8 block, u8* reply_buf)
|
||||
@ -644,7 +658,7 @@ void SkylanderPortal::WriteBlock(u8 sky_num, u8 block, const u8* to_write_buf, u
|
||||
{
|
||||
reply_buf[1] = (0x10 | sky_num);
|
||||
memcpy(skylander.data.data() + (block * 16), to_write_buf, 16);
|
||||
skylander.save();
|
||||
skylander.Save();
|
||||
}
|
||||
else
|
||||
{
|
||||
@ -652,6 +666,76 @@ void SkylanderPortal::WriteBlock(u8 sky_num, u8 block, const u8* to_write_buf, u
|
||||
}
|
||||
}
|
||||
|
||||
u16 SkylanderCRC16(u16 init_value, const u8* buffer, u32 size)
|
||||
{
|
||||
const unsigned short CRC_CCITT_TABLE[256] = {
|
||||
0x0000, 0x1021, 0x2042, 0x3063, 0x4084, 0x50A5, 0x60C6, 0x70E7, 0x8108, 0x9129, 0xA14A,
|
||||
0xB16B, 0xC18C, 0xD1AD, 0xE1CE, 0xF1EF, 0x1231, 0x0210, 0x3273, 0x2252, 0x52B5, 0x4294,
|
||||
0x72F7, 0x62D6, 0x9339, 0x8318, 0xB37B, 0xA35A, 0xD3BD, 0xC39C, 0xF3FF, 0xE3DE, 0x2462,
|
||||
0x3443, 0x0420, 0x1401, 0x64E6, 0x74C7, 0x44A4, 0x5485, 0xA56A, 0xB54B, 0x8528, 0x9509,
|
||||
0xE5EE, 0xF5CF, 0xC5AC, 0xD58D, 0x3653, 0x2672, 0x1611, 0x0630, 0x76D7, 0x66F6, 0x5695,
|
||||
0x46B4, 0xB75B, 0xA77A, 0x9719, 0x8738, 0xF7DF, 0xE7FE, 0xD79D, 0xC7BC, 0x48C4, 0x58E5,
|
||||
0x6886, 0x78A7, 0x0840, 0x1861, 0x2802, 0x3823, 0xC9CC, 0xD9ED, 0xE98E, 0xF9AF, 0x8948,
|
||||
0x9969, 0xA90A, 0xB92B, 0x5AF5, 0x4AD4, 0x7AB7, 0x6A96, 0x1A71, 0x0A50, 0x3A33, 0x2A12,
|
||||
0xDBFD, 0xCBDC, 0xFBBF, 0xEB9E, 0x9B79, 0x8B58, 0xBB3B, 0xAB1A, 0x6CA6, 0x7C87, 0x4CE4,
|
||||
0x5CC5, 0x2C22, 0x3C03, 0x0C60, 0x1C41, 0xEDAE, 0xFD8F, 0xCDEC, 0xDDCD, 0xAD2A, 0xBD0B,
|
||||
0x8D68, 0x9D49, 0x7E97, 0x6EB6, 0x5ED5, 0x4EF4, 0x3E13, 0x2E32, 0x1E51, 0x0E70, 0xFF9F,
|
||||
0xEFBE, 0xDFDD, 0xCFFC, 0xBF1B, 0xAF3A, 0x9F59, 0x8F78, 0x9188, 0x81A9, 0xB1CA, 0xA1EB,
|
||||
0xD10C, 0xC12D, 0xF14E, 0xE16F, 0x1080, 0x00A1, 0x30C2, 0x20E3, 0x5004, 0x4025, 0x7046,
|
||||
0x6067, 0x83B9, 0x9398, 0xA3FB, 0xB3DA, 0xC33D, 0xD31C, 0xE37F, 0xF35E, 0x02B1, 0x1290,
|
||||
0x22F3, 0x32D2, 0x4235, 0x5214, 0x6277, 0x7256, 0xB5EA, 0xA5CB, 0x95A8, 0x8589, 0xF56E,
|
||||
0xE54F, 0xD52C, 0xC50D, 0x34E2, 0x24C3, 0x14A0, 0x0481, 0x7466, 0x6447, 0x5424, 0x4405,
|
||||
0xA7DB, 0xB7FA, 0x8799, 0x97B8, 0xE75F, 0xF77E, 0xC71D, 0xD73C, 0x26D3, 0x36F2, 0x0691,
|
||||
0x16B0, 0x6657, 0x7676, 0x4615, 0x5634, 0xD94C, 0xC96D, 0xF90E, 0xE92F, 0x99C8, 0x89E9,
|
||||
0xB98A, 0xA9AB, 0x5844, 0x4865, 0x7806, 0x6827, 0x18C0, 0x08E1, 0x3882, 0x28A3, 0xCB7D,
|
||||
0xDB5C, 0xEB3F, 0xFB1E, 0x8BF9, 0x9BD8, 0xABBB, 0xBB9A, 0x4A75, 0x5A54, 0x6A37, 0x7A16,
|
||||
0x0AF1, 0x1AD0, 0x2AB3, 0x3A92, 0xFD2E, 0xED0F, 0xDD6C, 0xCD4D, 0xBDAA, 0xAD8B, 0x9DE8,
|
||||
0x8DC9, 0x7C26, 0x6C07, 0x5C64, 0x4C45, 0x3CA2, 0x2C83, 0x1CE0, 0x0CC1, 0xEF1F, 0xFF3E,
|
||||
0xCF5D, 0xDF7C, 0xAF9B, 0xBFBA, 0x8FD9, 0x9FF8, 0x6E17, 0x7E36, 0x4E55, 0x5E74, 0x2E93,
|
||||
0x3EB2, 0x0ED1, 0x1EF0};
|
||||
|
||||
u16 crc = init_value;
|
||||
|
||||
for (u32 i = 0; i < size; i++)
|
||||
{
|
||||
const u16 tmp = (crc >> 8) ^ buffer[i];
|
||||
crc = (crc << 8) ^ CRC_CCITT_TABLE[tmp];
|
||||
}
|
||||
|
||||
return crc;
|
||||
}
|
||||
|
||||
bool SkylanderPortal::CreateSkylander(const std::string& file_path, u16 sky_id, u16 sky_var)
|
||||
{
|
||||
File::IOFile sky_file(file_path, "w+b");
|
||||
if (!sky_file)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
std::array<u8, 0x40 * 0x10> buf{};
|
||||
const auto file_data = buf.data();
|
||||
// Set the block permissions
|
||||
u32 first_block = 0x690F0F0F;
|
||||
u32 other_blocks = 0x69080F7F;
|
||||
memcpy(&file_data[0x36], &first_block, sizeof(first_block));
|
||||
for (u32 index = 1; index < 0x10; index++)
|
||||
{
|
||||
memcpy(&file_data[(index * 0x40) + 0x36], &other_blocks, sizeof(other_blocks));
|
||||
}
|
||||
// Set the skylander info
|
||||
u16 sky_info = (sky_id | sky_var) + 1;
|
||||
memcpy(&file_data[0], &sky_info, sizeof(sky_info));
|
||||
memcpy(&file_data[0x10], &sky_id, sizeof(sky_id));
|
||||
memcpy(&file_data[0x1C], &sky_var, sizeof(sky_var));
|
||||
// Set checksum
|
||||
u16 checksum = SkylanderCRC16(0xFFFF, file_data, 0x1E);
|
||||
memcpy(&file_data[0x1E], &checksum, sizeof(checksum));
|
||||
|
||||
sky_file.WriteBytes(buf.data(), buf.size());
|
||||
return true;
|
||||
}
|
||||
|
||||
bool SkylanderPortal::RemoveSkylander(u8 sky_num)
|
||||
{
|
||||
DEBUG_LOG_FMT(IOS_USB, "Cleared Skylander from slot {}", sky_num);
|
||||
@ -660,10 +744,9 @@ bool SkylanderPortal::RemoveSkylander(u8 sky_num)
|
||||
|
||||
if (skylander.status & 1)
|
||||
{
|
||||
skylander.status = 2;
|
||||
skylander.queued_status.push(2);
|
||||
skylander.queued_status.push(0);
|
||||
skylander.sky_file.Close();
|
||||
skylander.status = Skylander::REMOVING;
|
||||
skylander.queued_status.push(Skylander::REMOVING);
|
||||
skylander.queued_status.push(Skylander::REMOVED);
|
||||
return true;
|
||||
}
|
||||
|
||||
@ -710,9 +793,9 @@ u8 SkylanderPortal::LoadSkylander(u8* buf, File::IOFile in_file)
|
||||
DEBUG_LOG_FMT(IOS_USB, "Skylander Data: \n{}",
|
||||
HexDump(skylander.data.data(), skylander.data.size()));
|
||||
skylander.sky_file = std::move(in_file);
|
||||
skylander.status = 3;
|
||||
skylander.queued_status.push(3);
|
||||
skylander.queued_status.push(1);
|
||||
skylander.status = Skylander::ADDED;
|
||||
skylander.queued_status.push(Skylander::ADDED);
|
||||
skylander.queued_status.push(Skylander::READY);
|
||||
skylander.last_id = sky_serial;
|
||||
}
|
||||
return found_slot;
|
||||
|
@ -8,8 +8,9 @@
|
||||
#include <queue>
|
||||
#include <vector>
|
||||
|
||||
#include "Common/CommonTypes.h"
|
||||
#include "Common/IOFile.h"
|
||||
#include "Core/IOS/USB/EmulatedUSBDevice.h"
|
||||
#include "Core/IOS/USB/Common.h"
|
||||
|
||||
// The maximum possible characters the portal can handle.
|
||||
// The status array is 32 bits and every character takes 2 bits.
|
||||
@ -18,7 +19,7 @@ constexpr u8 MAX_SKYLANDERS = 16;
|
||||
|
||||
namespace IOS::HLE::USB
|
||||
{
|
||||
class SkylanderUSB final : public EmulatedUSBDevice
|
||||
class SkylanderUSB final : public Device
|
||||
{
|
||||
public:
|
||||
SkylanderUSB(Kernel& ios, const std::string& device_name);
|
||||
@ -37,20 +38,20 @@ public:
|
||||
int SubmitTransfer(std::unique_ptr<BulkMessage> message) override;
|
||||
int SubmitTransfer(std::unique_ptr<IntrMessage> message) override;
|
||||
int SubmitTransfer(std::unique_ptr<IsoMessage> message) override;
|
||||
|
||||
protected:
|
||||
std::queue<std::array<u8, 64>> q_queries;
|
||||
void ScheduleTransfer(std::unique_ptr<TransferCommand> command, const std::array<u8, 64>& data,
|
||||
s32 expected_count, u64 expected_time_us);
|
||||
|
||||
private:
|
||||
Kernel& m_ios;
|
||||
u16 m_vid = 0;
|
||||
u16 m_pid = 0;
|
||||
u8 m_active_interface = 0;
|
||||
bool m_device_attached = false;
|
||||
DeviceDescriptor deviceDesc;
|
||||
std::vector<ConfigDescriptor> configDesc;
|
||||
std::vector<InterfaceDescriptor> interfaceDesc;
|
||||
std::vector<EndpointDescriptor> endpointDesc;
|
||||
bool m_has_initialised = false;
|
||||
DeviceDescriptor m_device_descriptor;
|
||||
std::vector<ConfigDescriptor> m_config_descriptor;
|
||||
std::vector<InterfaceDescriptor> m_interface_descriptor;
|
||||
std::vector<EndpointDescriptor> m_endpoint_descriptor;
|
||||
std::queue<std::array<u8, 64>> m_queries;
|
||||
};
|
||||
|
||||
struct Skylander final
|
||||
@ -60,12 +61,22 @@ struct Skylander final
|
||||
std::queue<u8> queued_status;
|
||||
std::array<u8, 0x40 * 0x10> data{};
|
||||
u32 last_id = 0;
|
||||
void save();
|
||||
void Save();
|
||||
|
||||
enum : u8
|
||||
{
|
||||
REMOVED = 0,
|
||||
READY = 1,
|
||||
REMOVING = 2,
|
||||
ADDED = 3
|
||||
};
|
||||
};
|
||||
|
||||
struct LedColor final
|
||||
struct SkylanderLEDColor final
|
||||
{
|
||||
u8 r = 0, g = 0, b = 0;
|
||||
u8 red = 0;
|
||||
u8 green = 0;
|
||||
u8 blue = 0;
|
||||
};
|
||||
|
||||
class SkylanderPortal final
|
||||
@ -81,22 +92,21 @@ public:
|
||||
void QueryBlock(u8 sky_num, u8 block, u8* reply_buf);
|
||||
void WriteBlock(u8 sky_num, u8 block, const u8* to_write_buf, u8* reply_buf);
|
||||
|
||||
bool CreateSkylander(const std::string& file_path, u16 sky_id, u16 sky_var);
|
||||
bool RemoveSkylander(u8 sky_num);
|
||||
u8 LoadSkylander(u8* buf, File::IOFile in_file);
|
||||
|
||||
protected:
|
||||
std::mutex sky_mutex;
|
||||
|
||||
bool activated = true;
|
||||
bool status_updated = false;
|
||||
u8 interrupt_counter = 0;
|
||||
LedColor color_right = {};
|
||||
LedColor color_left = {};
|
||||
LedColor color_trap = {};
|
||||
bool m_activated = true;
|
||||
bool m_status_updated = false;
|
||||
u8 m_interrupt_counter = 0;
|
||||
SkylanderLEDColor m_color_right = {};
|
||||
SkylanderLEDColor m_color_left = {};
|
||||
SkylanderLEDColor m_color_trap = {};
|
||||
|
||||
Skylander skylanders[MAX_SKYLANDERS];
|
||||
std::array<Skylander, MAX_SKYLANDERS> skylanders;
|
||||
};
|
||||
|
||||
extern SkylanderPortal g_skyportal;
|
||||
|
||||
} // namespace IOS::HLE::USB
|
||||
|
@ -1,82 +0,0 @@
|
||||
#include "Core/IOS/USB/EmulatedUSBDevice.h"
|
||||
|
||||
#include <thread>
|
||||
|
||||
#include "Common/Thread.h"
|
||||
#include "Common/Timer.h"
|
||||
#include "Core/Core.h"
|
||||
|
||||
namespace IOS::HLE::USB
|
||||
{
|
||||
EmulatedUSBDevice::EmulatedUSBDevice(Kernel& ios, const std::string& device_name) : m_ios(ios)
|
||||
{
|
||||
}
|
||||
|
||||
EmulatedUSBDevice::~EmulatedUSBDevice()
|
||||
{
|
||||
}
|
||||
|
||||
EmulatedUSBDevice::FakeTransferThread::~FakeTransferThread()
|
||||
{
|
||||
Stop();
|
||||
}
|
||||
|
||||
void EmulatedUSBDevice::FakeTransferThread::Start()
|
||||
{
|
||||
if (Core::WantsDeterminism())
|
||||
return;
|
||||
|
||||
if (m_thread_running.TestAndSet())
|
||||
{
|
||||
m_thread = std::thread([this] {
|
||||
Common::SetCurrentThreadName("Fake Transfer Thread");
|
||||
while (m_thread_running.IsSet())
|
||||
{
|
||||
if (!m_transfers.empty())
|
||||
{
|
||||
std::lock_guard lk{m_transfers_mutex};
|
||||
u64 timestamp = Common::Timer::NowUs();
|
||||
for (auto iterator = m_transfers.begin(); iterator != m_transfers.end();)
|
||||
{
|
||||
auto* command = iterator->second.get();
|
||||
if (command->expected_time > timestamp)
|
||||
{
|
||||
++iterator;
|
||||
continue;
|
||||
}
|
||||
command->FillBuffer(iterator->first.data(), command->expected_count);
|
||||
command->OnTransferComplete(command->expected_count);
|
||||
iterator = m_transfers.erase(iterator);
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
void EmulatedUSBDevice::FakeTransferThread::Stop()
|
||||
{
|
||||
if (m_thread_running.TestAndClear())
|
||||
m_thread.join();
|
||||
}
|
||||
|
||||
void EmulatedUSBDevice::FakeTransferThread::AddTransfer(std::unique_ptr<TransferCommand> command,
|
||||
std::array<u8, 64> data)
|
||||
{
|
||||
std::lock_guard lk{m_transfers_mutex};
|
||||
m_transfers.emplace(data, std::move(command));
|
||||
}
|
||||
|
||||
void EmulatedUSBDevice::FakeTransferThread::ClearTransfers()
|
||||
{
|
||||
std::lock_guard lk{m_transfers_mutex};
|
||||
m_transfers.clear();
|
||||
}
|
||||
|
||||
bool EmulatedUSBDevice::FakeTransferThread::GetTransfers()
|
||||
{
|
||||
std::lock_guard lk{m_transfers_mutex};
|
||||
return m_transfers.empty();
|
||||
}
|
||||
|
||||
} // namespace IOS::HLE::USB
|
@ -1,43 +0,0 @@
|
||||
#pragma once
|
||||
|
||||
#include <string>
|
||||
#include <thread>
|
||||
|
||||
#include "Common/CommonTypes.h"
|
||||
#include "Core/IOS/USB/Common.h"
|
||||
|
||||
namespace IOS::HLE::USB
|
||||
{
|
||||
class EmulatedUSBDevice : public Device
|
||||
{
|
||||
public:
|
||||
EmulatedUSBDevice(Kernel& ios, const std::string& device_name);
|
||||
virtual ~EmulatedUSBDevice();
|
||||
|
||||
protected:
|
||||
class FakeTransferThread final
|
||||
{
|
||||
public:
|
||||
explicit FakeTransferThread(EmulatedUSBDevice* device) : m_device(device) {}
|
||||
~FakeTransferThread();
|
||||
void Start();
|
||||
void Stop();
|
||||
void AddTransfer(std::unique_ptr<TransferCommand> command, std::array<u8, 64> data);
|
||||
void ClearTransfers();
|
||||
bool GetTransfers();
|
||||
|
||||
private:
|
||||
EmulatedUSBDevice* m_device = nullptr;
|
||||
Common::Flag m_thread_running;
|
||||
std::thread m_thread;
|
||||
Common::Flag m_is_initialized;
|
||||
std::map<std::array<u8, 64>, std::unique_ptr<TransferCommand>> m_transfers;
|
||||
std::mutex m_transfers_mutex;
|
||||
};
|
||||
FakeTransferThread m_transfer_thread{this};
|
||||
FakeTransferThread& GetTransferThread() { return m_transfer_thread; }
|
||||
|
||||
private:
|
||||
Kernel& m_ios;
|
||||
};
|
||||
} // namespace IOS::HLE::USB
|
@ -24,6 +24,8 @@
|
||||
#include "Core/IOS/USB/Common.h"
|
||||
#include "Core/IOS/USB/Emulated/Skylander.h"
|
||||
#include "Core/IOS/USB/LibusbDevice.h"
|
||||
#include "Core/NetPlayProto.h"
|
||||
#include "Core/System.h"
|
||||
|
||||
namespace IOS::HLE
|
||||
{
|
||||
@ -35,7 +37,7 @@ USBHost::~USBHost() = default;
|
||||
|
||||
std::optional<IPCReply> USBHost::Open(const OpenRequest& request)
|
||||
{
|
||||
if (!m_has_initialised && !Core::WantsDeterminism())
|
||||
if (!m_has_initialised)
|
||||
{
|
||||
GetScanThread().Start();
|
||||
// Force a device scan to complete, because some games (including Your Shape) only care
|
||||
@ -97,12 +99,15 @@ bool USBHost::ShouldAddDevice(const USB::Device& device) const
|
||||
return true;
|
||||
}
|
||||
|
||||
void USBHost::Update()
|
||||
{
|
||||
if (Core::WantsDeterminism())
|
||||
UpdateDevices();
|
||||
}
|
||||
|
||||
// This is called from the scan thread. Returns false if we failed to update the device list.
|
||||
bool USBHost::UpdateDevices(const bool always_add_hooks)
|
||||
{
|
||||
if (Core::WantsDeterminism())
|
||||
return true;
|
||||
|
||||
DeviceChangeHooks hooks;
|
||||
std::set<u64> plugged_devices;
|
||||
// If we failed to get a new, up-to-date list of devices, we cannot detect device removals.
|
||||
@ -116,43 +121,35 @@ bool USBHost::UpdateDevices(const bool always_add_hooks)
|
||||
bool USBHost::AddNewDevices(std::set<u64>& new_devices, DeviceChangeHooks& hooks,
|
||||
const bool always_add_hooks)
|
||||
{
|
||||
if (Config::EmulateSkylanderPortal())
|
||||
{
|
||||
auto skylanderportal = std::make_unique<USB::SkylanderUSB>(m_ios, "Skylander Portal");
|
||||
if (!ShouldAddDevice(*skylanderportal))
|
||||
return true;
|
||||
const u64 skyid = skylanderportal->GetId();
|
||||
new_devices.insert(skyid);
|
||||
if (AddDevice(std::move(skylanderportal)))
|
||||
{
|
||||
hooks.emplace(GetDeviceById(skyid), ChangeEvent::Inserted);
|
||||
}
|
||||
}
|
||||
AddEmulatedDevices(new_devices, hooks, always_add_hooks);
|
||||
#ifdef __LIBUSB__
|
||||
auto whitelist = Config::GetUSBDeviceWhitelist();
|
||||
if (whitelist.empty())
|
||||
return true;
|
||||
|
||||
if (m_context.IsValid())
|
||||
if (!Core::WantsDeterminism())
|
||||
{
|
||||
const int ret = m_context.GetDeviceList([&](libusb_device* device) {
|
||||
libusb_device_descriptor descriptor;
|
||||
libusb_get_device_descriptor(device, &descriptor);
|
||||
if (whitelist.count({descriptor.idVendor, descriptor.idProduct}) == 0)
|
||||
return true;
|
||||
|
||||
auto usb_device = std::make_unique<USB::LibusbDevice>(m_ios, device, descriptor);
|
||||
if (!ShouldAddDevice(*usb_device))
|
||||
return true;
|
||||
|
||||
const u64 id = usb_device->GetId();
|
||||
new_devices.insert(id);
|
||||
if (AddDevice(std::move(usb_device)) || always_add_hooks)
|
||||
hooks.emplace(GetDeviceById(id), ChangeEvent::Inserted);
|
||||
auto whitelist = Config::GetUSBDeviceWhitelist();
|
||||
if (whitelist.empty())
|
||||
return true;
|
||||
});
|
||||
if (ret != LIBUSB_SUCCESS)
|
||||
WARN_LOG_FMT(IOS_USB, "GetDeviceList failed: {}", LibusbUtils::ErrorWrap(ret));
|
||||
|
||||
if (m_context.IsValid())
|
||||
{
|
||||
const int ret = m_context.GetDeviceList([&](libusb_device* device) {
|
||||
libusb_device_descriptor descriptor;
|
||||
libusb_get_device_descriptor(device, &descriptor);
|
||||
if (whitelist.count({descriptor.idVendor, descriptor.idProduct}) == 0)
|
||||
return true;
|
||||
|
||||
auto usb_device = std::make_unique<USB::LibusbDevice>(m_ios, device, descriptor);
|
||||
if (!ShouldAddDevice(*usb_device))
|
||||
return true;
|
||||
|
||||
const u64 id = usb_device->GetId();
|
||||
new_devices.insert(id);
|
||||
if (AddDevice(std::move(usb_device)) || always_add_hooks)
|
||||
hooks.emplace(GetDeviceById(id), ChangeEvent::Inserted);
|
||||
return true;
|
||||
});
|
||||
if (ret != LIBUSB_SUCCESS)
|
||||
WARN_LOG_FMT(IOS_USB, "GetDeviceList failed: {}", LibusbUtils::ErrorWrap(ret));
|
||||
}
|
||||
}
|
||||
#endif
|
||||
return true;
|
||||
@ -188,6 +185,24 @@ void USBHost::DispatchHooks(const DeviceChangeHooks& hooks)
|
||||
OnDeviceChangeEnd();
|
||||
}
|
||||
|
||||
void USBHost::AddEmulatedDevices(std::set<u64>& new_devices, DeviceChangeHooks& hooks,
|
||||
bool always_add_hooks)
|
||||
{
|
||||
if (Config::Get(Config::MAIN_EMULATE_SKYLANDER_PORTAL) && !NetPlay::IsNetPlayRunning())
|
||||
{
|
||||
auto skylanderportal = std::make_unique<USB::SkylanderUSB>(m_ios, "Skylander Portal");
|
||||
if (ShouldAddDevice(*skylanderportal))
|
||||
{
|
||||
const u64 skyid = skylanderportal->GetId();
|
||||
new_devices.insert(skyid);
|
||||
if (AddDevice(std::move(skylanderportal)) || always_add_hooks)
|
||||
{
|
||||
hooks.emplace(GetDeviceById(skyid), ChangeEvent::Inserted);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
USBHost::ScanThread::~ScanThread()
|
||||
{
|
||||
Stop();
|
||||
@ -195,14 +210,19 @@ USBHost::ScanThread::~ScanThread()
|
||||
|
||||
void USBHost::ScanThread::WaitForFirstScan()
|
||||
{
|
||||
m_first_scan_complete_event.Wait();
|
||||
if (m_thread_running.IsSet())
|
||||
{
|
||||
m_first_scan_complete_event.Wait();
|
||||
}
|
||||
}
|
||||
|
||||
void USBHost::ScanThread::Start()
|
||||
{
|
||||
if (Core::WantsDeterminism())
|
||||
{
|
||||
m_host->UpdateDevices();
|
||||
return;
|
||||
|
||||
}
|
||||
if (m_thread_running.TestAndSet())
|
||||
{
|
||||
m_thread = std::thread([this] {
|
||||
|
@ -76,10 +76,13 @@ protected:
|
||||
|
||||
private:
|
||||
bool AddDevice(std::unique_ptr<USB::Device> device);
|
||||
void Update() override;
|
||||
bool UpdateDevices(bool always_add_hooks = false);
|
||||
bool AddNewDevices(std::set<u64>& new_devices, DeviceChangeHooks& hooks, bool always_add_hooks);
|
||||
void DetectRemovedDevices(const std::set<u64>& plugged_devices, DeviceChangeHooks& hooks);
|
||||
void DispatchHooks(const DeviceChangeHooks& hooks);
|
||||
void AddEmulatedDevices(std::set<u64>& new_devices, DeviceChangeHooks& hooks,
|
||||
bool always_add_hooks);
|
||||
|
||||
bool m_has_initialised = false;
|
||||
LibusbUtils::Context m_context;
|
||||
|
@ -20,6 +20,7 @@
|
||||
#include "Core/HW/SI/SI.h"
|
||||
#include "Core/HW/Sram.h"
|
||||
#include "Core/HW/VideoInterface.h"
|
||||
#include "IOS/USB/Emulated/Skylander.h"
|
||||
#include "VideoCommon/CommandProcessor.h"
|
||||
#include "VideoCommon/Fifo.h"
|
||||
#include "VideoCommon/GeometryShaderManager.h"
|
||||
@ -47,6 +48,7 @@ struct System::Impl
|
||||
Fifo::FifoManager m_fifo;
|
||||
GeometryShaderManager m_geometry_shader_manager;
|
||||
GPFifo::GPFifoManager m_gp_fifo;
|
||||
IOS::HLE::USB::SkylanderPortal m_skylander_portal;
|
||||
Memory::MemoryManager m_memory;
|
||||
MemoryInterface::MemoryInterfaceState m_memory_interface_state;
|
||||
PixelEngine::PixelEngineManager m_pixel_engine;
|
||||
@ -151,6 +153,11 @@ GPFifo::GPFifoManager& System::GetGPFifo() const
|
||||
return m_impl->m_gp_fifo;
|
||||
}
|
||||
|
||||
IOS::HLE::USB::SkylanderPortal& System::GetSkylanderPortal() const
|
||||
{
|
||||
return m_impl->m_skylander_portal;
|
||||
}
|
||||
|
||||
Memory::MemoryManager& System::GetMemory() const
|
||||
{
|
||||
return m_impl->m_memory;
|
||||
|
@ -47,6 +47,10 @@ namespace GPFifo
|
||||
{
|
||||
class GPFifoManager;
|
||||
}
|
||||
namespace IOS::HLE::USB
|
||||
{
|
||||
class SkylanderPortal;
|
||||
};
|
||||
namespace Memory
|
||||
{
|
||||
class MemoryManager;
|
||||
@ -116,6 +120,7 @@ public:
|
||||
Fifo::FifoManager& GetFifo() const;
|
||||
GeometryShaderManager& GetGeometryShaderManager() const;
|
||||
GPFifo::GPFifoManager& GetGPFifo() const;
|
||||
IOS::HLE::USB::SkylanderPortal& GetSkylanderPortal() const;
|
||||
Memory::MemoryManager& GetMemory() const;
|
||||
MemoryInterface::MemoryInterfaceState& GetMemoryInterfaceState() const;
|
||||
PixelEngine::PixelEngineManager& GetPixelEngine() const;
|
||||
|
@ -373,7 +373,6 @@
|
||||
<ClInclude Include="Core\IOS\USB\Bluetooth\WiimoteDevice.h" />
|
||||
<ClInclude Include="Core\IOS\USB\Bluetooth\WiimoteHIDAttr.h" />
|
||||
<ClInclude Include="Core\IOS\USB\Common.h" />
|
||||
<ClInclude Include="Core\IOS\USB\EmulatedUSBDevice.h" />
|
||||
<ClInclude Include="Core\IOS\USB\Emulated\Skylander.h" />
|
||||
<ClInclude Include="Core\IOS\USB\Host.h" />
|
||||
<ClInclude Include="Core\IOS\USB\LibusbDevice.h" />
|
||||
@ -987,7 +986,6 @@
|
||||
<ClCompile Include="Core\IOS\USB\Bluetooth\WiimoteDevice.cpp" />
|
||||
<ClCompile Include="Core\IOS\USB\Bluetooth\WiimoteHIDAttr.cpp" />
|
||||
<ClCompile Include="Core\IOS\USB\Common.cpp" />
|
||||
<ClCompile Include="Core\IOS\USB\EmulatedUSBDevice.cpp" />
|
||||
<ClCompile Include="Core\IOS\USB\Emulated\Skylander.cpp" />
|
||||
<ClCompile Include="Core\IOS\USB\Host.cpp" />
|
||||
<ClCompile Include="Core\IOS\USB\LibusbDevice.cpp" />
|
||||
|
@ -221,7 +221,7 @@ void MenuBar::AddToolsMenu()
|
||||
|
||||
tools_menu->addAction(tr("FIFO Player"), this, &MenuBar::ShowFIFOPlayer);
|
||||
|
||||
tools_menu->addAction(tr("&Skylanders Portal"), this, [this] { emit ShowSkylanderPortal(); });
|
||||
tools_menu->addAction(tr("&Skylanders Portal"), this, &MenuBar::ShowSkylanderPortal);
|
||||
|
||||
tools_menu->addSeparator();
|
||||
|
||||
|
@ -1,3 +1,6 @@
|
||||
// Copyright 2022 Dolphin Emulator Project
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
||||
// DolphinQt code copied and modified for Dolphin from the RPCS3 Qt utility for Creating, Loading
|
||||
// and Clearing skylanders
|
||||
|
||||
@ -6,12 +9,14 @@
|
||||
#include <QCheckBox>
|
||||
#include <QComboBox>
|
||||
#include <QCompleter>
|
||||
#include <QDialogButtonBox>
|
||||
#include <QEvent>
|
||||
#include <QFileDialog>
|
||||
#include <QGroupBox>
|
||||
#include <QKeyEvent>
|
||||
#include <QKeySequence>
|
||||
#include <QLabel>
|
||||
#include <QLineEdit>
|
||||
#include <QMessageBox>
|
||||
#include <QPushButton>
|
||||
#include <QScrollArea>
|
||||
@ -22,15 +27,16 @@
|
||||
#include "Common/IOFile.h"
|
||||
|
||||
#include "Core/Config/MainSettings.h"
|
||||
#include "Core/System.h"
|
||||
|
||||
#include "DolphinQt/QtUtils/DolphinFileDialog.h"
|
||||
#include "DolphinQt/Settings.h"
|
||||
|
||||
SkylanderPortalWindow* SkylanderPortalWindow::inst = nullptr;
|
||||
std::optional<std::tuple<u8, u16, u16>> SkylanderPortalWindow::sky_slots[MAX_SKYLANDERS];
|
||||
QString last_skylander_path;
|
||||
// Qt is not guaranteed to keep track of file paths using native file pickers, so we use this
|
||||
// static variable to ensure we open at the most recent Skylander file location
|
||||
static QString s_last_skylander_path;
|
||||
|
||||
const std::map<const std::pair<const u16, const u16>, const std::string> list_skylanders = {
|
||||
const std::map<const std::pair<const u16, const u16>, const char*> list_skylanders = {
|
||||
{{0, 0x0000}, "Whirlwind"},
|
||||
{{0, 0x1801}, "Series 2 Whirlwind"},
|
||||
{{0, 0x1C02}, "Polar Whirlwind"},
|
||||
@ -513,49 +519,10 @@ const std::map<const std::pair<const u16, const u16>, const std::string> list_sk
|
||||
{{3503, 0x0000}, "Kaos Trophy"},
|
||||
};
|
||||
|
||||
u16 skylander_crc16(u16 init_value, const u8* buffer, u32 size)
|
||||
{
|
||||
const unsigned short CRC_CCITT_TABLE[256] = {
|
||||
0x0000, 0x1021, 0x2042, 0x3063, 0x4084, 0x50A5, 0x60C6, 0x70E7, 0x8108, 0x9129, 0xA14A,
|
||||
0xB16B, 0xC18C, 0xD1AD, 0xE1CE, 0xF1EF, 0x1231, 0x0210, 0x3273, 0x2252, 0x52B5, 0x4294,
|
||||
0x72F7, 0x62D6, 0x9339, 0x8318, 0xB37B, 0xA35A, 0xD3BD, 0xC39C, 0xF3FF, 0xE3DE, 0x2462,
|
||||
0x3443, 0x0420, 0x1401, 0x64E6, 0x74C7, 0x44A4, 0x5485, 0xA56A, 0xB54B, 0x8528, 0x9509,
|
||||
0xE5EE, 0xF5CF, 0xC5AC, 0xD58D, 0x3653, 0x2672, 0x1611, 0x0630, 0x76D7, 0x66F6, 0x5695,
|
||||
0x46B4, 0xB75B, 0xA77A, 0x9719, 0x8738, 0xF7DF, 0xE7FE, 0xD79D, 0xC7BC, 0x48C4, 0x58E5,
|
||||
0x6886, 0x78A7, 0x0840, 0x1861, 0x2802, 0x3823, 0xC9CC, 0xD9ED, 0xE98E, 0xF9AF, 0x8948,
|
||||
0x9969, 0xA90A, 0xB92B, 0x5AF5, 0x4AD4, 0x7AB7, 0x6A96, 0x1A71, 0x0A50, 0x3A33, 0x2A12,
|
||||
0xDBFD, 0xCBDC, 0xFBBF, 0xEB9E, 0x9B79, 0x8B58, 0xBB3B, 0xAB1A, 0x6CA6, 0x7C87, 0x4CE4,
|
||||
0x5CC5, 0x2C22, 0x3C03, 0x0C60, 0x1C41, 0xEDAE, 0xFD8F, 0xCDEC, 0xDDCD, 0xAD2A, 0xBD0B,
|
||||
0x8D68, 0x9D49, 0x7E97, 0x6EB6, 0x5ED5, 0x4EF4, 0x3E13, 0x2E32, 0x1E51, 0x0E70, 0xFF9F,
|
||||
0xEFBE, 0xDFDD, 0xCFFC, 0xBF1B, 0xAF3A, 0x9F59, 0x8F78, 0x9188, 0x81A9, 0xB1CA, 0xA1EB,
|
||||
0xD10C, 0xC12D, 0xF14E, 0xE16F, 0x1080, 0x00A1, 0x30C2, 0x20E3, 0x5004, 0x4025, 0x7046,
|
||||
0x6067, 0x83B9, 0x9398, 0xA3FB, 0xB3DA, 0xC33D, 0xD31C, 0xE37F, 0xF35E, 0x02B1, 0x1290,
|
||||
0x22F3, 0x32D2, 0x4235, 0x5214, 0x6277, 0x7256, 0xB5EA, 0xA5CB, 0x95A8, 0x8589, 0xF56E,
|
||||
0xE54F, 0xD52C, 0xC50D, 0x34E2, 0x24C3, 0x14A0, 0x0481, 0x7466, 0x6447, 0x5424, 0x4405,
|
||||
0xA7DB, 0xB7FA, 0x8799, 0x97B8, 0xE75F, 0xF77E, 0xC71D, 0xD73C, 0x26D3, 0x36F2, 0x0691,
|
||||
0x16B0, 0x6657, 0x7676, 0x4615, 0x5634, 0xD94C, 0xC96D, 0xF90E, 0xE92F, 0x99C8, 0x89E9,
|
||||
0xB98A, 0xA9AB, 0x5844, 0x4865, 0x7806, 0x6827, 0x18C0, 0x08E1, 0x3882, 0x28A3, 0xCB7D,
|
||||
0xDB5C, 0xEB3F, 0xFB1E, 0x8BF9, 0x9BD8, 0xABBB, 0xBB9A, 0x4A75, 0x5A54, 0x6A37, 0x7A16,
|
||||
0x0AF1, 0x1AD0, 0x2AB3, 0x3A92, 0xFD2E, 0xED0F, 0xDD6C, 0xCD4D, 0xBDAA, 0xAD8B, 0x9DE8,
|
||||
0x8DC9, 0x7C26, 0x6C07, 0x5C64, 0x4C45, 0x3CA2, 0x2C83, 0x1CE0, 0x0CC1, 0xEF1F, 0xFF3E,
|
||||
0xCF5D, 0xDF7C, 0xAF9B, 0xBFBA, 0x8FD9, 0x9FF8, 0x6E17, 0x7E36, 0x4E55, 0x5E74, 0x2E93,
|
||||
0x3EB2, 0x0ED1, 0x1EF0};
|
||||
|
||||
u16 crc = init_value;
|
||||
|
||||
for (u32 i = 0; i < size; i++)
|
||||
{
|
||||
const u16 tmp = (crc >> 8) ^ buffer[i];
|
||||
crc = (crc << 8) ^ CRC_CCITT_TABLE[tmp];
|
||||
}
|
||||
|
||||
return crc;
|
||||
}
|
||||
|
||||
SkylanderPortalWindow::SkylanderPortalWindow(QWidget* parent) : QWidget(parent)
|
||||
{
|
||||
setWindowTitle(tr("Skylanders Manager"));
|
||||
setObjectName(QString::fromStdString("skylanders_manager"));
|
||||
setObjectName(tr("skylanders_manager"));
|
||||
setMinimumSize(QSize(700, 200));
|
||||
|
||||
CreateMainWindow();
|
||||
@ -568,33 +535,32 @@ SkylanderPortalWindow::SkylanderPortalWindow(QWidget* parent) : QWidget(parent)
|
||||
OnEmulationStateChanged(Core::GetState());
|
||||
};
|
||||
|
||||
SkylanderPortalWindow::~SkylanderPortalWindow()
|
||||
{
|
||||
}
|
||||
SkylanderPortalWindow::~SkylanderPortalWindow() = default;
|
||||
|
||||
void SkylanderPortalWindow::CreateMainWindow()
|
||||
{
|
||||
QVBoxLayout* mainLayout = new QVBoxLayout();
|
||||
auto* main_layout = new QVBoxLayout();
|
||||
|
||||
QGroupBox* checkbox_group = new QGroupBox();
|
||||
QHBoxLayout* checkboxLayout = new QHBoxLayout();
|
||||
checkboxLayout->setAlignment(Qt::AlignHCenter);
|
||||
checkbox = new QCheckBox(QString::fromStdString("Emulate Skylander Portal"), this);
|
||||
connect(checkbox, &QCheckBox::toggled, [=](bool checked) { EmulatePortal(checked); });
|
||||
checkboxLayout->addWidget(checkbox);
|
||||
checkbox_group->setLayout(checkboxLayout);
|
||||
mainLayout->addWidget(checkbox_group);
|
||||
auto* checkbox_group = new QGroupBox();
|
||||
auto* checkbox_layout = new QHBoxLayout();
|
||||
checkbox_layout->setAlignment(Qt::AlignHCenter);
|
||||
m_checkbox = new QCheckBox(tr("Emulate Skylander Portal"), this);
|
||||
m_checkbox->setChecked(Config::Get(Config::MAIN_EMULATE_SKYLANDER_PORTAL));
|
||||
connect(m_checkbox, &QCheckBox::toggled, [=](bool checked) { EmulatePortal(checked); });
|
||||
checkbox_layout->addWidget(m_checkbox);
|
||||
checkbox_group->setLayout(checkbox_layout);
|
||||
main_layout->addWidget(checkbox_group);
|
||||
|
||||
auto add_line = [](QVBoxLayout* vbox) {
|
||||
QFrame* line = new QFrame();
|
||||
auto* line = new QFrame();
|
||||
line->setFrameShape(QFrame::HLine);
|
||||
line->setFrameShadow(QFrame::Sunken);
|
||||
vbox->addWidget(line);
|
||||
};
|
||||
|
||||
group_skylanders = new QGroupBox(tr("Active Portal Skylanders:"));
|
||||
QVBoxLayout* vbox_group = new QVBoxLayout();
|
||||
QScrollArea* scroll_area = new QScrollArea();
|
||||
m_group_skylanders = new QGroupBox(tr("Active Portal Skylanders:"));
|
||||
auto* vbox_group = new QVBoxLayout();
|
||||
auto* scroll_area = new QScrollArea();
|
||||
|
||||
for (auto i = 0; i < MAX_SKYLANDERS; i++)
|
||||
{
|
||||
@ -603,21 +569,21 @@ void SkylanderPortalWindow::CreateMainWindow()
|
||||
add_line(vbox_group);
|
||||
}
|
||||
|
||||
QHBoxLayout* hbox_skylander = new QHBoxLayout();
|
||||
QLabel* label_skyname = new QLabel(QString(tr("Skylander %1")).arg(i + 1));
|
||||
edit_skylanders[i] = new QLineEdit();
|
||||
edit_skylanders[i]->setEnabled(false);
|
||||
auto* hbox_skylander = new QHBoxLayout();
|
||||
auto* label_skyname = new QLabel(QString(tr("Skylander %1")).arg(i + 1));
|
||||
m_edit_skylanders[i] = new QLineEdit();
|
||||
m_edit_skylanders[i]->setEnabled(false);
|
||||
|
||||
QPushButton* clear_btn = new QPushButton(tr("Clear"));
|
||||
QPushButton* create_btn = new QPushButton(tr("Create"));
|
||||
QPushButton* load_btn = new QPushButton(tr("Load"));
|
||||
auto* clear_btn = new QPushButton(tr("Clear"));
|
||||
auto* create_btn = new QPushButton(tr("Create"));
|
||||
auto* load_btn = new QPushButton(tr("Load"));
|
||||
|
||||
connect(clear_btn, &QAbstractButton::clicked, this, [this, i]() { ClearSkylander(i); });
|
||||
connect(create_btn, &QAbstractButton::clicked, this, [this, i]() { CreateSkylander(i); });
|
||||
connect(load_btn, &QAbstractButton::clicked, this, [this, i]() { LoadSkylander(i); });
|
||||
|
||||
hbox_skylander->addWidget(label_skyname);
|
||||
hbox_skylander->addWidget(edit_skylanders[i]);
|
||||
hbox_skylander->addWidget(m_edit_skylanders[i]);
|
||||
hbox_skylander->addWidget(clear_btn);
|
||||
hbox_skylander->addWidget(create_btn);
|
||||
hbox_skylander->addWidget(load_btn);
|
||||
@ -625,12 +591,12 @@ void SkylanderPortalWindow::CreateMainWindow()
|
||||
vbox_group->addLayout(hbox_skylander);
|
||||
}
|
||||
|
||||
group_skylanders->setLayout(vbox_group);
|
||||
scroll_area->setWidget(group_skylanders);
|
||||
m_group_skylanders->setLayout(vbox_group);
|
||||
scroll_area->setWidget(m_group_skylanders);
|
||||
scroll_area->setWidgetResizable(true);
|
||||
group_skylanders->setVisible(false);
|
||||
mainLayout->addWidget(scroll_area);
|
||||
setLayout(mainLayout);
|
||||
m_group_skylanders->setVisible(Config::Get(Config::MAIN_EMULATE_SKYLANDER_PORTAL));
|
||||
main_layout->addWidget(scroll_area);
|
||||
setLayout(main_layout);
|
||||
|
||||
UpdateEdits();
|
||||
}
|
||||
@ -639,29 +605,29 @@ void SkylanderPortalWindow::OnEmulationStateChanged(Core::State state)
|
||||
{
|
||||
const bool running = state != Core::State::Uninitialized;
|
||||
|
||||
checkbox->setEnabled(!running);
|
||||
m_checkbox->setEnabled(!running);
|
||||
}
|
||||
|
||||
CreateSkylanderDialog::CreateSkylanderDialog(QWidget* parent) : QDialog(parent)
|
||||
{
|
||||
setWindowTitle(tr("Skylander Creator"));
|
||||
setObjectName(QString::fromStdString("skylanders_creator"));
|
||||
setObjectName(tr("skylanders_creator"));
|
||||
setMinimumSize(QSize(500, 150));
|
||||
QVBoxLayout* layout = new QVBoxLayout;
|
||||
auto* layout = new QVBoxLayout;
|
||||
|
||||
QComboBox* combo_skylist = new QComboBox();
|
||||
auto* combo_skylist = new QComboBox();
|
||||
QStringList filterlist;
|
||||
for (const auto& entry : list_skylanders)
|
||||
{
|
||||
const uint qvar = (entry.first.first << 16) | entry.first.second;
|
||||
combo_skylist->addItem(QString::fromStdString(entry.second), QVariant(qvar));
|
||||
filterlist << QString::fromStdString(entry.second.c_str());
|
||||
combo_skylist->addItem(tr(entry.second), QVariant(qvar));
|
||||
filterlist << tr(entry.second);
|
||||
}
|
||||
combo_skylist->addItem(tr("--Unknown--"), QVariant(0xFFFFFFFF));
|
||||
combo_skylist->setEditable(true);
|
||||
combo_skylist->setInsertPolicy(QComboBox::NoInsert);
|
||||
|
||||
QCompleter* co_compl = new QCompleter(filterlist, this);
|
||||
auto* co_compl = new QCompleter(filterlist, this);
|
||||
co_compl->setCaseSensitivity(Qt::CaseInsensitive);
|
||||
co_compl->setCompletionMode(QCompleter::PopupCompletion);
|
||||
co_compl->setFilterMode(Qt::MatchContains);
|
||||
@ -669,18 +635,17 @@ CreateSkylanderDialog::CreateSkylanderDialog(QWidget* parent) : QDialog(parent)
|
||||
|
||||
layout->addWidget(combo_skylist);
|
||||
|
||||
QFrame* line = new QFrame();
|
||||
auto* line = new QFrame();
|
||||
line->setFrameShape(QFrame::HLine);
|
||||
line->setFrameShadow(QFrame::Sunken);
|
||||
layout->addWidget(line);
|
||||
|
||||
QHBoxLayout* hbox_idvar = new QHBoxLayout();
|
||||
QLabel* label_id = new QLabel(tr("ID:"));
|
||||
QLabel* label_var = new QLabel(tr("Variant:"));
|
||||
QLineEdit* edit_id = new QLineEdit(QString::fromStdString("0"));
|
||||
QLineEdit* edit_var = new QLineEdit(QString::fromStdString("0"));
|
||||
QRegularExpressionValidator* rxv =
|
||||
new QRegularExpressionValidator(QRegularExpression(QString::fromStdString("\\d*")), this);
|
||||
auto* hbox_idvar = new QHBoxLayout();
|
||||
auto* label_id = new QLabel(tr("ID:"));
|
||||
auto* label_var = new QLabel(tr("Variant:"));
|
||||
auto* edit_id = new QLineEdit(tr("0"));
|
||||
auto* edit_var = new QLineEdit(tr("0"));
|
||||
auto* rxv = new QRegularExpressionValidator(QRegularExpression(tr("\\d*")), this);
|
||||
edit_id->setValidator(rxv);
|
||||
edit_var->setValidator(rxv);
|
||||
hbox_idvar->addWidget(label_id);
|
||||
@ -689,13 +654,9 @@ CreateSkylanderDialog::CreateSkylanderDialog(QWidget* parent) : QDialog(parent)
|
||||
hbox_idvar->addWidget(edit_var);
|
||||
layout->addLayout(hbox_idvar);
|
||||
|
||||
QHBoxLayout* hbox_buttons = new QHBoxLayout();
|
||||
QPushButton* btn_create = new QPushButton(tr("Create"), this);
|
||||
QPushButton* btn_cancel = new QPushButton(tr("Cancel"), this);
|
||||
hbox_buttons->addStretch();
|
||||
hbox_buttons->addWidget(btn_create);
|
||||
hbox_buttons->addWidget(btn_cancel);
|
||||
layout->addLayout(hbox_buttons);
|
||||
auto* buttons = new QDialogButtonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel);
|
||||
buttons->button(QDialogButtonBox::Ok)->setText(tr("Create"));
|
||||
layout->addWidget(buttons);
|
||||
|
||||
setLayout(layout);
|
||||
|
||||
@ -711,7 +672,7 @@ CreateSkylanderDialog::CreateSkylanderDialog(QWidget* parent) : QDialog(parent)
|
||||
}
|
||||
});
|
||||
|
||||
connect(btn_create, &QAbstractButton::clicked, this, [=, this]() {
|
||||
connect(buttons, &QDialogButtonBox::accepted, this, [=, this]() {
|
||||
bool ok_id = false, ok_var = false;
|
||||
const u16 sky_id = edit_id->text().toUShort(&ok_id);
|
||||
if (!ok_id)
|
||||
@ -728,61 +689,40 @@ CreateSkylanderDialog::CreateSkylanderDialog(QWidget* parent) : QDialog(parent)
|
||||
return;
|
||||
}
|
||||
|
||||
QString predef_name = last_skylander_path;
|
||||
QString predef_name = s_last_skylander_path;
|
||||
const auto found_sky = list_skylanders.find(std::make_pair(sky_id, sky_var));
|
||||
if (found_sky != list_skylanders.end())
|
||||
{
|
||||
predef_name += QString::fromStdString(found_sky->second + ".sky");
|
||||
std::string name = std::string(found_sky->second) + ".sky";
|
||||
predef_name += tr(name.c_str());
|
||||
}
|
||||
else
|
||||
{
|
||||
QString str = QString::fromStdString("Unknown(%1 %2).sky");
|
||||
QString str = tr("Unknown(%1 %2).sky");
|
||||
predef_name += str.arg(sky_id, sky_var);
|
||||
}
|
||||
|
||||
file_path = QFileDialog::getSaveFileName(this, tr("Create Skylander File"), predef_name,
|
||||
tr("Skylander Object (*.sky);;"));
|
||||
if (file_path.isEmpty())
|
||||
m_file_path = QFileDialog::getSaveFileName(this, tr("Create Skylander File"), predef_name,
|
||||
tr("Skylander Object (*.sky);;"));
|
||||
if (m_file_path.isEmpty())
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
File::IOFile sky_file(file_path.toStdString(), "w+b");
|
||||
if (!sky_file)
|
||||
auto& system = Core::System::GetInstance();
|
||||
|
||||
if (!system.GetSkylanderPortal().CreateSkylander(m_file_path.toStdString(), sky_id, sky_var))
|
||||
{
|
||||
QMessageBox::warning(this, tr("Failed to create skylander file!"),
|
||||
tr("Failed to create skylander file:\n%1").arg(file_path),
|
||||
tr("Failed to create skylander file:\n%1").arg(m_file_path),
|
||||
QMessageBox::Ok);
|
||||
return;
|
||||
}
|
||||
|
||||
std::array<u8, 0x40 * 0x10> buf{};
|
||||
const auto file_data = buf.data();
|
||||
// Set the block permissions
|
||||
u32 first_block = 0x690F0F0F;
|
||||
u32 other_blocks = 0x69080F7F;
|
||||
memcpy(&file_data[0x36], &first_block, sizeof(first_block));
|
||||
for (u32 index = 1; index < 0x10; index++)
|
||||
{
|
||||
memcpy(&file_data[(index * 0x40) + 0x36], &other_blocks, sizeof(other_blocks));
|
||||
}
|
||||
// Set the skylander info
|
||||
u16 sky_info = (sky_id | sky_var) + 1;
|
||||
memcpy(&file_data[0], &sky_info, sizeof(sky_info));
|
||||
memcpy(&file_data[0x10], &sky_id, sizeof(sky_id));
|
||||
memcpy(&file_data[0x1C], &sky_var, sizeof(sky_var));
|
||||
// Set checksum
|
||||
u16 checksum = skylander_crc16(0xFFFF, file_data, 0x1E);
|
||||
memcpy(&file_data[0x1E], &checksum, sizeof(checksum));
|
||||
|
||||
sky_file.WriteBytes(buf.data(), buf.size());
|
||||
sky_file.Close();
|
||||
|
||||
last_skylander_path = QFileInfo(file_path).absolutePath() + QString::fromStdString("/");
|
||||
s_last_skylander_path = QFileInfo(m_file_path).absolutePath() + tr("/");
|
||||
accept();
|
||||
});
|
||||
|
||||
connect(btn_cancel, &QAbstractButton::clicked, this, &QDialog::reject);
|
||||
connect(buttons, &QDialogButtonBox::rejected, this, &QDialog::reject);
|
||||
|
||||
connect(co_compl, QOverload<const QString&>::of(&QCompleter::activated),
|
||||
[=](const QString& text) {
|
||||
@ -793,13 +733,13 @@ CreateSkylanderDialog::CreateSkylanderDialog(QWidget* parent) : QDialog(parent)
|
||||
|
||||
QString CreateSkylanderDialog::GetFilePath() const
|
||||
{
|
||||
return file_path;
|
||||
return m_file_path;
|
||||
}
|
||||
|
||||
void SkylanderPortalWindow::EmulatePortal(bool emulate)
|
||||
{
|
||||
Config::SetBaseOrCurrent(Config::MAIN_EMULATE_SKYLANDER_PORTAL, emulate);
|
||||
group_skylanders->setVisible(emulate);
|
||||
m_group_skylanders->setVisible(emulate);
|
||||
}
|
||||
|
||||
void SkylanderPortalWindow::CreateSkylander(u8 slot)
|
||||
@ -814,12 +754,12 @@ void SkylanderPortalWindow::CreateSkylander(u8 slot)
|
||||
void SkylanderPortalWindow::LoadSkylander(u8 slot)
|
||||
{
|
||||
const QString file_path = DolphinFileDialog::getOpenFileName(
|
||||
this, tr("Select Skylander File"), last_skylander_path, tr("Skylander (*.sky);;"));
|
||||
this, tr("Select Skylander File"), s_last_skylander_path, tr("Skylander (*.sky);;"));
|
||||
if (file_path.isEmpty())
|
||||
{
|
||||
return;
|
||||
}
|
||||
last_skylander_path = QFileInfo(file_path).absolutePath() + QString::fromStdString("/");
|
||||
s_last_skylander_path = QFileInfo(file_path).absolutePath() + tr("/");
|
||||
|
||||
LoadSkylanderPath(slot, file_path);
|
||||
}
|
||||
@ -859,19 +799,26 @@ void SkylanderPortalWindow::LoadSkylanderPath(u8 slot, const QString& path)
|
||||
DEBUG_LOG_FMT(IOS_USB, "Sky Var: {}, 0x1D: {} 0x1C: {}", sky_var, file_data[0x1D],
|
||||
file_data[0x1C]);
|
||||
|
||||
u8 portal_slot = IOS::HLE::USB::g_skyportal.LoadSkylander(file_data.data(), std::move(sky_file));
|
||||
sky_slots[slot] = std::tuple(portal_slot, sky_id, sky_var);
|
||||
auto& system = Core::System::GetInstance();
|
||||
|
||||
u8 portal_slot = system.GetSkylanderPortal().LoadSkylander(file_data.data(), std::move(sky_file));
|
||||
if (portal_slot == 0xFF)
|
||||
{
|
||||
QMessageBox::warning(this, tr("Failed to load the skylander file!"),
|
||||
tr("Failed to load the skylander file(%1)!\n").arg(path), QMessageBox::Ok);
|
||||
return;
|
||||
}
|
||||
m_sky_slots[slot] = {portal_slot, sky_id, sky_var};
|
||||
UpdateEdits();
|
||||
}
|
||||
|
||||
void SkylanderPortalWindow::ClearSkylander(u8 slot)
|
||||
{
|
||||
if (auto slot_infos = sky_slots[slot])
|
||||
auto& system = Core::System::GetInstance();
|
||||
if (auto slot_infos = m_sky_slots[slot])
|
||||
{
|
||||
auto [cur_slot, id, var] = slot_infos.value();
|
||||
IOS::HLE::USB::g_skyportal.RemoveSkylander(cur_slot);
|
||||
sky_slots[slot] = {};
|
||||
system.GetSkylanderPortal().RemoveSkylander(slot_infos->portal_slot);
|
||||
m_sky_slots[slot].reset();
|
||||
UpdateEdits();
|
||||
}
|
||||
}
|
||||
@ -881,26 +828,24 @@ void SkylanderPortalWindow::UpdateEdits()
|
||||
for (auto i = 0; i < MAX_SKYLANDERS; i++)
|
||||
{
|
||||
QString display_string;
|
||||
if (auto sd = sky_slots[i])
|
||||
if (auto sd = m_sky_slots[i])
|
||||
{
|
||||
auto [portal_slot, sky_id, sky_var] = sd.value();
|
||||
auto found_sky = list_skylanders.find(std::make_pair(sky_id, sky_var));
|
||||
auto found_sky = list_skylanders.find(std::make_pair(sd->sky_id, sd->sky_var));
|
||||
if (found_sky != list_skylanders.end())
|
||||
{
|
||||
display_string = QString::fromStdString(found_sky->second);
|
||||
display_string = tr(found_sky->second);
|
||||
}
|
||||
else
|
||||
{
|
||||
display_string =
|
||||
QString(QString::fromStdString("Unknown (Id:%1 Var:%2)")).arg(sky_id).arg(sky_var);
|
||||
display_string = QString(tr("Unknown (Id:%1 Var:%2)")).arg(sd->sky_id).arg(sd->sky_var);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
display_string = QString::fromStdString("None");
|
||||
display_string = tr("None");
|
||||
}
|
||||
|
||||
edit_skylanders[i]->setText(display_string);
|
||||
m_edit_skylanders[i]->setText(display_string);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1,32 +1,39 @@
|
||||
// Copyright 2022 Dolphin Emulator Project
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <array>
|
||||
#include <optional>
|
||||
|
||||
#include <QCheckBox>
|
||||
#include <QDialog>
|
||||
#include <QGroupBox>
|
||||
#include <QLineEdit>
|
||||
#include <QString>
|
||||
#include <QWidget>
|
||||
|
||||
#include "Core/Core.h"
|
||||
#include "Core/IOS/USB/Emulated/Skylander.h"
|
||||
|
||||
class QDialogButtonBox;
|
||||
class QLabel;
|
||||
class QPushButton;
|
||||
class QSpinBox;
|
||||
class QTabWidget;
|
||||
class QCheckBox;
|
||||
class QGroupBox;
|
||||
class QLineEdit;
|
||||
|
||||
struct Skylander
|
||||
{
|
||||
u8 portal_slot;
|
||||
u16 sky_id;
|
||||
u16 sky_var;
|
||||
};
|
||||
|
||||
class SkylanderPortalWindow : public QWidget
|
||||
{
|
||||
Q_OBJECT
|
||||
public:
|
||||
explicit SkylanderPortalWindow(QWidget* parent = nullptr);
|
||||
~SkylanderPortalWindow();
|
||||
~SkylanderPortalWindow() override;
|
||||
|
||||
protected:
|
||||
QLineEdit* edit_skylanders[MAX_SKYLANDERS]{};
|
||||
static std::optional<std::tuple<u8, u16, u16>> sky_slots[MAX_SKYLANDERS];
|
||||
std::array<QLineEdit*, MAX_SKYLANDERS> m_edit_skylanders;
|
||||
std::array<std::optional<Skylander>, MAX_SKYLANDERS> m_sky_slots;
|
||||
|
||||
private:
|
||||
void CreateMainWindow();
|
||||
@ -38,13 +45,10 @@ private:
|
||||
void LoadSkylanderPath(u8 slot, const QString& path);
|
||||
void UpdateEdits();
|
||||
void closeEvent(QCloseEvent* bar) override;
|
||||
|
||||
static SkylanderPortalWindow* inst;
|
||||
|
||||
QCheckBox* checkbox;
|
||||
QGroupBox* group_skylanders;
|
||||
|
||||
bool eventFilter(QObject* object, QEvent* event) final override;
|
||||
|
||||
QCheckBox* m_checkbox;
|
||||
QGroupBox* m_group_skylanders;
|
||||
};
|
||||
|
||||
class CreateSkylanderDialog : public QDialog
|
||||
@ -56,5 +60,5 @@ public:
|
||||
QString GetFilePath() const;
|
||||
|
||||
protected:
|
||||
QString file_path;
|
||||
QString m_file_path;
|
||||
};
|
||||
|
Loading…
x
Reference in New Issue
Block a user