rpcsx/rpcsx-os/iodev/hid.cpp

126 lines
3.3 KiB
C++
Raw Normal View History

#include "amdgpu/bridge/bridge.hpp"
#include "bridge.hpp"
2023-06-23 02:28:14 +02:00
#include "io-device.hpp"
2023-07-04 18:19:17 +02:00
#include "orbis/KernelAllocator.hpp"
#include "orbis/file.hpp"
2023-07-29 23:46:23 +02:00
#include "orbis/thread/Thread.hpp"
#include "orbis/utils/Logs.hpp"
#include <chrono>
2023-06-23 02:28:14 +02:00
struct HidDevice : public IoDevice {
orbis::ErrorCode open(orbis::Ref<orbis::File> *file, const char *path,
2023-07-29 21:46:28 +02:00
std::uint32_t flags, std::uint32_t mode,
orbis::Thread *thread) override;
};
struct HidFile : public orbis::File {};
2023-06-23 02:28:14 +02:00
struct PadState {
std::uint64_t timestamp;
std::uint32_t unk;
std::uint32_t buttons;
std::uint8_t leftStickX;
std::uint8_t leftStickY;
std::uint8_t rightStickX;
std::uint8_t rightStickY;
std::uint8_t l2;
std::uint8_t r2;
};
static orbis::ErrorCode hid_ioctl(orbis::File *file, std::uint64_t request,
void *argp, orbis::Thread *thread) {
2023-07-29 23:46:23 +02:00
switch (request) {
case 0x800c4802:
ORBIS_LOG_FATAL("hid ioctl", request);
2023-07-29 23:46:23 +02:00
thread->retval[0] = 1; // hid id
return {};
case 0x8030482e: {
ORBIS_LOG_FATAL("hid ioctl", request);
2023-07-29 23:46:23 +02:00
// read state
struct ReadStateArgs {
std::uint32_t hidId;
std::uint32_t unk0;
amdgpu::bridge::PadState *state;
2023-07-29 23:46:23 +02:00
std::uint32_t unk2;
std::uint32_t *connected;
std::uint32_t *unk4;
2023-07-29 23:46:23 +02:00
std::uint64_t unk5;
};
// enum {
// kPadBtnL3 = 1 << 1,
// kPadBtnR3 = 1 << 2,
// kPadBtnOptions = 1 << 3,
// kPadBtnUp = 1 << 4,
// kPadBtnRight = 1 << 5,
// kPadBtnDown = 1 << 6,
// kPadBtnLeft = 1 << 7,
// kPadBtnL2 = 1 << 8,
// kPadBtnR2 = 1 << 9,
// kPadBtnL1 = 1 << 10,
// kPadBtnR1 = 1 << 11,
// kPadBtnTriangle = 1 << 12,
// kPadBtnCircle = 1 << 13,
// kPadBtnCross = 1 << 14,
// kPadBtnSquare = 1 << 15,
// kPadBtnTouchPad = 1 << 20,
// kPadBtnIntercepted = 1 << 31,
// };
2023-07-29 23:46:23 +02:00
auto args = *reinterpret_cast<ReadStateArgs *>(argp);
// ORBIS_LOG_ERROR("hid read state", args.hidId, args.unk0, args.state,
// args.unk2, args.connected, args.unk4, args.unk5);
*args.state = rx::bridge.header->kbPadState;
*args.connected = 1;
*args.unk4 = 1; // is wireless?
thread->retval[0] = 1;
return {};
}
case 0x8010480e: {
// read information
ORBIS_LOG_ERROR("hid read information");
return {};
2023-07-29 23:46:23 +02:00
}
case 0x80204829: {
struct MiniReadStateArgs {
orbis::uint hidId;
orbis::uint unk0;
orbis::ptr<amdgpu::bridge::PadState> state;
orbis::uint count;
orbis::uint padding;
orbis::ptr<orbis::uint> unk5;
};
auto args = *reinterpret_cast<MiniReadStateArgs *>(argp);
*args.state = rx::bridge.header->kbPadState;
thread->retval[0] = 1;
return{};
}
2023-07-29 23:46:23 +02:00
default:
ORBIS_LOG_FATAL("Unhandled hid ioctl", request);
thread->where();
2023-07-29 23:46:23 +02:00
}
return {};
2023-06-23 02:28:14 +02:00
}
static const orbis::FileOps ops = {
.ioctl = hid_ioctl,
};
2023-06-23 02:28:14 +02:00
orbis::ErrorCode HidDevice::open(orbis::Ref<orbis::File> *file,
const char *path, std::uint32_t flags,
2023-07-29 21:46:28 +02:00
std::uint32_t mode, orbis::Thread *thread) {
auto newFile = orbis::knew<HidFile>();
newFile->device = this;
newFile->ops = &ops;
*file = newFile;
return {};
2023-06-23 02:28:14 +02:00
}
IoDevice *createHidCharacterDevice() { return orbis::knew<HidDevice>(); }