rpcsx/rpcsx-os/iodev/hid.cpp
DH 65e653f5ef [rpcsx-os/orbis-kernel] random bugfixes
ipmi: fixed respond sync, get message, try get message, try send message
event: detach event emitter from file
signals: basic implementation
linker: fixed zero symbol relocation, fixed exec relocation
shared_cv/mutex: implement eintr response support
shared_cv: fixed possible loop instead of wait
ipmi: implement invoke async, respond async, get result, get client app id, client get name
rpcsx-os: add safemode flag
2024-01-13 20:57:02 +03:00

126 lines
3.3 KiB
C++

#include "amdgpu/bridge/bridge.hpp"
#include "bridge.hpp"
#include "io-device.hpp"
#include "orbis/KernelAllocator.hpp"
#include "orbis/file.hpp"
#include "orbis/thread/Thread.hpp"
#include "orbis/utils/Logs.hpp"
#include <chrono>
struct HidDevice : public IoDevice {
orbis::ErrorCode open(orbis::Ref<orbis::File> *file, const char *path,
std::uint32_t flags, std::uint32_t mode,
orbis::Thread *thread) override;
};
struct HidFile : public orbis::File {};
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) {
switch (request) {
case 0x800c4802:
ORBIS_LOG_FATAL("hid ioctl", request);
thread->retval[0] = 1; // hid id
return {};
case 0x8030482e: {
ORBIS_LOG_FATAL("hid ioctl", request);
// read state
struct ReadStateArgs {
std::uint32_t hidId;
std::uint32_t unk0;
amdgpu::bridge::PadState *state;
std::uint32_t unk2;
std::uint32_t *connected;
std::uint32_t *unk4;
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,
// };
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 {};
}
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{};
}
default:
ORBIS_LOG_FATAL("Unhandled hid ioctl", request);
thread->where();
}
return {};
}
static const orbis::FileOps ops = {
.ioctl = hid_ioctl,
};
orbis::ErrorCode HidDevice::open(orbis::Ref<orbis::File> *file,
const char *path, std::uint32_t flags,
std::uint32_t mode, orbis::Thread *thread) {
auto newFile = orbis::knew<HidFile>();
newFile->device = this;
newFile->ops = &ops;
*file = newFile;
return {};
}
IoDevice *createHidCharacterDevice() { return orbis::knew<HidDevice>(); }