mirror of
https://github.com/PabloMK7/citra.git
synced 2025-09-09 04:10:05 +00:00
Emulated Skylander Infrared Portal
This commit is contained in:
parent
4f174f1c0b
commit
bbaca4cb6b
11 changed files with 1441 additions and 12 deletions
|
@ -136,6 +136,8 @@ add_executable(citra-qt
|
||||||
game_list_worker.h
|
game_list_worker.h
|
||||||
hotkeys.cpp
|
hotkeys.cpp
|
||||||
hotkeys.h
|
hotkeys.h
|
||||||
|
infrared/skylanderportal/skylander_dialog.cpp
|
||||||
|
infrared/skylanderportal/skylander_dialog.h
|
||||||
loading_screen.cpp
|
loading_screen.cpp
|
||||||
loading_screen.h
|
loading_screen.h
|
||||||
loading_screen.ui
|
loading_screen.ui
|
||||||
|
|
854
src/citra_qt/infrared/skylanderportal/skylander_dialog.cpp
Normal file
854
src/citra_qt/infrared/skylanderportal/skylander_dialog.cpp
Normal file
|
@ -0,0 +1,854 @@
|
||||||
|
// Copyright 2024 Citra Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include "skylander_dialog.h"
|
||||||
|
|
||||||
|
#include "common/file_util.h"
|
||||||
|
#include "core/hle/service/ir/ir_portal.h"
|
||||||
|
|
||||||
|
#include <random>
|
||||||
|
|
||||||
|
#include <QComboBox>
|
||||||
|
#include <QCompleter>
|
||||||
|
#include <QFileDialog>
|
||||||
|
#include <QGroupBox>
|
||||||
|
#include <QLabel>
|
||||||
|
#include <QMessageBox>
|
||||||
|
#include <QPushButton>
|
||||||
|
#include <QStringList>
|
||||||
|
#include <QVBoxLayout>
|
||||||
|
|
||||||
|
SkylanderPortalWindow* SkylanderPortalWindow::inst = nullptr;
|
||||||
|
std::optional<std::tuple<u8, u16, u16>> SkylanderPortalWindow::sky_slots[UI_SKY_NUM];
|
||||||
|
QString last_skylander_path;
|
||||||
|
|
||||||
|
const std::map<const std::pair<const u16, const u16>, const std::string> list_skylanders = {
|
||||||
|
{{0, 0x0000}, "Whirlwind"},
|
||||||
|
{{0, 0x1801}, "Series 2 Whirlwind"},
|
||||||
|
{{0, 0x1C02}, "Polar Whirlwind"},
|
||||||
|
{{0, 0x2805}, "Horn Blast Whirlwind"},
|
||||||
|
{{0, 0x3810}, "Eon's Elite Whirlwind"},
|
||||||
|
{{1, 0x0000}, "Sonic Boom"},
|
||||||
|
{{1, 0x1801}, "Series 2 Sonic Boom"},
|
||||||
|
{{2, 0x0000}, "Warnado"},
|
||||||
|
{{2, 0x2206}, "LightCore Warnado"},
|
||||||
|
{{3, 0x0000}, "Lightning Rod"},
|
||||||
|
{{3, 0x1801}, "Series 2 Lightning Rod"},
|
||||||
|
{{4, 0x0000}, "Bash"},
|
||||||
|
{{4, 0x1801}, "Series 2 Bash"},
|
||||||
|
{{5, 0x0000}, "Terrafin"},
|
||||||
|
{{5, 0x1801}, "Series 2 Terrafin"},
|
||||||
|
{{5, 0x2805}, "Knockout Terrafin"},
|
||||||
|
{{5, 0x3810}, "Eon's Elite Terrafin"},
|
||||||
|
{{6, 0x0000}, "Dino Rang"},
|
||||||
|
{{6, 0x4810}, "Eon's Elite Dino Rang"},
|
||||||
|
{{7, 0x0000}, "Prism Break"},
|
||||||
|
{{7, 0x1801}, "Series 2 Prism Break"},
|
||||||
|
{{7, 0x2805}, "Hyper Beam Prism Break"},
|
||||||
|
{{7, 0x1206}, "LightCore Prism Break"},
|
||||||
|
{{8, 0x0000}, "Sunburn"},
|
||||||
|
{{9, 0x0000}, "Eruptor"},
|
||||||
|
{{9, 0x1801}, "Series 2 Eruptor"},
|
||||||
|
{{9, 0x2C02}, "Volcanic Eruptor"},
|
||||||
|
{{9, 0x2805}, "Lava Barf Eruptor"},
|
||||||
|
{{9, 0x1206}, "LightCore Eruptor"},
|
||||||
|
{{9, 0x3810}, "Eon's Elite Eruptor"},
|
||||||
|
{{10, 0x0000}, "Ignitor"},
|
||||||
|
{{10, 0x1801}, "Series 2 Ignitor"},
|
||||||
|
{{10, 0x1C03}, "Legendary Ignitor"},
|
||||||
|
{{11, 0x0000}, "Flameslinger"},
|
||||||
|
{{11, 0x1801}, "Series 2 Flameslinger"},
|
||||||
|
{{12, 0x0000}, "Zap"},
|
||||||
|
{{12, 0x1801}, "Series 2 Zap"},
|
||||||
|
{{13, 0x0000}, "Wham Shell"},
|
||||||
|
{{13, 0x2206}, "LightCore Wham Shell"},
|
||||||
|
{{14, 0x0000}, "Gill Grunt"},
|
||||||
|
{{14, 0x1801}, "Series 2 Gill Grunt"},
|
||||||
|
{{14, 0x2805}, "Anchors Away Gill Grunt"},
|
||||||
|
{{14, 0x3805}, "Tidal Wave Gill Grunt"},
|
||||||
|
{{14, 0x3810}, "Eon's Elite Gill Grunt"},
|
||||||
|
{{15, 0x0000}, "Slam Bam"},
|
||||||
|
{{15, 0x1801}, "Series 2 Slam Bam"},
|
||||||
|
{{15, 0x1C03}, "Legendary Slam Bam"},
|
||||||
|
{{15, 0x4810}, "Eon's Elite Slam Bam"},
|
||||||
|
{{16, 0x0000}, "Spyro"},
|
||||||
|
{{16, 0x1801}, "Series 2 Spyro"},
|
||||||
|
{{16, 0x2C02}, "Dark Mega Ram Spyro"},
|
||||||
|
{{16, 0x2805}, "Mega Ram Spyro"},
|
||||||
|
{{16, 0x3810}, "Eon's Elite Spyro"},
|
||||||
|
{{17, 0x0000}, "Voodood"},
|
||||||
|
{{17, 0x4810}, "Eon's Elite Voodood"},
|
||||||
|
{{18, 0x0000}, "Double Trouble"},
|
||||||
|
{{18, 0x1801}, "Series 2 Double Trouble"},
|
||||||
|
{{18, 0x1C02}, "Royal Double Trouble"},
|
||||||
|
{{19, 0x0000}, "Trigger Happy"},
|
||||||
|
{{19, 0x1801}, "Series 2 Trigger Happy"},
|
||||||
|
{{19, 0x2C02}, "Springtime Trigger Happy"},
|
||||||
|
{{19, 0x2805}, "Big Bang Trigger Happy"},
|
||||||
|
{{19, 0x3810}, "Eon's Elite Trigger Happy"},
|
||||||
|
{{20, 0x0000}, "Drobot"},
|
||||||
|
{{20, 0x1801}, "Series 2 Drobot"},
|
||||||
|
{{20, 0x1206}, "LightCore Drobot"},
|
||||||
|
{{21, 0x0000}, "Drill Seargeant"},
|
||||||
|
{{21, 0x1801}, "Series 2 Drill Seargeant"},
|
||||||
|
{{22, 0x0000}, "Boomer"},
|
||||||
|
{{22, 0x4810}, "Eon's Elite Boomer"},
|
||||||
|
{{23, 0x0000}, "Wrecking Ball"},
|
||||||
|
{{23, 0x1801}, "Series 2 Wrecking Ball"},
|
||||||
|
{{24, 0x0000}, "Camo"},
|
||||||
|
{{24, 0x2805}, "Thorn Horn Camo"},
|
||||||
|
{{25, 0x0000}, "Zook"},
|
||||||
|
{{25, 0x1801}, "Series 2 Zook"},
|
||||||
|
{{25, 0x4810}, "Eon's Elite Zook"},
|
||||||
|
{{26, 0x0000}, "Stealth Elf"},
|
||||||
|
{{26, 0x1801}, "Series 2 Stealth Elf"},
|
||||||
|
{{26, 0x2C02}, "Dark Stealth Elf"},
|
||||||
|
{{26, 0x1C03}, "Legendary Stealth Elf"},
|
||||||
|
{{26, 0x2805}, "Ninja Stealth Elf"},
|
||||||
|
{{26, 0x3810}, "Eon's Elite Stealth Elf"},
|
||||||
|
{{27, 0x0000}, "Stump Smash"},
|
||||||
|
{{27, 0x1801}, "Series 2 Stump Smash"},
|
||||||
|
{{28, 0x0000}, "Dark Spyro"},
|
||||||
|
{{29, 0x0000}, "Hex"},
|
||||||
|
{{29, 0x1801}, "Series 2 Hex"},
|
||||||
|
{{29, 0x1206}, "LightCore Hex"},
|
||||||
|
{{30, 0x0000}, "Chop Chop"},
|
||||||
|
{{30, 0x1801}, "Series 2 Chop Chop"},
|
||||||
|
{{30, 0x2805}, "Twin Blade Chop Chop"},
|
||||||
|
{{30, 0x3810}, "Eon's Elite Chop Chop"},
|
||||||
|
{{31, 0x0000}, "Ghost Roaster"},
|
||||||
|
{{31, 0x4810}, "Eon's Elite Ghost Roaster"},
|
||||||
|
{{32, 0x0000}, "Cynder"},
|
||||||
|
{{32, 0x1801}, "Series 2 Cynder"},
|
||||||
|
{{32, 0x2805}, "Phantom Cynder"},
|
||||||
|
{{100, 0x0000}, "Jet Vac"},
|
||||||
|
{{100, 0x1403}, "Legendary Jet Vac"},
|
||||||
|
{{100, 0x2805}, "Turbo Jet Vac"},
|
||||||
|
{{100, 0x3805}, "Full Blast Jet Vac"},
|
||||||
|
{{100, 0x1206}, "LightCore Jet Vac"},
|
||||||
|
{{101, 0x0000}, "Swarm"},
|
||||||
|
{{102, 0x0000}, "Crusher"},
|
||||||
|
{{102, 0x1602}, "Granite Crusher"},
|
||||||
|
{{103, 0x0000}, "Flashwing"},
|
||||||
|
{{103, 0x1402}, "Jade Flash Wing"},
|
||||||
|
{{103, 0x2206}, "LightCore Flashwing"},
|
||||||
|
{{104, 0x0000}, "Hot Head"},
|
||||||
|
{{105, 0x0000}, "Hot Dog"},
|
||||||
|
{{105, 0x1402}, "Molten Hot Dog"},
|
||||||
|
{{105, 0x2805}, "Fire Bone Hot Dog"},
|
||||||
|
{{106, 0x0000}, "Chill"},
|
||||||
|
{{106, 0x1603}, "Legendary Chill"},
|
||||||
|
{{106, 0x2805}, "Blizzard Chill"},
|
||||||
|
{{106, 0x1206}, "LightCore Chill"},
|
||||||
|
{{107, 0x0000}, "Thumpback"},
|
||||||
|
{{108, 0x0000}, "Pop Fizz"},
|
||||||
|
{{108, 0x1402}, "Punch Pop Fizz"},
|
||||||
|
{{108, 0x3C02}, "Love Potion Pop Fizz"},
|
||||||
|
{{108, 0x2805}, "Super Gulp Pop Fizz"},
|
||||||
|
{{108, 0x3805}, "Fizzy Frenzy Pop Fizz"},
|
||||||
|
{{108, 0x1206}, "LightCore Pop Fizz"},
|
||||||
|
{{109, 0x0000}, "Ninjini"},
|
||||||
|
{{109, 0x1602}, "Scarlet Ninjini"},
|
||||||
|
{{110, 0x0000}, "Bouncer"},
|
||||||
|
{{110, 0x1603}, "Legendary Bouncer"},
|
||||||
|
{{111, 0x0000}, "Sprocket"},
|
||||||
|
{{111, 0x2805}, "Heavy Duty Sprocket"},
|
||||||
|
{{112, 0x0000}, "Tree Rex"},
|
||||||
|
{{112, 0x1602}, "Gnarly Tree Rex"},
|
||||||
|
{{113, 0x0000}, "Shroomboom"},
|
||||||
|
{{113, 0x3805}, "Sure Shot Shroomboom"},
|
||||||
|
{{113, 0x1206}, "LightCore Shroomboom"},
|
||||||
|
{{114, 0x0000}, "Eye Brawl"},
|
||||||
|
{{115, 0x0000}, "Fright Rider"},
|
||||||
|
{{200, 0x0000}, "Anvil Rain"},
|
||||||
|
{{201, 0x0000}, "Hidden Treasure"},
|
||||||
|
{{201, 0x2000}, "Platinum Hidden Treasure"},
|
||||||
|
{{202, 0x0000}, "Healing Elixir"},
|
||||||
|
{{203, 0x0000}, "Ghost Pirate Swords"},
|
||||||
|
{{204, 0x0000}, "Time Twist Hourglass"},
|
||||||
|
{{205, 0x0000}, "Sky Iron Shield"},
|
||||||
|
{{206, 0x0000}, "Winged Boots"},
|
||||||
|
{{207, 0x0000}, "Sparx the Dragonfly"},
|
||||||
|
{{208, 0x0000}, "Dragonfire Cannon"},
|
||||||
|
{{208, 0x1602}, "Golden Dragonfire Cannon"},
|
||||||
|
{{209, 0x0000}, "Scorpion Striker"},
|
||||||
|
{{210, 0x3002}, "Biter's Bane"},
|
||||||
|
{{210, 0x3008}, "Sorcerous Skull"},
|
||||||
|
{{210, 0x300B}, "Axe of Illusion"},
|
||||||
|
{{210, 0x300E}, "Arcane Hourglass"},
|
||||||
|
{{210, 0x3012}, "Spell Slapper"},
|
||||||
|
{{210, 0x3014}, "Rune Rocket"},
|
||||||
|
{{211, 0x3001}, "Tidal Tiki"},
|
||||||
|
{{211, 0x3002}, "Wet Walter"},
|
||||||
|
{{211, 0x3006}, "Flood Flask"},
|
||||||
|
{{211, 0x3406}, "Legendary Flood Flask"},
|
||||||
|
{{211, 0x3007}, "Soaking Staff"},
|
||||||
|
{{211, 0x300B}, "Aqua Axe"},
|
||||||
|
{{211, 0x3016}, "Frost Helm"},
|
||||||
|
{{212, 0x3003}, "Breezy Bird"},
|
||||||
|
{{212, 0x3006}, "Drafty Decanter"},
|
||||||
|
{{212, 0x300D}, "Tempest Timer"},
|
||||||
|
{{212, 0x3010}, "Cloudy Cobra"},
|
||||||
|
{{212, 0x3011}, "Storm Warning"},
|
||||||
|
{{212, 0x3018}, "Cyclone Saber"},
|
||||||
|
{{213, 0x3004}, "Spirit Sphere"},
|
||||||
|
{{213, 0x3404}, "Legendary Spirit Sphere"},
|
||||||
|
{{213, 0x3008}, "Spectral Skull"},
|
||||||
|
{{213, 0x3408}, "Legendary Spectral Skull"},
|
||||||
|
{{213, 0x300B}, "Haunted Hatchet"},
|
||||||
|
{{213, 0x300C}, "Grim Gripper"},
|
||||||
|
{{213, 0x3010}, "Spooky Snake"},
|
||||||
|
{{213, 0x3017}, "Dream Piercer"},
|
||||||
|
{{214, 0x3000}, "Tech Totem"},
|
||||||
|
{{214, 0x3007}, "Automatic Angel"},
|
||||||
|
{{214, 0x3009}, "Factory Flower"},
|
||||||
|
{{214, 0x300C}, "Grabbing Gadget"},
|
||||||
|
{{214, 0x3016}, "Makers Mana"},
|
||||||
|
{{214, 0x301A}, "Topsy Techy"},
|
||||||
|
{{215, 0x3005}, "Eternal Flame"},
|
||||||
|
{{215, 0x3009}, "Fire Flower"},
|
||||||
|
{{215, 0x3011}, "Scorching Stopper"},
|
||||||
|
{{215, 0x3012}, "Searing Spinner"},
|
||||||
|
{{215, 0x3017}, "Spark Spear"},
|
||||||
|
{{215, 0x301B}, "Blazing Belch"},
|
||||||
|
{{216, 0x3000}, "Banded Boulder"},
|
||||||
|
{{216, 0x3003}, "Rock Hawk"},
|
||||||
|
{{216, 0x300A}, "Slag Hammer"},
|
||||||
|
{{216, 0x300E}, "Dust Of Time"},
|
||||||
|
{{216, 0x3013}, "Spinning Sandstorm"},
|
||||||
|
{{216, 0x301A}, "Rubble Trouble"},
|
||||||
|
{{217, 0x3003}, "Oak Eagle"},
|
||||||
|
{{217, 0x3005}, "Emerald Energy"},
|
||||||
|
{{217, 0x300A}, "Weed Whacker"},
|
||||||
|
{{217, 0x3010}, "Seed Serpent"},
|
||||||
|
{{217, 0x3018}, "Jade Blade"},
|
||||||
|
{{217, 0x301B}, "Shrub Shrieker"},
|
||||||
|
{{218, 0x3000}, "Dark Dagger"},
|
||||||
|
{{218, 0x3014}, "Shadow Spider"},
|
||||||
|
{{218, 0x301A}, "Ghastly Grimace"},
|
||||||
|
{{219, 0x3000}, "Shining Ship"},
|
||||||
|
{{219, 0x300F}, "Heavenly Hawk"},
|
||||||
|
{{219, 0x301B}, "Beam Scream"},
|
||||||
|
{{220, 0x301E}, "Kaos Trap"},
|
||||||
|
{{220, 0x351F}, "Ultimate Kaos Trap"},
|
||||||
|
{{230, 0x0000}, "Hand of Fate"},
|
||||||
|
{{230, 0x3403}, "Legendary Hand of Fate"},
|
||||||
|
{{231, 0x0000}, "Piggy Bank"},
|
||||||
|
{{232, 0x0000}, "Rocket Ram"},
|
||||||
|
{{233, 0x0000}, "Tiki Speaky"},
|
||||||
|
{{300, 0x0000}, "Dragon’s Peak"},
|
||||||
|
{{301, 0x0000}, "Empire of Ice"},
|
||||||
|
{{302, 0x0000}, "Pirate Seas"},
|
||||||
|
{{303, 0x0000}, "Darklight Crypt"},
|
||||||
|
{{304, 0x0000}, "Volcanic Vault"},
|
||||||
|
{{305, 0x0000}, "Mirror of Mystery"},
|
||||||
|
{{306, 0x0000}, "Nightmare Express"},
|
||||||
|
{{307, 0x0000}, "Sunscraper Spire"},
|
||||||
|
{{308, 0x0000}, "Midnight Museum"},
|
||||||
|
{{404, 0x0000}, "Legendary Bash"},
|
||||||
|
{{416, 0x0000}, "Legendary Spyro"},
|
||||||
|
{{419, 0x0000}, "Legendary Trigger Happy"},
|
||||||
|
{{430, 0x0000}, "Legendary Chop Chop"},
|
||||||
|
{{450, 0x0000}, "Gusto"},
|
||||||
|
{{451, 0x0000}, "Thunderbolt"},
|
||||||
|
{{452, 0x0000}, "Fling Kong"},
|
||||||
|
{{453, 0x0000}, "Blades"},
|
||||||
|
{{453, 0x3403}, "Legendary Blades"},
|
||||||
|
{{454, 0x0000}, "Wallop"},
|
||||||
|
{{455, 0x0000}, "Head Rush"},
|
||||||
|
{{455, 0x3402}, "Nitro Head Rush"},
|
||||||
|
{{456, 0x0000}, "Fist Bump"},
|
||||||
|
{{457, 0x0000}, "Rocky Roll"},
|
||||||
|
{{458, 0x0000}, "Wildfire"},
|
||||||
|
{{458, 0x3402}, "Dark Wildfire"},
|
||||||
|
{{459, 0x0000}, "Ka Boom"},
|
||||||
|
{{460, 0x0000}, "Trail Blazer"},
|
||||||
|
{{461, 0x0000}, "Torch"},
|
||||||
|
{{462, 0x0000}, "Snap Shot"},
|
||||||
|
{{462, 0x3402}, "Dark Snap Shot"},
|
||||||
|
{{463, 0x0000}, "Lob Star"},
|
||||||
|
{{463, 0x3402}, "Winterfest Lob-Star"},
|
||||||
|
{{464, 0x0000}, "Flip Wreck"},
|
||||||
|
{{465, 0x0000}, "Echo"},
|
||||||
|
{{466, 0x0000}, "Blastermind"},
|
||||||
|
{{467, 0x0000}, "Enigma"},
|
||||||
|
{{468, 0x0000}, "Deja Vu"},
|
||||||
|
{{468, 0x3403}, "Legendary Deja Vu"},
|
||||||
|
{{469, 0x0000}, "Cobra Candabra"},
|
||||||
|
{{469, 0x3402}, "King Cobra Cadabra"},
|
||||||
|
{{470, 0x0000}, "Jawbreaker"},
|
||||||
|
{{470, 0x3403}, "Legendary Jawbreaker"},
|
||||||
|
{{471, 0x0000}, "Gearshift"},
|
||||||
|
{{472, 0x0000}, "Chopper"},
|
||||||
|
{{473, 0x0000}, "Tread Head"},
|
||||||
|
{{474, 0x0000}, "Bushwack"},
|
||||||
|
{{474, 0x3403}, "Legendary Bushwack"},
|
||||||
|
{{475, 0x0000}, "Tuff Luck"},
|
||||||
|
{{476, 0x0000}, "Food Fight"},
|
||||||
|
{{476, 0x3402}, "Dark Food Fight"},
|
||||||
|
{{477, 0x0000}, "High Five"},
|
||||||
|
{{478, 0x0000}, "Krypt King"},
|
||||||
|
{{478, 0x3402}, "Nitro Krypt King"},
|
||||||
|
{{479, 0x0000}, "Short Cut"},
|
||||||
|
{{480, 0x0000}, "Bat Spin"},
|
||||||
|
{{481, 0x0000}, "Funny Bone"},
|
||||||
|
{{482, 0x0000}, "Knight Light"},
|
||||||
|
{{483, 0x0000}, "Spotlight"},
|
||||||
|
{{484, 0x0000}, "Knight Mare"},
|
||||||
|
{{485, 0x0000}, "Blackout"},
|
||||||
|
{{502, 0x0000}, "Bop"},
|
||||||
|
{{505, 0x0000}, "Terrabite"},
|
||||||
|
{{506, 0x0000}, "Breeze"},
|
||||||
|
{{508, 0x0000}, "Pet Vac"},
|
||||||
|
{{508, 0x3402}, "Power Punch Pet Vac"},
|
||||||
|
{{507, 0x0000}, "Weeruptor"},
|
||||||
|
{{507, 0x3402}, "Eggcellent Weeruptor"},
|
||||||
|
{{509, 0x0000}, "Small Fry"},
|
||||||
|
{{510, 0x0000}, "Drobit"},
|
||||||
|
{{519, 0x0000}, "Trigger Snappy"},
|
||||||
|
{{526, 0x0000}, "Whisper Elf"},
|
||||||
|
{{540, 0x0000}, "Barkley"},
|
||||||
|
{{540, 0x3402}, "Gnarly Barkley"},
|
||||||
|
{{541, 0x0000}, "Thumpling"},
|
||||||
|
{{514, 0x0000}, "Gill Runt"},
|
||||||
|
{{542, 0x0000}, "Mini-Jini"},
|
||||||
|
{{503, 0x0000}, "Spry"},
|
||||||
|
{{504, 0x0000}, "Hijinx"},
|
||||||
|
{{543, 0x0000}, "Eye Small"},
|
||||||
|
{{601, 0x0000}, "King Pen"},
|
||||||
|
{{602, 0x0000}, "Tri-Tip"},
|
||||||
|
{{603, 0x0000}, "Chopscotch"},
|
||||||
|
{{604, 0x0000}, "Boom Bloom"},
|
||||||
|
{{605, 0x0000}, "Pit Boss"},
|
||||||
|
{{606, 0x0000}, "Barbella"},
|
||||||
|
{{607, 0x0000}, "Air Strike"},
|
||||||
|
{{608, 0x0000}, "Ember"},
|
||||||
|
{{609, 0x0000}, "Ambush"},
|
||||||
|
{{610, 0x0000}, "Dr. Krankcase"},
|
||||||
|
{{611, 0x0000}, "Hood Sickle"},
|
||||||
|
{{612, 0x0000}, "Tae Kwon Crow"},
|
||||||
|
{{613, 0x0000}, "Golden Queen"},
|
||||||
|
{{614, 0x0000}, "Wolfgang"},
|
||||||
|
{{615, 0x0000}, "Pain-Yatta"},
|
||||||
|
{{616, 0x0000}, "Mysticat"},
|
||||||
|
{{617, 0x0000}, "Starcast"},
|
||||||
|
{{618, 0x0000}, "Buckshot"},
|
||||||
|
{{619, 0x0000}, "Aurora"},
|
||||||
|
{{620, 0x0000}, "Flare Wolf"},
|
||||||
|
{{621, 0x0000}, "Chompy Mage"},
|
||||||
|
{{622, 0x0000}, "Bad Juju"},
|
||||||
|
{{623, 0x0000}, "Grave Clobber"},
|
||||||
|
{{624, 0x0000}, "Blaster-Tron"},
|
||||||
|
{{625, 0x0000}, "Ro-Bow"},
|
||||||
|
{{626, 0x0000}, "Chain Reaction"},
|
||||||
|
{{627, 0x0000}, "Kaos"},
|
||||||
|
{{628, 0x0000}, "Wild Storm"},
|
||||||
|
{{629, 0x0000}, "Tidepool"},
|
||||||
|
{{630, 0x0000}, "Crash Bandicoot"},
|
||||||
|
{{631, 0x0000}, "Dr. Neo Cortex"},
|
||||||
|
{{1000, 0x0000}, "Boom Jet (Bottom)"},
|
||||||
|
{{1001, 0x0000}, "Free Ranger (Bottom)"},
|
||||||
|
{{1001, 0x2403}, "Legendary Free Ranger (Bottom)"},
|
||||||
|
{{1002, 0x0000}, "Rubble Rouser (Bottom)"},
|
||||||
|
{{1003, 0x0000}, "Doom Stone (Bottom)"},
|
||||||
|
{{1004, 0x0000}, "Blast Zone (Bottom)"},
|
||||||
|
{{1004, 0x2402}, "Dark Blast Zone (Bottom)"},
|
||||||
|
{{1005, 0x0000}, "Fire Kraken (Bottom)"},
|
||||||
|
{{1005, 0x2402}, "Jade Fire Kraken (Bottom)"},
|
||||||
|
{{1006, 0x0000}, "Stink Bomb (Bottom)"},
|
||||||
|
{{1007, 0x0000}, "Grilla Drilla (Bottom)"},
|
||||||
|
{{1008, 0x0000}, "Hoot Loop (Bottom)"},
|
||||||
|
{{1008, 0x2402}, "Enchanted Hoot Loop (Bottom)"},
|
||||||
|
{{1009, 0x0000}, "Trap Shadow (Bottom)"},
|
||||||
|
{{1010, 0x0000}, "Magna Charge (Bottom)"},
|
||||||
|
{{1010, 0x2402}, "Nitro Magna Charge (Bottom)"},
|
||||||
|
{{1011, 0x0000}, "Spy Rise (Bottom)"},
|
||||||
|
{{1012, 0x0000}, "Night Shift (Bottom)"},
|
||||||
|
{{1012, 0x2403}, "Legendary Night Shift (Bottom)"},
|
||||||
|
{{1013, 0x0000}, "Rattle Shake (Bottom)"},
|
||||||
|
{{1013, 0x2402}, "Quick Draw Rattle Shake (Bottom)"},
|
||||||
|
{{1014, 0x0000}, "Freeze Blade (Bottom)"},
|
||||||
|
{{1014, 0x2402}, "Nitro Freeze Blade (Bottom)"},
|
||||||
|
{{1015, 0x0000}, "Wash Buckler (Bottom)"},
|
||||||
|
{{1015, 0x2402}, "Dark Wash Buckler (Bottom)"},
|
||||||
|
{{2000, 0x0000}, "Boom Jet (Top)"},
|
||||||
|
{{2001, 0x0000}, "Free Ranger (Top)"},
|
||||||
|
{{2001, 0x2403}, "Legendary Free Ranger (Top)"},
|
||||||
|
{{2002, 0x0000}, "Rubble Rouser (Top)"},
|
||||||
|
{{2003, 0x0000}, "Doom Stone (Top)"},
|
||||||
|
{{2004, 0x0000}, "Blast Zone (Top)"},
|
||||||
|
{{2004, 0x2402}, "Dark Blast Zone (Top)"},
|
||||||
|
{{2005, 0x0000}, "Fire Kraken (Top)"},
|
||||||
|
{{2005, 0x2402}, "Jade Fire Kraken (Top)"},
|
||||||
|
{{2006, 0x0000}, "Stink Bomb (Top)"},
|
||||||
|
{{2007, 0x0000}, "Grilla Drilla (Top)"},
|
||||||
|
{{2008, 0x0000}, "Hoot Loop (Top)"},
|
||||||
|
{{2008, 0x2402}, "Enchanted Hoot Loop (Top)"},
|
||||||
|
{{2009, 0x0000}, "Trap Shadow (Top)"},
|
||||||
|
{{2010, 0x0000}, "Magna Charge (Top)"},
|
||||||
|
{{2010, 0x2402}, "Nitro Magna Charge (Top)"},
|
||||||
|
{{2011, 0x0000}, "Spy Rise (Top)"},
|
||||||
|
{{2012, 0x0000}, "Night Shift (Top)"},
|
||||||
|
{{2012, 0x2403}, "Legendary Night Shift (Top)"},
|
||||||
|
{{2013, 0x0000}, "Rattle Shake (Top)"},
|
||||||
|
{{2013, 0x2402}, "Quick Draw Rattle Shake (Top)"},
|
||||||
|
{{2014, 0x0000}, "Freeze Blade (Top)"},
|
||||||
|
{{2014, 0x2402}, "Nitro Freeze Blade (Top)"},
|
||||||
|
{{2015, 0x0000}, "Wash Buckler (Top)"},
|
||||||
|
{{2015, 0x2402}, "Dark Wash Buckler (Top)"},
|
||||||
|
{{3000, 0x0000}, "Scratch"},
|
||||||
|
{{3001, 0x0000}, "Pop Thorn"},
|
||||||
|
{{3002, 0x0000}, "Slobber Tooth"},
|
||||||
|
{{3002, 0x2402}, "Dark Slobber Tooth"},
|
||||||
|
{{3003, 0x0000}, "Scorp"},
|
||||||
|
{{3004, 0x0000}, "Fryno"},
|
||||||
|
{{3004, 0x3805}, "Hog Wild Fryno"},
|
||||||
|
{{3005, 0x0000}, "Smolderdash"},
|
||||||
|
{{3005, 0x2206}, "LightCore Smolderdash"},
|
||||||
|
{{3006, 0x0000}, "Bumble Blast"},
|
||||||
|
{{3006, 0x2402}, "Jolly Bumble Blast"},
|
||||||
|
{{3006, 0x2206}, "LightCore Bumble Blast"},
|
||||||
|
{{3007, 0x0000}, "Zoo Lou"},
|
||||||
|
{{3007, 0x2403}, "Legendary Zoo Lou"},
|
||||||
|
{{3008, 0x0000}, "Dune Bug"},
|
||||||
|
{{3009, 0x0000}, "Star Strike"},
|
||||||
|
{{3009, 0x2602}, "Enchanted Star Strike"},
|
||||||
|
{{3009, 0x2206}, "LightCore Star Strike"},
|
||||||
|
{{3010, 0x0000}, "Countdown"},
|
||||||
|
{{3010, 0x2402}, "Kickoff Countdown"},
|
||||||
|
{{3010, 0x2206}, "LightCore Countdown"},
|
||||||
|
{{3011, 0x0000}, "Wind Up"},
|
||||||
|
{{3012, 0x0000}, "Roller Brawl"},
|
||||||
|
{{3013, 0x0000}, "Grim Creeper"},
|
||||||
|
{{3013, 0x2603}, "Legendary Grim Creeper"},
|
||||||
|
{{3013, 0x2206}, "LightCore Grim Creeper"},
|
||||||
|
{{3014, 0x0000}, "Rip Tide"},
|
||||||
|
{{3015, 0x0000}, "Punk Shock"},
|
||||||
|
{{3200, 0x0000}, "Battle Hammer"},
|
||||||
|
{{3201, 0x0000}, "Sky Diamond"},
|
||||||
|
{{3202, 0x0000}, "Platinum Sheep"},
|
||||||
|
{{3203, 0x0000}, "Groove Machine"},
|
||||||
|
{{3204, 0x0000}, "UFO Hat"},
|
||||||
|
{{3300, 0x0000}, "Sheep Wreck Island"},
|
||||||
|
{{3301, 0x0000}, "Tower of Time"},
|
||||||
|
{{3302, 0x0000}, "Fiery Forge"},
|
||||||
|
{{3303, 0x0000}, "Arkeyan Crossbow"},
|
||||||
|
{{3220, 0x0000}, "Jet Stream"},
|
||||||
|
{{3221, 0x0000}, "Tomb Buggy"},
|
||||||
|
{{3222, 0x0000}, "Reef Ripper"},
|
||||||
|
{{3223, 0x0000}, "Burn Cycle"},
|
||||||
|
{{3224, 0x0000}, "Hot Streak"},
|
||||||
|
{{3224, 0x4402}, "Dark Hot Streak"},
|
||||||
|
{{3224, 0x4004}, "E3 Hot Streak"},
|
||||||
|
{{3224, 0x441E}, "Golden Hot Streak"},
|
||||||
|
{{3225, 0x0000}, "Shark Tank"},
|
||||||
|
{{3226, 0x0000}, "Thump Truck"},
|
||||||
|
{{3227, 0x0000}, "Crypt Crusher"},
|
||||||
|
{{3228, 0x0000}, "Stealth Stinger"},
|
||||||
|
{{3228, 0x4402}, "Nitro Stealth Stinger"},
|
||||||
|
{{3231, 0x0000}, "Dive Bomber"},
|
||||||
|
{{3231, 0x4402}, "Spring Ahead Dive Bomber"},
|
||||||
|
{{3232, 0x0000}, "Sky Slicer"},
|
||||||
|
//{{3233, 0x0000}, "Clown Cruiser (Nintendo Only)"},
|
||||||
|
//{{3233, 0x4402}, "Dark Clown Cruiser (Nintendo Only)"},
|
||||||
|
{{3234, 0x0000}, "Gold Rusher"},
|
||||||
|
{{3234, 0x4402}, "Power Blue Gold Rusher"},
|
||||||
|
{{3235, 0x0000}, "Shield Striker"},
|
||||||
|
{{3236, 0x0000}, "Sun Runner"},
|
||||||
|
{{3236, 0x4403}, "Legendary Sun Runner"},
|
||||||
|
{{3237, 0x0000}, "Sea Shadow"},
|
||||||
|
{{3237, 0x4402}, "Dark Sea Shadow"},
|
||||||
|
{{3238, 0x0000}, "Splatter Splasher"},
|
||||||
|
{{3238, 0x4402}, "Power Blue Splatter Splasher"},
|
||||||
|
{{3239, 0x0000}, "Soda Skimmer"},
|
||||||
|
{{3239, 0x4402}, "Nitro Soda Skimmer"},
|
||||||
|
//{{3240, 0x0000}, "Barrel Blaster (Nintendo Only)"},
|
||||||
|
//{{3240, 0x4402}, "Dark Barrel Blaster (Nintendo Only)"},
|
||||||
|
{{3241, 0x0000}, "Buzz Wing"},
|
||||||
|
{{3400, 0x0000}, "Fiesta"},
|
||||||
|
{{3400, 0x4515}, "Frightful Fiesta"},
|
||||||
|
{{3401, 0x0000}, "High Volt"},
|
||||||
|
{{3402, 0x0000}, "Splat"},
|
||||||
|
{{3402, 0x4502}, "Power Blue Splat"},
|
||||||
|
{{3406, 0x0000}, "Stormblade"},
|
||||||
|
{{3411, 0x0000}, "Smash Hit"},
|
||||||
|
{{3411, 0x4502}, "Steel Plated Smash Hit"},
|
||||||
|
{{3412, 0x0000}, "Spitfire"},
|
||||||
|
{{3412, 0x4502}, "Dark Spitfire"},
|
||||||
|
{{3413, 0x0000}, "Hurricane Jet Vac"},
|
||||||
|
{{3413, 0x4503}, "Legendary Hurricane Jet Vac"},
|
||||||
|
{{3414, 0x0000}, "Double Dare Trigger Happy"},
|
||||||
|
{{3414, 0x4502}, "Power Blue Double Dare Trigger Happy"},
|
||||||
|
{{3415, 0x0000}, "Super Shot Stealth Elf"},
|
||||||
|
{{3415, 0x4502}, "Dark Super Shot Stealth Elf"},
|
||||||
|
{{3416, 0x0000}, "Shark Shooter Terrafin"},
|
||||||
|
{{3417, 0x0000}, "Bone Bash Roller Brawl"},
|
||||||
|
{{3417, 0x4503}, "Legendary Bone Bash Roller Brawl"},
|
||||||
|
{{3420, 0x0000}, "Big Bubble Pop Fizz"},
|
||||||
|
{{3420, 0x450E}, "Birthday Bash Big Bubble Pop Fizz"},
|
||||||
|
{{3421, 0x0000}, "Lava Lance Eruptor"},
|
||||||
|
{{3422, 0x0000}, "Deep Dive Gill Grunt"},
|
||||||
|
//{{3423, 0x0000}, "Turbo Charge Donkey Kong (Nintendo Only)"},
|
||||||
|
//{{3423, 0x4502}, "Dark Turbo Charge Donkey Kong (Nintendo Only)"},
|
||||||
|
//{{3424, 0x0000}, "Hammer Slam Bowser (Nintendo Only)"},
|
||||||
|
//{{3424, 0x4502}, "Dark Hammer Slam Bowser (Nintendo Only)"},
|
||||||
|
{{3425, 0x0000}, "Dive-Clops"},
|
||||||
|
{{3425, 0x450E}, "Missile-Tow Dive-Clops"},
|
||||||
|
{{3426, 0x0000}, "Astroblast"},
|
||||||
|
{{3426, 0x4503}, "Legendary Astroblast"},
|
||||||
|
{{3427, 0x0000}, "Nightfall"},
|
||||||
|
{{3428, 0x0000}, "Thrillipede"},
|
||||||
|
{{3428, 0x450D}, "Eggcited Thrillipede"},
|
||||||
|
{{3500, 0x0000}, "Sky Trophy"},
|
||||||
|
{{3501, 0x0000}, "Land Trophy"},
|
||||||
|
{{3502, 0x0000}, "Sea Trophy"},
|
||||||
|
{{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;
|
||||||
|
}
|
||||||
|
|
||||||
|
CreateSkylanderDialog::CreateSkylanderDialog(QWidget* parent) : QDialog(parent) {
|
||||||
|
setWindowTitle(tr("Skylander Creator"));
|
||||||
|
setObjectName("skylanders_creator");
|
||||||
|
setMinimumSize(QSize(500, 150));
|
||||||
|
|
||||||
|
QVBoxLayout* vbox_panel = new QVBoxLayout();
|
||||||
|
|
||||||
|
QComboBox* 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);
|
||||||
|
}
|
||||||
|
combo_skylist->addItem(tr("--Unknown--"), QVariant(0xFFFFFFFF));
|
||||||
|
combo_skylist->setEditable(true);
|
||||||
|
combo_skylist->setInsertPolicy(QComboBox::NoInsert);
|
||||||
|
|
||||||
|
QCompleter* co_compl = new QCompleter(filterlist, this);
|
||||||
|
co_compl->setCaseSensitivity(Qt::CaseInsensitive);
|
||||||
|
co_compl->setCompletionMode(QCompleter::PopupCompletion);
|
||||||
|
co_compl->setFilterMode(Qt::MatchContains);
|
||||||
|
combo_skylist->setCompleter(co_compl);
|
||||||
|
|
||||||
|
vbox_panel->addWidget(combo_skylist);
|
||||||
|
|
||||||
|
QFrame* line = new QFrame();
|
||||||
|
line->setFrameShape(QFrame::HLine);
|
||||||
|
line->setFrameShadow(QFrame::Sunken);
|
||||||
|
vbox_panel->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);
|
||||||
|
edit_id->setValidator(rxv);
|
||||||
|
edit_var->setValidator(rxv);
|
||||||
|
hbox_idvar->addWidget(label_id);
|
||||||
|
hbox_idvar->addWidget(edit_id);
|
||||||
|
hbox_idvar->addWidget(label_var);
|
||||||
|
hbox_idvar->addWidget(edit_var);
|
||||||
|
vbox_panel->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);
|
||||||
|
vbox_panel->addLayout(hbox_buttons);
|
||||||
|
|
||||||
|
setLayout(vbox_panel);
|
||||||
|
|
||||||
|
connect(combo_skylist, QOverload<int>::of(&QComboBox::currentIndexChanged), [=](int index) {
|
||||||
|
const u32 sky_info = combo_skylist->itemData(index).toUInt();
|
||||||
|
if (sky_info != 0xFFFFFFFF) {
|
||||||
|
const u16 sky_id = sky_info >> 16;
|
||||||
|
const u16 sky_var = sky_info & 0xFFFF;
|
||||||
|
|
||||||
|
edit_id->setText(QString::number(sky_id));
|
||||||
|
edit_var->setText(QString::number(sky_var));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
connect(btn_create, &QAbstractButton::clicked, this, [=, this]() {
|
||||||
|
bool ok_id = false, ok_var = false;
|
||||||
|
const u16 sky_id = edit_id->text().toUShort(&ok_id);
|
||||||
|
if (!ok_id) {
|
||||||
|
QMessageBox::warning(this, tr("Error converting value"), tr("ID entered is invalid!"),
|
||||||
|
QMessageBox::Ok);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const u16 sky_var = edit_var->text().toUShort(&ok_var);
|
||||||
|
if (!ok_var) {
|
||||||
|
QMessageBox::warning(this, tr("Error converting value"),
|
||||||
|
tr("Variant entered is invalid!"), QMessageBox::Ok);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
QString predef_name = 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");
|
||||||
|
} else {
|
||||||
|
predef_name +=
|
||||||
|
QString(QString::fromStdString("Unknown(%1 %2).sky")).arg(sky_id).arg(sky_var);
|
||||||
|
}
|
||||||
|
|
||||||
|
file_path = QFileDialog::getSaveFileName(this, tr("Create Skylander File"), predef_name,
|
||||||
|
tr("Skylander Object (*.sky);;All Files (*)"));
|
||||||
|
if (file_path.isEmpty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
FileUtil::IOFile sky_file(file_path.toStdString(), "wb");
|
||||||
|
if (!sky_file) {
|
||||||
|
QMessageBox::warning(this, tr("Failed to create skylander file!"),
|
||||||
|
tr("Failed to create skylander file:\n%1").arg(file_path),
|
||||||
|
QMessageBox::Ok);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::array<u8, 0x40 * 0x10> buf{};
|
||||||
|
const auto data = buf.data();
|
||||||
|
// Set the block permissions
|
||||||
|
u32 first_block = 0x690F0F0F;
|
||||||
|
u32 other_blocks = 0x69080F7F;
|
||||||
|
memcpy(&data[0x36], &first_block, sizeof(first_block));
|
||||||
|
for (size_t index = 1; index < 0x10; index++) {
|
||||||
|
memcpy(&data[(index * 0x40) + 0x36], &other_blocks, sizeof(other_blocks));
|
||||||
|
}
|
||||||
|
|
||||||
|
std::random_device rd;
|
||||||
|
std::mt19937 mt(rd());
|
||||||
|
std::uniform_int_distribution<int> dist(0, 255);
|
||||||
|
|
||||||
|
data[0] = dist(mt);
|
||||||
|
data[1] = dist(mt);
|
||||||
|
data[2] = dist(mt);
|
||||||
|
data[3] = dist(mt);
|
||||||
|
|
||||||
|
// The BCC (Block Check Character)
|
||||||
|
data[4] = data[0] ^ data[1] ^ data[2] ^ data[3];
|
||||||
|
|
||||||
|
// ATQA
|
||||||
|
data[5] = 0x81;
|
||||||
|
data[6] = 0x01;
|
||||||
|
|
||||||
|
// SAK
|
||||||
|
data[7] = 0x0F;
|
||||||
|
|
||||||
|
// Set the skylander info
|
||||||
|
memcpy(&data[0x10], &sky_id, sizeof(sky_id));
|
||||||
|
memcpy(&data[0x1C], &sky_var, sizeof(sky_var));
|
||||||
|
// Set checksum
|
||||||
|
u16 crc = skylander_crc16(0xFFFF, data, 0x1E);
|
||||||
|
memcpy(&data[0x1E], &crc, 2);
|
||||||
|
|
||||||
|
sky_file.Seek(0, SEEK_SET);
|
||||||
|
if (sky_file.WriteBytes(buf.data(), buf.size()) != buf.size()) {
|
||||||
|
throw std::runtime_error("Could not write to file " + file_path.toStdString());
|
||||||
|
}
|
||||||
|
sky_file.Close();
|
||||||
|
|
||||||
|
last_skylander_path = QFileInfo(file_path).absolutePath() + QString::fromStdString("/");
|
||||||
|
accept();
|
||||||
|
});
|
||||||
|
|
||||||
|
connect(btn_cancel, &QAbstractButton::clicked, this, &QDialog::reject);
|
||||||
|
|
||||||
|
connect(co_compl, QOverload<const QString&>::of(&QCompleter::activated),
|
||||||
|
[=](const QString& text) {
|
||||||
|
combo_skylist->setCurrentText(text);
|
||||||
|
combo_skylist->setCurrentIndex(combo_skylist->findText(text));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
QString CreateSkylanderDialog::get_file_path() const {
|
||||||
|
return file_path;
|
||||||
|
}
|
||||||
|
|
||||||
|
SkylanderPortalWindow::SkylanderPortalWindow(QWidget* parent) : QDialog(parent) {
|
||||||
|
setWindowTitle(tr("Skylanders Manager"));
|
||||||
|
setObjectName("skylanders_manager");
|
||||||
|
setAttribute(Qt::WA_DeleteOnClose);
|
||||||
|
setMinimumSize(QSize(700, 200));
|
||||||
|
|
||||||
|
QVBoxLayout* vbox_panel = new QVBoxLayout();
|
||||||
|
|
||||||
|
auto add_line = [](QVBoxLayout* vbox) {
|
||||||
|
QFrame* line = new QFrame();
|
||||||
|
line->setFrameShape(QFrame::HLine);
|
||||||
|
line->setFrameShadow(QFrame::Sunken);
|
||||||
|
vbox->addWidget(line);
|
||||||
|
};
|
||||||
|
|
||||||
|
QGroupBox* group_skylanders = new QGroupBox(tr("Active Portal Skylanders:"));
|
||||||
|
QVBoxLayout* vbox_group = new QVBoxLayout();
|
||||||
|
|
||||||
|
for (auto i = 0; i < UI_SKY_NUM; i++) {
|
||||||
|
if (i != 0) {
|
||||||
|
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);
|
||||||
|
|
||||||
|
QPushButton* clear_btn = new QPushButton(tr("Clear"));
|
||||||
|
QPushButton* create_btn = new QPushButton(tr("Create"));
|
||||||
|
QPushButton* load_btn = new QPushButton(tr("Load"));
|
||||||
|
|
||||||
|
connect(clear_btn, &QAbstractButton::clicked, this, [this, i]() { clear_skylander(i); });
|
||||||
|
connect(create_btn, &QAbstractButton::clicked, this, [this, i]() { create_skylander(i); });
|
||||||
|
connect(load_btn, &QAbstractButton::clicked, this, [this, i]() { load_skylander(i); });
|
||||||
|
|
||||||
|
hbox_skylander->addWidget(label_skyname);
|
||||||
|
hbox_skylander->addWidget(edit_skylanders[i]);
|
||||||
|
hbox_skylander->addWidget(clear_btn);
|
||||||
|
hbox_skylander->addWidget(create_btn);
|
||||||
|
hbox_skylander->addWidget(load_btn);
|
||||||
|
|
||||||
|
vbox_group->addLayout(hbox_skylander);
|
||||||
|
}
|
||||||
|
|
||||||
|
group_skylanders->setLayout(vbox_group);
|
||||||
|
vbox_panel->addWidget(group_skylanders);
|
||||||
|
setLayout(vbox_panel);
|
||||||
|
|
||||||
|
update_edits();
|
||||||
|
}
|
||||||
|
|
||||||
|
SkylanderPortalWindow::~SkylanderPortalWindow() {
|
||||||
|
inst = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
SkylanderPortalWindow* SkylanderPortalWindow::get_dlg(QWidget* parent) {
|
||||||
|
if (inst == nullptr)
|
||||||
|
inst = new SkylanderPortalWindow(parent);
|
||||||
|
|
||||||
|
return inst;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SkylanderPortalWindow::clear_skylander(u8 slot) {
|
||||||
|
if (auto slot_infos = sky_slots[slot]) {
|
||||||
|
auto [cur_slot, id, var] = slot_infos.value();
|
||||||
|
Service::IR::g_skyportal.RemoveSkylander(cur_slot);
|
||||||
|
sky_slots[slot] = {};
|
||||||
|
update_edits();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SkylanderPortalWindow::create_skylander(u8 slot) {
|
||||||
|
CreateSkylanderDialog create_dlg(this);
|
||||||
|
if (create_dlg.exec() == Accepted) {
|
||||||
|
LOG_INFO(Service_IR, "{}", create_dlg.get_file_path().toStdString());
|
||||||
|
load_skylander_path(slot, create_dlg.get_file_path());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SkylanderPortalWindow::load_skylander(u8 slot) {
|
||||||
|
const QString file_path = QFileDialog::getOpenFileName(
|
||||||
|
this, tr("Select Skylander File"), last_skylander_path, tr("Skylander (*.sky *.bin *.dmp *.dump);;All Files (*)"));
|
||||||
|
if (file_path.isEmpty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
last_skylander_path = QFileInfo(file_path).absolutePath() + QString::fromStdString("/");
|
||||||
|
|
||||||
|
load_skylander_path(slot, file_path);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SkylanderPortalWindow::load_skylander_path(u8 slot, const QString& path) {
|
||||||
|
FileUtil::IOFile sky_file(path.toStdString(), "rb");
|
||||||
|
if (!sky_file.IsGood()) {
|
||||||
|
QMessageBox::warning(
|
||||||
|
this, tr("Failed to open the skylander file!"),
|
||||||
|
tr("Failed to open the skylander file(%1)!\nFile may already be in use on the portal.")
|
||||||
|
.arg(path),
|
||||||
|
QMessageBox::Ok);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::array<u8, 0x40 * 0x10> data;
|
||||||
|
sky_file.Seek(0, SEEK_SET);
|
||||||
|
if (sky_file.ReadBytes(data.data(), data.size()) != data.size()) {
|
||||||
|
QMessageBox::warning(
|
||||||
|
this, tr("Failed to read the skylander file!"),
|
||||||
|
tr("Failed to read the skylander file(%1)!\nFile was too small.").arg(path),
|
||||||
|
QMessageBox::Ok);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
clear_skylander(slot);
|
||||||
|
|
||||||
|
u16 sky_id = 0;
|
||||||
|
u16 sky_var = 0;
|
||||||
|
|
||||||
|
memcpy(&sky_id, &data[0x10], sizeof(sky_id));
|
||||||
|
memcpy(&sky_var, &data[0x1C], sizeof(sky_var));
|
||||||
|
|
||||||
|
u8 portal_slot = Service::IR::g_skyportal.LoadSkylander(data.data(), std::move(sky_file));
|
||||||
|
sky_slots[slot] = std::tuple(portal_slot, sky_id, sky_var);
|
||||||
|
|
||||||
|
update_edits();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SkylanderPortalWindow::update_edits() {
|
||||||
|
for (auto i = 0; i < UI_SKY_NUM; i++) {
|
||||||
|
QString display_string;
|
||||||
|
if (auto sd = 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));
|
||||||
|
if (found_sky != list_skylanders.end()) {
|
||||||
|
display_string = QString::fromStdString(found_sky->second);
|
||||||
|
} else {
|
||||||
|
display_string = QString(tr("Unknown (Id:%1 Var:%2)")).arg(sky_id).arg(sky_var);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
display_string = tr("None");
|
||||||
|
}
|
||||||
|
|
||||||
|
edit_skylanders[i]->setText(display_string);
|
||||||
|
}
|
||||||
|
}
|
52
src/citra_qt/infrared/skylanderportal/skylander_dialog.h
Normal file
52
src/citra_qt/infrared/skylanderportal/skylander_dialog.h
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
// Copyright 2024 Citra Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <optional>
|
||||||
|
|
||||||
|
#include <QDialog>
|
||||||
|
#include <QLineEdit>
|
||||||
|
|
||||||
|
constexpr auto UI_SKY_NUM = 8;
|
||||||
|
|
||||||
|
class CreateSkylanderDialog : public QDialog
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit CreateSkylanderDialog(QWidget* parent);
|
||||||
|
QString get_file_path() const;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
QString file_path;
|
||||||
|
};
|
||||||
|
|
||||||
|
class SkylanderPortalWindow : public QDialog
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit SkylanderPortalWindow(QWidget* parent);
|
||||||
|
~SkylanderPortalWindow();
|
||||||
|
static SkylanderPortalWindow* get_dlg(QWidget* parent);
|
||||||
|
|
||||||
|
SkylanderPortalWindow(SkylanderPortalWindow const&) = delete;
|
||||||
|
void operator=(SkylanderPortalWindow const&) = delete;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void clear_skylander(u8 slot);
|
||||||
|
void create_skylander(u8 slot);
|
||||||
|
void load_skylander(u8 slot);
|
||||||
|
void load_skylander_path(u8 slot, const QString& path);
|
||||||
|
|
||||||
|
void update_edits();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
QLineEdit* edit_skylanders[UI_SKY_NUM]{};
|
||||||
|
static std::optional<std::tuple<u8, u16, u16>> sky_slots[UI_SKY_NUM];
|
||||||
|
|
||||||
|
private:
|
||||||
|
static SkylanderPortalWindow* inst;
|
||||||
|
};
|
|
@ -54,6 +54,7 @@
|
||||||
#include "citra_qt/dumping/dumping_dialog.h"
|
#include "citra_qt/dumping/dumping_dialog.h"
|
||||||
#include "citra_qt/game_list.h"
|
#include "citra_qt/game_list.h"
|
||||||
#include "citra_qt/hotkeys.h"
|
#include "citra_qt/hotkeys.h"
|
||||||
|
#include "citra_qt/infrared/skylanderportal/skylander_dialog.h"
|
||||||
#include "citra_qt/loading_screen.h"
|
#include "citra_qt/loading_screen.h"
|
||||||
#include "citra_qt/main.h"
|
#include "citra_qt/main.h"
|
||||||
#include "citra_qt/movie/movie_play_dialog.h"
|
#include "citra_qt/movie/movie_play_dialog.h"
|
||||||
|
@ -938,6 +939,7 @@ void GMainWindow::ConnectMenuEvents() {
|
||||||
});
|
});
|
||||||
connect_menu(ui->action_Capture_Screenshot, &GMainWindow::OnCaptureScreenshot);
|
connect_menu(ui->action_Capture_Screenshot, &GMainWindow::OnCaptureScreenshot);
|
||||||
connect_menu(ui->action_Dump_Video, &GMainWindow::OnDumpVideo);
|
connect_menu(ui->action_Dump_Video, &GMainWindow::OnDumpVideo);
|
||||||
|
connect_menu(ui->action_Manage_Skylanders, &GMainWindow::ShowSkylanderPortal);
|
||||||
|
|
||||||
// Help
|
// Help
|
||||||
connect_menu(ui->action_Open_Citra_Folder, &GMainWindow::OnOpenCitraFolder);
|
connect_menu(ui->action_Open_Citra_Folder, &GMainWindow::OnOpenCitraFolder);
|
||||||
|
@ -2606,6 +2608,16 @@ void GMainWindow::OnStopVideoDumping() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GMainWindow::ShowSkylanderPortal() {
|
||||||
|
if (!m_skylander_window) {
|
||||||
|
m_skylander_window = SkylanderPortalWindow::get_dlg(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
m_skylander_window->show();
|
||||||
|
m_skylander_window->raise();
|
||||||
|
m_skylander_window->activateWindow();
|
||||||
|
}
|
||||||
|
|
||||||
void GMainWindow::UpdateStatusBar() {
|
void GMainWindow::UpdateStatusBar() {
|
||||||
if (!emu_thread) [[unlikely]] {
|
if (!emu_thread) [[unlikely]] {
|
||||||
status_bar_update_timer.stop();
|
status_bar_update_timer.stop();
|
||||||
|
|
|
@ -49,6 +49,7 @@ class QProgressBar;
|
||||||
class QPushButton;
|
class QPushButton;
|
||||||
class QSlider;
|
class QSlider;
|
||||||
class RegistersWidget;
|
class RegistersWidget;
|
||||||
|
class SkylanderPortalWindow;
|
||||||
#if ENABLE_QT_UPDATER
|
#if ENABLE_QT_UPDATER
|
||||||
class Updater;
|
class Updater;
|
||||||
#endif
|
#endif
|
||||||
|
@ -248,6 +249,7 @@ private slots:
|
||||||
void OnSaveMovie();
|
void OnSaveMovie();
|
||||||
void OnCaptureScreenshot();
|
void OnCaptureScreenshot();
|
||||||
void OnDumpVideo();
|
void OnDumpVideo();
|
||||||
|
void ShowSkylanderPortal();
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
void OnOpenFFmpeg();
|
void OnOpenFFmpeg();
|
||||||
#endif
|
#endif
|
||||||
|
@ -346,6 +348,8 @@ private:
|
||||||
// Whether game was paused due to stopping video dumping
|
// Whether game was paused due to stopping video dumping
|
||||||
bool game_paused_for_dumping = false;
|
bool game_paused_for_dumping = false;
|
||||||
|
|
||||||
|
SkylanderPortalWindow* m_skylander_window = nullptr;
|
||||||
|
|
||||||
QString gl_renderer;
|
QString gl_renderer;
|
||||||
std::vector<QString> physical_devices;
|
std::vector<QString> physical_devices;
|
||||||
|
|
||||||
|
|
|
@ -193,6 +193,8 @@
|
||||||
<addaction name="separator"/>
|
<addaction name="separator"/>
|
||||||
<addaction name="action_Capture_Screenshot"/>
|
<addaction name="action_Capture_Screenshot"/>
|
||||||
<addaction name="action_Dump_Video"/>
|
<addaction name="action_Dump_Video"/>
|
||||||
|
<addaction name="separator"/>
|
||||||
|
<addaction name="action_Manage_Skylanders"/>
|
||||||
</widget>
|
</widget>
|
||||||
<widget class="QMenu" name="menu_Help">
|
<widget class="QMenu" name="menu_Help">
|
||||||
<property name="title">
|
<property name="title">
|
||||||
|
@ -435,6 +437,11 @@
|
||||||
<string>Dump Video</string>
|
<string>Dump Video</string>
|
||||||
</property>
|
</property>
|
||||||
</action>
|
</action>
|
||||||
|
<action name="action_Manage_Skylanders">
|
||||||
|
<property name="text">
|
||||||
|
<string>Manage Skylanders</string>
|
||||||
|
</property>
|
||||||
|
</action>
|
||||||
<action name="action_View_Lobby">
|
<action name="action_View_Lobby">
|
||||||
<property name="enabled">
|
<property name="enabled">
|
||||||
<bool>true</bool>
|
<bool>true</bool>
|
||||||
|
|
|
@ -325,6 +325,8 @@ add_library(citra_core STATIC
|
||||||
hle/service/ir/extra_hid.h
|
hle/service/ir/extra_hid.h
|
||||||
hle/service/ir/ir.cpp
|
hle/service/ir/ir.cpp
|
||||||
hle/service/ir/ir.h
|
hle/service/ir/ir.h
|
||||||
|
hle/service/ir/ir_portal.cpp
|
||||||
|
hle/service/ir/ir_portal.h
|
||||||
hle/service/ir/ir_rst.cpp
|
hle/service/ir/ir_rst.cpp
|
||||||
hle/service/ir/ir_rst.h
|
hle/service/ir/ir_rst.h
|
||||||
hle/service/ir/ir_u.cpp
|
hle/service/ir/ir_u.cpp
|
||||||
|
|
311
src/core/hle/service/ir/ir_portal.cpp
Normal file
311
src/core/hle/service/ir/ir_portal.cpp
Normal file
|
@ -0,0 +1,311 @@
|
||||||
|
// Copyright 2024 Citra Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include <fmt/format.h>
|
||||||
|
#include "common/alignment.h"
|
||||||
|
#include "common/settings.h"
|
||||||
|
#include "core/core_timing.h"
|
||||||
|
#include "core/hle/service/ir/ir_portal.h"
|
||||||
|
#include "core/movie.h"
|
||||||
|
|
||||||
|
namespace Service::IR {
|
||||||
|
|
||||||
|
SkylanderPortal g_skyportal;
|
||||||
|
|
||||||
|
IRPortal::IRPortal(SendFunc send_func) : IRDevice(send_func) {}
|
||||||
|
|
||||||
|
IRPortal::~IRPortal() {
|
||||||
|
OnDisconnect();
|
||||||
|
}
|
||||||
|
|
||||||
|
void IRPortal::OnConnect() {}
|
||||||
|
|
||||||
|
void IRPortal::OnDisconnect() {}
|
||||||
|
|
||||||
|
void IRPortal::OnReceive(std::span<const u8> data) {
|
||||||
|
HandlePortalCommand(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
void IRPortal::HandlePortalCommand(std::span<const u8> data) {
|
||||||
|
// Data to be queued to be sent back via the Interrupt Transfer (if needed)
|
||||||
|
std::array<u8, 64> response = {};
|
||||||
|
|
||||||
|
// The first byte of the Control Request is always a char for Skylanders (offset by 3 for infrared commands)
|
||||||
|
switch (data[3]) {
|
||||||
|
case 'A': {
|
||||||
|
response = {0x41, data[4], 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};
|
||||||
|
if (data[4] == 0x01) {
|
||||||
|
g_skyportal.Activate();
|
||||||
|
}
|
||||||
|
if (data[4] == 0x00) {
|
||||||
|
g_skyportal.Deactivate();
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'C': {
|
||||||
|
g_skyportal.SetLEDs(0x01, data[4], data[5], data[6]);
|
||||||
|
response = g_skyportal.GetStatus();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'J': {
|
||||||
|
response = {data[3]};
|
||||||
|
g_skyportal.SetLEDs(data[4], data[5], data[6], data[7]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'L': {
|
||||||
|
u8 side = data[4];
|
||||||
|
if (side == 0x02) {
|
||||||
|
side = 0x04;
|
||||||
|
}
|
||||||
|
g_skyportal.SetLEDs(side, data[5], data[6], data[7]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'M': {
|
||||||
|
response = {data[3], data[4], 0x00, 0x19};
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'Q': {
|
||||||
|
const u8 sky_num = data[4] & 0xF;
|
||||||
|
const u8 block = data[5];
|
||||||
|
g_skyportal.QueryBlock(sky_num, block, response.data());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'R': {
|
||||||
|
response = {0x52, 0x02, 0x02, 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};
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'S': {
|
||||||
|
response = g_skyportal.GetStatus();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'V': {
|
||||||
|
response = g_skyportal.GetStatus();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'W': {
|
||||||
|
const u8 sky_num = data[4] & 0xF;
|
||||||
|
const u8 block = data[5];
|
||||||
|
g_skyportal.WriteBlock(sky_num, block, &data[6], response.data());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
LOG_ERROR(Service_IR, "Unhandled Skylander Portal Query: {}", data[3]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
Send(response);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Skylander::Save() {
|
||||||
|
if (!sky_file) {
|
||||||
|
LOG_ERROR(Service_IR, "Tried to save a Skylander but no file was open");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
sky_file.Seek(0, SEEK_SET);
|
||||||
|
sky_file.WriteBytes(data.data(), 0x40 * 0x10);
|
||||||
|
sky_file.Close();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SkylanderPortal::IsActivated() {
|
||||||
|
return m_activated;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SkylanderPortal::Activate() {
|
||||||
|
std::lock_guard lock(sky_mutex);
|
||||||
|
if (m_activated) {
|
||||||
|
// If the portal was already active no change is needed
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If not we need to advertise change to all the figures present on the portal
|
||||||
|
for (auto& s : skylanders) {
|
||||||
|
if (s.status & 1) {
|
||||||
|
s.queued_status.push(Skylander::ADDED);
|
||||||
|
s.queued_status.push(Skylander::READY);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
m_activated = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SkylanderPortal::Deactivate() {
|
||||||
|
std::lock_guard lock(sky_mutex);
|
||||||
|
|
||||||
|
for (auto& s : skylanders) {
|
||||||
|
// check if at the end of the updates there would be a figure on the portal
|
||||||
|
if (!s.queued_status.empty()) {
|
||||||
|
s.status = s.queued_status.back();
|
||||||
|
s.queued_status = std::queue<u8>();
|
||||||
|
}
|
||||||
|
|
||||||
|
s.status &= 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_activated = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SkylanderPortal::SetLEDs(u8 side, u8 red, u8 green, u8 blue) {
|
||||||
|
std::lock_guard lock(sky_mutex);
|
||||||
|
if (side == 0x00) {
|
||||||
|
m_color_right.red = red;
|
||||||
|
m_color_right.green = green;
|
||||||
|
m_color_right.blue = blue;
|
||||||
|
} else if (side == 0x01) {
|
||||||
|
m_color_right.red = red;
|
||||||
|
m_color_right.green = green;
|
||||||
|
m_color_right.blue = blue;
|
||||||
|
|
||||||
|
m_color_left.red = red;
|
||||||
|
m_color_left.green = green;
|
||||||
|
m_color_left.blue = blue;
|
||||||
|
} else if (side == 0x02) {
|
||||||
|
m_color_left.red = red;
|
||||||
|
m_color_left.green = green;
|
||||||
|
m_color_left.blue = blue;
|
||||||
|
} else if (side == 0x03) {
|
||||||
|
m_color_trap.red = red;
|
||||||
|
m_color_trap.green = green;
|
||||||
|
m_color_trap.blue = blue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::array<u8, 64> SkylanderPortal::GetStatus() {
|
||||||
|
std::lock_guard lock(sky_mutex);
|
||||||
|
|
||||||
|
u32 status = 0;
|
||||||
|
u8 active = 0x00;
|
||||||
|
|
||||||
|
if (m_activated) {
|
||||||
|
active = 0x01;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = MAX_SKYLANDERS - 1; i >= 0; i--) {
|
||||||
|
auto& s = skylanders[i];
|
||||||
|
|
||||||
|
if (!s.queued_status.empty()) {
|
||||||
|
s.status = s.queued_status.front();
|
||||||
|
s.queued_status.pop();
|
||||||
|
}
|
||||||
|
status <<= 2;
|
||||||
|
status |= s.status;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::array<u8, 64> response = {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(&response[1], &status, sizeof(status));
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SkylanderPortal::QueryBlock(u8 sky_num, u8 block, u8* reply_buf) {
|
||||||
|
if (!IsSkylanderNumberValid(sky_num) || !IsBlockNumberValid(block))
|
||||||
|
return;
|
||||||
|
|
||||||
|
std::lock_guard lock(sky_mutex);
|
||||||
|
|
||||||
|
const auto& skylander = skylanders[sky_num];
|
||||||
|
|
||||||
|
reply_buf[0] = 'Q';
|
||||||
|
reply_buf[2] = block;
|
||||||
|
if (skylander.status & Skylander::READY) {
|
||||||
|
reply_buf[1] = (0x10 | sky_num);
|
||||||
|
memcpy(&reply_buf[3], skylander.data.data() + (block * 0x10), 0x10);
|
||||||
|
} else {
|
||||||
|
reply_buf[1] = 0x01;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SkylanderPortal::WriteBlock(u8 sky_num, u8 block, const u8* to_write_buf, u8* reply_buf) {
|
||||||
|
if (!IsSkylanderNumberValid(sky_num) || !IsBlockNumberValid(block))
|
||||||
|
return;
|
||||||
|
|
||||||
|
std::lock_guard lock(sky_mutex);
|
||||||
|
|
||||||
|
auto& skylander = skylanders[sky_num];
|
||||||
|
|
||||||
|
reply_buf[0] = 'W';
|
||||||
|
reply_buf[2] = block;
|
||||||
|
|
||||||
|
if (skylander.status & 1) {
|
||||||
|
reply_buf[1] = (0x10 | sky_num);
|
||||||
|
memcpy(skylander.data.data() + (block * 0x10), to_write_buf, 0x10);
|
||||||
|
skylander.Save();
|
||||||
|
} else {
|
||||||
|
reply_buf[1] = 0x01;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SkylanderPortal::RemoveSkylander(u8 sky_num) {
|
||||||
|
if (!IsSkylanderNumberValid(sky_num))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_IR, "Cleared Skylander from slot {}", sky_num);
|
||||||
|
std::lock_guard lock(sky_mutex);
|
||||||
|
auto& skylander = skylanders[sky_num];
|
||||||
|
|
||||||
|
skylander.Save();
|
||||||
|
|
||||||
|
if (skylander.status & Skylander::READY) {
|
||||||
|
skylander.status = Skylander::REMOVING;
|
||||||
|
skylander.queued_status.push(Skylander::REMOVING);
|
||||||
|
skylander.queued_status.push(Skylander::REMOVED);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
u8 SkylanderPortal::LoadSkylander(u8* buf, FileUtil::IOFile in_file) {
|
||||||
|
std::lock_guard lock(sky_mutex);
|
||||||
|
|
||||||
|
u32 sky_serial = 0;
|
||||||
|
for (int i = 3; i > -1; i--) {
|
||||||
|
sky_serial <<= 8;
|
||||||
|
sky_serial |= buf[i];
|
||||||
|
}
|
||||||
|
u8 found_slot = 0xFF;
|
||||||
|
|
||||||
|
// mimics spot retaining on the portal
|
||||||
|
for (u8 i = 0; i < 8; i++) {
|
||||||
|
if ((skylanders[i].status & 1) == 0) {
|
||||||
|
if (skylanders[i].last_id == sky_serial) {
|
||||||
|
found_slot = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (i < found_slot) {
|
||||||
|
found_slot = i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (found_slot != 0xff) {
|
||||||
|
Skylander& thesky = skylanders[found_slot];
|
||||||
|
memcpy(thesky.data.data(), buf, thesky.data.size());
|
||||||
|
thesky.sky_file = std::move(in_file);
|
||||||
|
thesky.status = 3;
|
||||||
|
thesky.queued_status.push(3);
|
||||||
|
thesky.queued_status.push(1);
|
||||||
|
thesky.last_id = sky_serial;
|
||||||
|
}
|
||||||
|
|
||||||
|
return found_slot;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SkylanderPortal::IsSkylanderNumberValid(u8 sky_num) {
|
||||||
|
return sky_num < MAX_SKYLANDERS;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SkylanderPortal::IsBlockNumberValid(u8 block) {
|
||||||
|
return block < 64;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Service::IR
|
104
src/core/hle/service/ir/ir_portal.h
Normal file
104
src/core/hle/service/ir/ir_portal.h
Normal file
|
@ -0,0 +1,104 @@
|
||||||
|
// Copyright 2024 Citra Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
#include <atomic>
|
||||||
|
#include <span>
|
||||||
|
#include <boost/serialization/array.hpp>
|
||||||
|
#include <queue>
|
||||||
|
#include "common/bit_field.h"
|
||||||
|
#include "common/file_util.h"
|
||||||
|
#include "common/swap.h"
|
||||||
|
#include "core/frontend/input.h"
|
||||||
|
#include "core/hle/service/ir/ir_user.h"
|
||||||
|
|
||||||
|
constexpr u8 MAX_SKYLANDERS = 16;
|
||||||
|
|
||||||
|
namespace Core {
|
||||||
|
struct TimingEventType;
|
||||||
|
class Timing;
|
||||||
|
class Movie;
|
||||||
|
} // namespace Core
|
||||||
|
|
||||||
|
namespace Service::IR {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An IRDevice emulating Circle Pad Pro or New 3DS additional HID hardware.
|
||||||
|
* This device sends periodic udates at a rate configured by the 3DS, and sends calibration data if
|
||||||
|
* requested.
|
||||||
|
*/
|
||||||
|
class IRPortal final : public IRDevice {
|
||||||
|
public:
|
||||||
|
explicit IRPortal(SendFunc send_func);
|
||||||
|
~IRPortal();
|
||||||
|
|
||||||
|
void OnConnect() override;
|
||||||
|
void OnDisconnect() override;
|
||||||
|
void OnReceive(std::span<const u8> data) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void HandlePortalCommand(std::span<const u8> request);
|
||||||
|
|
||||||
|
template <class Archive>
|
||||||
|
void serialize(Archive& ar, const unsigned int) {
|
||||||
|
if (Archive::is_loading::value) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
friend class boost::serialization::access;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Skylander final {
|
||||||
|
FileUtil::IOFile sky_file;
|
||||||
|
u8 status = 0;
|
||||||
|
std::queue<u8> queued_status;
|
||||||
|
std::array<u8, 0x40 * 0x10> data{};
|
||||||
|
u32 last_id = 0;
|
||||||
|
|
||||||
|
void Save();
|
||||||
|
|
||||||
|
enum : u8 { REMOVED = 0, READY = 1, REMOVING = 2, ADDED = 3 };
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SkylanderLEDColor final {
|
||||||
|
u8 red = 0;
|
||||||
|
u8 green = 0;
|
||||||
|
u8 blue = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class SkylanderPortal final {
|
||||||
|
public:
|
||||||
|
void Activate();
|
||||||
|
void Deactivate();
|
||||||
|
bool IsActivated();
|
||||||
|
void SetLEDs(u8 side, u8 r, u8 g, u8 b);
|
||||||
|
|
||||||
|
std::array<u8, 64> GetStatus();
|
||||||
|
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 RemoveSkylander(u8 sky_num);
|
||||||
|
u8 LoadSkylander(u8* data, FileUtil::IOFile sky_file);
|
||||||
|
|
||||||
|
private:
|
||||||
|
static bool IsSkylanderNumberValid(u8 sky_num);
|
||||||
|
static bool IsBlockNumberValid(u8 block);
|
||||||
|
|
||||||
|
std::mutex sky_mutex;
|
||||||
|
|
||||||
|
bool m_activated = false;
|
||||||
|
u8 m_interrupt_counter = 0;
|
||||||
|
SkylanderLEDColor m_color_right = {};
|
||||||
|
SkylanderLEDColor m_color_left = {};
|
||||||
|
SkylanderLEDColor m_color_trap = {};
|
||||||
|
|
||||||
|
std::array<Skylander, MAX_SKYLANDERS> skylanders;
|
||||||
|
};
|
||||||
|
|
||||||
|
extern SkylanderPortal g_skyportal;
|
||||||
|
|
||||||
|
} // namespace Service::IR
|
||||||
|
|
||||||
|
BOOST_CLASS_EXPORT_KEY(Service::IR::IRPortal)
|
|
@ -16,6 +16,7 @@
|
||||||
#include "core/hle/kernel/event.h"
|
#include "core/hle/kernel/event.h"
|
||||||
#include "core/hle/kernel/shared_memory.h"
|
#include "core/hle/kernel/shared_memory.h"
|
||||||
#include "core/hle/service/ir/extra_hid.h"
|
#include "core/hle/service/ir/extra_hid.h"
|
||||||
|
#include "core/hle/service/ir/ir_portal.h"
|
||||||
#include "core/hle/service/ir/ir_user.h"
|
#include "core/hle/service/ir/ir_user.h"
|
||||||
|
|
||||||
SERIALIZE_EXPORT_IMPL(Service::IR::IR_USER)
|
SERIALIZE_EXPORT_IMPL(Service::IR::IR_USER)
|
||||||
|
@ -30,9 +31,11 @@ void IR_USER::serialize(Archive& ar, const unsigned int) {
|
||||||
ar& send_event;
|
ar& send_event;
|
||||||
ar& receive_event;
|
ar& receive_event;
|
||||||
ar& shared_memory;
|
ar& shared_memory;
|
||||||
ar& connected_device;
|
ar& connected_circle_pad;
|
||||||
|
ar& connected_portal;
|
||||||
ar& receive_buffer;
|
ar& receive_buffer;
|
||||||
ar&* extra_hid.get();
|
ar&* extra_hid.get();
|
||||||
|
ar&* ir_portal.get();
|
||||||
}
|
}
|
||||||
|
|
||||||
// This is a header that will present in the ir:USER shared memory if it is initialized with
|
// This is a header that will present in the ir:USER shared memory if it is initialized with
|
||||||
|
@ -315,7 +318,7 @@ void IR_USER::RequireConnection(Kernel::HLERequestContext& ctx) {
|
||||||
shared_memory_ptr[offsetof(SharedMemoryHeader, connection_role)] = 2;
|
shared_memory_ptr[offsetof(SharedMemoryHeader, connection_role)] = 2;
|
||||||
shared_memory_ptr[offsetof(SharedMemoryHeader, connected)] = 1;
|
shared_memory_ptr[offsetof(SharedMemoryHeader, connected)] = 1;
|
||||||
|
|
||||||
connected_device = true;
|
connected_circle_pad = true;
|
||||||
extra_hid->OnConnect();
|
extra_hid->OnConnect();
|
||||||
conn_status_event->Signal();
|
conn_status_event->Signal();
|
||||||
} else {
|
} else {
|
||||||
|
@ -330,6 +333,42 @@ void IR_USER::RequireConnection(Kernel::HLERequestContext& ctx) {
|
||||||
LOG_INFO(Service_IR, "called, device_id = {}", device_id);
|
LOG_INFO(Service_IR, "called, device_id = {}", device_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void IR_USER::AutoConnection(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp(ctx);
|
||||||
|
const u32 param_one = rp.Pop<u32>();
|
||||||
|
const u32 param_two = rp.Pop<u32>();
|
||||||
|
const u8 param_three = rp.Pop<u8>();
|
||||||
|
const u32 param_four = rp.Pop<u32>();
|
||||||
|
const u8 param_five = rp.Pop<u8>();
|
||||||
|
const u32 param_six = rp.Pop<u32>();
|
||||||
|
const u8 param_seven = rp.Pop<u8>();
|
||||||
|
const u32 param_eight = rp.Pop<u32>();
|
||||||
|
const u8 param_nine = rp.Pop<u8>();
|
||||||
|
const u32 param_ten = rp.Pop<u32>();
|
||||||
|
const u8 param_eleven = rp.Pop<u8>();
|
||||||
|
|
||||||
|
u8* shared_memory_ptr = shared_memory->GetPointer();
|
||||||
|
shared_memory_ptr[offsetof(SharedMemoryHeader, connection_status)] = 2;
|
||||||
|
shared_memory_ptr[offsetof(SharedMemoryHeader, connection_role)] = 2;
|
||||||
|
shared_memory_ptr[offsetof(SharedMemoryHeader, connected)] = 1;
|
||||||
|
|
||||||
|
connected_portal = true;
|
||||||
|
conn_status_event->Signal();
|
||||||
|
|
||||||
|
IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
|
||||||
|
LOG_INFO(Service_IR,
|
||||||
|
"called, one={}, two={}, "
|
||||||
|
"three={}, four={}, "
|
||||||
|
"five={}, six={}, "
|
||||||
|
"seven={}, eight={}, "
|
||||||
|
"nine={}, ten={}, "
|
||||||
|
"eleven={}",
|
||||||
|
param_one, param_two, param_three, param_four, param_five, param_six, param_seven,
|
||||||
|
param_eight, param_nine, param_ten, param_eleven);
|
||||||
|
}
|
||||||
|
|
||||||
void IR_USER::GetReceiveEvent(Kernel::HLERequestContext& ctx) {
|
void IR_USER::GetReceiveEvent(Kernel::HLERequestContext& ctx) {
|
||||||
IPC::RequestBuilder rb(ctx, 0x0A, 1, 2);
|
IPC::RequestBuilder rb(ctx, 0x0A, 1, 2);
|
||||||
|
|
||||||
|
@ -349,9 +388,14 @@ void IR_USER::GetSendEvent(Kernel::HLERequestContext& ctx) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void IR_USER::Disconnect(Kernel::HLERequestContext& ctx) {
|
void IR_USER::Disconnect(Kernel::HLERequestContext& ctx) {
|
||||||
if (connected_device) {
|
if (connected_circle_pad) {
|
||||||
extra_hid->OnDisconnect();
|
extra_hid->OnDisconnect();
|
||||||
connected_device = false;
|
connected_circle_pad = false;
|
||||||
|
conn_status_event->Signal();
|
||||||
|
}
|
||||||
|
if (connected_portal) {
|
||||||
|
ir_portal->OnDisconnect();
|
||||||
|
connected_portal = false;
|
||||||
conn_status_event->Signal();
|
conn_status_event->Signal();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -374,10 +418,29 @@ void IR_USER::GetConnectionStatusEvent(Kernel::HLERequestContext& ctx) {
|
||||||
LOG_INFO(Service_IR, "called");
|
LOG_INFO(Service_IR, "called");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void IR_USER::GetConnectionStatus(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestBuilder rb(ctx, 0x13, 1, 0);
|
||||||
|
|
||||||
|
if (connected_portal) {
|
||||||
|
conn_status_event->Signal();
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
} else {
|
||||||
|
LOG_ERROR(Service_IR, "not connected");
|
||||||
|
rb.Push(Result(static_cast<ErrorDescription>(0x13), ErrorModule::IR,
|
||||||
|
ErrorSummary::InvalidState, ErrorLevel::Status));
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG_INFO(Service_IR, "called");
|
||||||
|
}
|
||||||
|
|
||||||
void IR_USER::FinalizeIrNop(Kernel::HLERequestContext& ctx) {
|
void IR_USER::FinalizeIrNop(Kernel::HLERequestContext& ctx) {
|
||||||
if (connected_device) {
|
if (connected_circle_pad) {
|
||||||
extra_hid->OnDisconnect();
|
extra_hid->OnDisconnect();
|
||||||
connected_device = false;
|
connected_circle_pad = false;
|
||||||
|
}
|
||||||
|
if (connected_portal) {
|
||||||
|
ir_portal->OnDisconnect();
|
||||||
|
connected_portal = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
shared_memory = nullptr;
|
shared_memory = nullptr;
|
||||||
|
@ -396,10 +459,15 @@ void IR_USER::SendIrNop(Kernel::HLERequestContext& ctx) {
|
||||||
ASSERT(size == buffer.size());
|
ASSERT(size == buffer.size());
|
||||||
|
|
||||||
IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
|
IPC::RequestBuilder rb = rp.MakeBuilder(1, 0);
|
||||||
if (connected_device) {
|
if (connected_circle_pad) {
|
||||||
extra_hid->OnReceive(buffer);
|
extra_hid->OnReceive(buffer);
|
||||||
send_event->Signal();
|
send_event->Signal();
|
||||||
rb.Push(ResultSuccess);
|
rb.Push(ResultSuccess);
|
||||||
|
} else if (connected_portal) {
|
||||||
|
ir_portal->OnReceive(buffer);
|
||||||
|
send_event->Signal();
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
LOG_ERROR(Service_IR, "not connected");
|
LOG_ERROR(Service_IR, "not connected");
|
||||||
rb.Push(Result(static_cast<ErrorDescription>(13), ErrorModule::IR,
|
rb.Push(Result(static_cast<ErrorDescription>(13), ErrorModule::IR,
|
||||||
|
@ -435,7 +503,7 @@ IR_USER::IR_USER(Core::System& system) : ServiceFramework("ir:USER", 1) {
|
||||||
{0x0004, nullptr, "ClearSendBuffer"},
|
{0x0004, nullptr, "ClearSendBuffer"},
|
||||||
{0x0005, nullptr, "WaitConnection"},
|
{0x0005, nullptr, "WaitConnection"},
|
||||||
{0x0006, &IR_USER::RequireConnection, "RequireConnection"},
|
{0x0006, &IR_USER::RequireConnection, "RequireConnection"},
|
||||||
{0x0007, nullptr, "AutoConnection"},
|
{0x0007, &IR_USER::AutoConnection, "AutoConnection"},
|
||||||
{0x0008, nullptr, "AnyConnection"},
|
{0x0008, nullptr, "AnyConnection"},
|
||||||
{0x0009, &IR_USER::Disconnect, "Disconnect"},
|
{0x0009, &IR_USER::Disconnect, "Disconnect"},
|
||||||
{0x000A, &IR_USER::GetReceiveEvent, "GetReceiveEvent"},
|
{0x000A, &IR_USER::GetReceiveEvent, "GetReceiveEvent"},
|
||||||
|
@ -447,7 +515,7 @@ IR_USER::IR_USER(Core::System& system) : ServiceFramework("ir:USER", 1) {
|
||||||
{0x0010, nullptr, "ReceiveIrnopLarge"},
|
{0x0010, nullptr, "ReceiveIrnopLarge"},
|
||||||
{0x0011, nullptr, "GetLatestReceiveErrorResult"},
|
{0x0011, nullptr, "GetLatestReceiveErrorResult"},
|
||||||
{0x0012, nullptr, "GetLatestSendErrorResult"},
|
{0x0012, nullptr, "GetLatestSendErrorResult"},
|
||||||
{0x0013, nullptr, "GetConnectionStatus"},
|
{0x0013, &IR_USER::GetConnectionStatus, "GetConnectionStatus"},
|
||||||
{0x0014, nullptr, "GetTryingToConnectStatus"},
|
{0x0014, nullptr, "GetTryingToConnectStatus"},
|
||||||
{0x0015, nullptr, "GetReceiveSizeFreeAndUsed"},
|
{0x0015, nullptr, "GetReceiveSizeFreeAndUsed"},
|
||||||
{0x0016, nullptr, "GetSendSizeFreeAndUsed"},
|
{0x0016, nullptr, "GetSendSizeFreeAndUsed"},
|
||||||
|
@ -461,19 +529,25 @@ IR_USER::IR_USER(Core::System& system) : ServiceFramework("ir:USER", 1) {
|
||||||
|
|
||||||
using namespace Kernel;
|
using namespace Kernel;
|
||||||
|
|
||||||
connected_device = false;
|
connected_circle_pad = false;
|
||||||
|
connected_portal = false;
|
||||||
conn_status_event = system.Kernel().CreateEvent(ResetType::OneShot, "IR:ConnectionStatusEvent");
|
conn_status_event = system.Kernel().CreateEvent(ResetType::OneShot, "IR:ConnectionStatusEvent");
|
||||||
send_event = system.Kernel().CreateEvent(ResetType::OneShot, "IR:SendEvent");
|
send_event = system.Kernel().CreateEvent(ResetType::OneShot, "IR:SendEvent");
|
||||||
receive_event = system.Kernel().CreateEvent(ResetType::OneShot, "IR:ReceiveEvent");
|
receive_event = system.Kernel().CreateEvent(ResetType::OneShot, "IR:ReceiveEvent");
|
||||||
|
|
||||||
extra_hid = std::make_unique<ExtraHID>([this](std::span<const u8> data) { PutToReceive(data); },
|
extra_hid = std::make_unique<ExtraHID>([this](std::span<const u8> data) { PutToReceive(data); },
|
||||||
system.CoreTiming(), system.Movie());
|
system.CoreTiming(), system.Movie());
|
||||||
|
ir_portal =
|
||||||
|
std::make_unique<IRPortal>([this](std::span<const u8> data) { PutToReceive(data); });
|
||||||
}
|
}
|
||||||
|
|
||||||
IR_USER::~IR_USER() {
|
IR_USER::~IR_USER() {
|
||||||
if (connected_device) {
|
if (connected_circle_pad) {
|
||||||
extra_hid->OnDisconnect();
|
extra_hid->OnDisconnect();
|
||||||
}
|
}
|
||||||
|
if (connected_portal) {
|
||||||
|
ir_portal->OnDisconnect();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void IR_USER::ReloadInputDevices() {
|
void IR_USER::ReloadInputDevices() {
|
||||||
|
|
|
@ -18,6 +18,7 @@ namespace Service::IR {
|
||||||
|
|
||||||
class BufferManager;
|
class BufferManager;
|
||||||
class ExtraHID;
|
class ExtraHID;
|
||||||
|
class IRPortal;
|
||||||
|
|
||||||
/// An interface representing a device that can communicate with 3DS via ir:USER service
|
/// An interface representing a device that can communicate with 3DS via ir:USER service
|
||||||
class IRDevice {
|
class IRDevice {
|
||||||
|
@ -91,6 +92,8 @@ private:
|
||||||
*/
|
*/
|
||||||
void RequireConnection(Kernel::HLERequestContext& ctx);
|
void RequireConnection(Kernel::HLERequestContext& ctx);
|
||||||
|
|
||||||
|
void AutoConnection(Kernel::HLERequestContext& ctx);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* GetReceiveEvent service function
|
* GetReceiveEvent service function
|
||||||
* Gets an event that is signaled when a packet is received from the IR device.
|
* Gets an event that is signaled when a packet is received from the IR device.
|
||||||
|
@ -129,6 +132,8 @@ private:
|
||||||
*/
|
*/
|
||||||
void GetConnectionStatusEvent(Kernel::HLERequestContext& ctx);
|
void GetConnectionStatusEvent(Kernel::HLERequestContext& ctx);
|
||||||
|
|
||||||
|
void GetConnectionStatus(Kernel::HLERequestContext& ctx);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* FinalizeIrNop service function
|
* FinalizeIrNop service function
|
||||||
* Finalize ir:USER service.
|
* Finalize ir:USER service.
|
||||||
|
@ -165,9 +170,11 @@ private:
|
||||||
|
|
||||||
std::shared_ptr<Kernel::Event> conn_status_event, send_event, receive_event;
|
std::shared_ptr<Kernel::Event> conn_status_event, send_event, receive_event;
|
||||||
std::shared_ptr<Kernel::SharedMemory> shared_memory;
|
std::shared_ptr<Kernel::SharedMemory> shared_memory;
|
||||||
bool connected_device;
|
bool connected_circle_pad;
|
||||||
|
bool connected_portal;
|
||||||
std::unique_ptr<BufferManager> receive_buffer;
|
std::unique_ptr<BufferManager> receive_buffer;
|
||||||
std::unique_ptr<ExtraHID> extra_hid;
|
std::unique_ptr<ExtraHID> extra_hid;
|
||||||
|
std::unique_ptr<IRPortal> ir_portal;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
template <class Archive>
|
template <class Archive>
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue