2020-06-13 19:27:07 +00:00
|
|
|
#include "bluetooth_hid_report.hpp"
|
2020-06-14 22:18:46 +00:00
|
|
|
|
|
|
|
#include <atomic>
|
2020-06-22 21:14:54 +00:00
|
|
|
#include <mutex>
|
|
|
|
#include <cstring>
|
2020-06-13 22:41:14 +00:00
|
|
|
#include "bluetooth_circularbuffer.hpp"
|
2020-06-13 00:08:43 +00:00
|
|
|
#include "../btdrv_mitm_flags.hpp"
|
2020-07-11 19:10:38 +00:00
|
|
|
#include "../controllers/controllermanager.hpp"
|
2020-06-13 00:08:43 +00:00
|
|
|
|
|
|
|
#include "../btdrv_mitm_logging.hpp"
|
|
|
|
|
2020-06-13 17:15:59 +00:00
|
|
|
namespace ams::bluetooth::hid::report {
|
2020-06-13 00:08:43 +00:00
|
|
|
|
2020-06-13 17:15:59 +00:00
|
|
|
namespace {
|
2020-06-13 00:08:43 +00:00
|
|
|
|
2020-06-14 22:18:46 +00:00
|
|
|
std::atomic<bool> g_isInitialized(false);
|
|
|
|
|
2020-06-13 17:15:59 +00:00
|
|
|
os::ThreadType g_eventHandlerThread;
|
|
|
|
alignas(os::ThreadStackAlignment) u8 g_eventHandlerThreadStack[0x2000];
|
2020-06-13 00:08:43 +00:00
|
|
|
|
2020-06-14 22:18:46 +00:00
|
|
|
// This is only required on fw < 7.0.0
|
|
|
|
os::Mutex g_eventDataLock(false);
|
|
|
|
u8 g_eventDataBuffer[0x480];
|
2020-07-11 11:43:21 +00:00
|
|
|
bluetooth::HidEventType g_currentEventType;
|
2020-06-14 22:18:46 +00:00
|
|
|
|
2020-06-13 17:15:59 +00:00
|
|
|
SharedMemory g_realBtShmem;
|
|
|
|
SharedMemory g_fakeBtShmem;
|
|
|
|
|
2020-06-14 11:10:09 +00:00
|
|
|
bluetooth::CircularBuffer *g_realBuffer;
|
|
|
|
bluetooth::CircularBuffer *g_fakeBuffer;
|
2020-06-13 17:15:59 +00:00
|
|
|
|
|
|
|
os::SystemEventType g_btHidReportSystemEvent;
|
|
|
|
os::SystemEventType g_btHidReportSystemEventFwd;
|
|
|
|
os::SystemEventType g_btHidReportSystemEventUser;
|
|
|
|
|
|
|
|
u8 g_fakeReportBuffer[0x42] = {};
|
2020-07-11 11:43:21 +00:00
|
|
|
bluetooth::HidReportData *g_fakeReportData = reinterpret_cast<bluetooth::HidReportData *>(g_fakeReportBuffer);
|
2020-06-13 17:15:59 +00:00
|
|
|
|
2020-07-09 21:31:39 +00:00
|
|
|
// Buffer for hid report responses. Might be able to replace the above
|
2020-07-11 11:43:21 +00:00
|
|
|
bluetooth::HidReport g_hidReport = {};
|
2020-07-09 21:31:39 +00:00
|
|
|
|
2020-06-14 22:18:46 +00:00
|
|
|
void EventThreadFunc(void *arg) {
|
|
|
|
while (true) {
|
|
|
|
os::WaitSystemEvent(&g_btHidReportSystemEvent);
|
|
|
|
HandleEvent();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsInitialized(void) {
|
|
|
|
return g_isInitialized;
|
2020-06-13 17:15:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SharedMemory *GetRealSharedMemory(void) {
|
|
|
|
return &g_realBtShmem;
|
|
|
|
}
|
|
|
|
|
|
|
|
SharedMemory *GetFakeSharedMemory(void) {
|
|
|
|
return &g_fakeBtShmem;
|
|
|
|
}
|
|
|
|
|
|
|
|
os::SystemEventType *GetSystemEvent(void) {
|
|
|
|
return &g_btHidReportSystemEvent;
|
|
|
|
}
|
|
|
|
|
|
|
|
os::SystemEventType *GetForwardEvent(void) {
|
|
|
|
return &g_btHidReportSystemEventFwd;
|
|
|
|
}
|
|
|
|
|
|
|
|
os::SystemEventType *GetUserForwardEvent(void) {
|
|
|
|
return &g_btHidReportSystemEventUser;
|
|
|
|
}
|
|
|
|
|
2020-06-14 22:18:46 +00:00
|
|
|
Result Initialize(Handle eventHandle) {
|
|
|
|
os::AttachReadableHandleToSystemEvent(&g_btHidReportSystemEvent, eventHandle, false, os::EventClearMode_AutoClear);
|
|
|
|
|
|
|
|
R_TRY(os::CreateSystemEvent(&g_btHidReportSystemEventFwd, os::EventClearMode_AutoClear, true));
|
|
|
|
R_TRY(os::CreateSystemEvent(&g_btHidReportSystemEventUser, os::EventClearMode_AutoClear, true));
|
|
|
|
|
|
|
|
R_TRY(os::CreateThread(&g_eventHandlerThread,
|
|
|
|
EventThreadFunc,
|
|
|
|
nullptr,
|
|
|
|
g_eventHandlerThreadStack,
|
|
|
|
sizeof(g_eventHandlerThreadStack),
|
|
|
|
-10
|
|
|
|
));
|
|
|
|
|
|
|
|
os::StartThread(&g_eventHandlerThread);
|
|
|
|
|
|
|
|
g_isInitialized = true;
|
|
|
|
|
|
|
|
return ams::ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Finalize(void) {
|
|
|
|
os::DestroyThread(&g_eventHandlerThread);
|
|
|
|
|
2020-07-09 17:44:12 +00:00
|
|
|
//if (hos::GetVersion() < hos::Version_7_0_0)
|
|
|
|
//delete g_fakeBuffer;
|
|
|
|
|
2020-06-14 22:18:46 +00:00
|
|
|
os::DestroySystemEvent(&g_btHidReportSystemEventUser);
|
|
|
|
os::DestroySystemEvent(&g_btHidReportSystemEventFwd);
|
|
|
|
|
|
|
|
g_isInitialized = false;
|
|
|
|
}
|
|
|
|
|
2020-06-13 17:15:59 +00:00
|
|
|
Result MapRemoteSharedMemory(Handle handle) {
|
|
|
|
shmemLoadRemote(&g_realBtShmem, handle, BLUETOOTH_SHAREDMEM_SIZE, Perm_Rw);
|
|
|
|
R_TRY(shmemMap(&g_realBtShmem));
|
2020-06-14 11:10:09 +00:00
|
|
|
g_realBuffer = reinterpret_cast<bluetooth::CircularBuffer *>(shmemGetAddr(&g_realBtShmem));
|
2020-06-13 17:15:59 +00:00
|
|
|
|
|
|
|
return ams::ResultSuccess();
|
|
|
|
}
|
|
|
|
|
2020-07-09 17:44:12 +00:00
|
|
|
Result InitializeReportBuffer(void) {
|
|
|
|
BTDRV_LOG_FMT("btdrv-mitm: InitializeReportBuffer");
|
|
|
|
|
|
|
|
// Todo: maybe just create shared memory for all fw?
|
|
|
|
if (hos::GetVersion() < hos::Version_7_0_0) {
|
|
|
|
g_fakeBuffer = new CircularBuffer();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
R_TRY(shmemCreate(&g_fakeBtShmem, BLUETOOTH_SHAREDMEM_SIZE, Perm_Rw, Perm_Rw));
|
|
|
|
R_TRY(shmemMap(&g_fakeBtShmem));
|
2020-07-11 11:43:21 +00:00
|
|
|
g_fakeBuffer = reinterpret_cast<bluetooth::CircularBuffer *>(shmemGetAddr(&g_fakeBtShmem));
|
2020-07-09 17:44:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
g_fakeBuffer->Initialize("HID Report");
|
2020-07-11 11:43:21 +00:00
|
|
|
g_fakeBuffer->type = bluetooth::CircularBufferType_HidReport;
|
2020-07-09 17:44:12 +00:00
|
|
|
g_fakeBuffer->_unk3 = 1;
|
|
|
|
|
|
|
|
return ams::ResultSuccess();
|
|
|
|
}
|
|
|
|
|
2020-07-09 19:26:22 +00:00
|
|
|
/* Write a fake report into the circular buffer */
|
2020-07-11 11:43:21 +00:00
|
|
|
Result WriteFakeHidData(const bluetooth::Address *address, const bluetooth::HidReport *report) {
|
2020-06-30 20:23:16 +00:00
|
|
|
|
2020-08-04 20:19:12 +00:00
|
|
|
//BTDRV_LOG_DATA_MSG((void*)report, report->size + sizeof(report->size), "btdrv-mitm: WriteFakeHidData");
|
2020-06-30 20:23:16 +00:00
|
|
|
|
2020-07-11 11:43:21 +00:00
|
|
|
u16 bufferSize = report->size + 0x11;
|
2020-06-30 20:23:16 +00:00
|
|
|
u8 buffer[bufferSize] = {};
|
2020-07-11 13:24:00 +00:00
|
|
|
auto fakeReportData = reinterpret_cast<bluetooth::HidReportData *>(buffer);
|
2020-06-30 20:23:16 +00:00
|
|
|
|
|
|
|
if (hos::GetVersion() < hos::Version_9_0_0) {
|
|
|
|
fakeReportData->size = bufferSize;
|
2020-07-11 11:43:21 +00:00
|
|
|
std::memcpy(&fakeReportData->address, address, sizeof(bluetooth::Address));
|
|
|
|
std::memcpy(&fakeReportData->report, report, report->size + sizeof(report->size));
|
2020-06-30 20:23:16 +00:00
|
|
|
}
|
|
|
|
else {
|
2020-07-11 11:43:21 +00:00
|
|
|
std::memcpy(&fakeReportData->v2.address, address, sizeof(bluetooth::Address));
|
|
|
|
std::memcpy(&fakeReportData->v2.report, report, report->size + sizeof(report->size));
|
2020-06-30 20:23:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
g_fakeBuffer->Write(4, fakeReportData, bufferSize);
|
|
|
|
os::SignalSystemEvent(&g_btHidReportSystemEventFwd);
|
|
|
|
|
|
|
|
return ams::ResultSuccess();
|
|
|
|
}
|
|
|
|
|
2020-07-09 21:31:39 +00:00
|
|
|
/* Write a fake subcommand response into buffer */
|
2020-07-09 21:36:17 +00:00
|
|
|
Result FakeSubCmdResponse(const bluetooth::Address *address, const u8 response[], size_t size) {
|
2020-07-11 13:24:00 +00:00
|
|
|
auto report = &g_hidReport;
|
|
|
|
auto reportData = reinterpret_cast<controller::SwitchReportData *>(&report->data);
|
|
|
|
report->size = sizeof(controller::SwitchInputReport0x21);
|
|
|
|
reportData->id = 0x21;
|
|
|
|
reportData->input0x21.conn_info = 0;
|
|
|
|
reportData->input0x21.battery = 8;
|
|
|
|
reportData->input0x21.buttons = {0x00, 0x00, 0x00};
|
|
|
|
reportData->input0x21.left_stick = {0x0b, 0xb8, 0x78};
|
|
|
|
reportData->input0x21.right_stick = {0xd9, 0xd7, 0x81};
|
|
|
|
reportData->input0x21.vibrator = 0;
|
|
|
|
std::memcpy(&reportData->input0x21.subcmd, response, size);
|
|
|
|
|
|
|
|
reportData->input0x21.timer = os::ConvertToTimeSpan(os::GetSystemTick()).GetMilliSeconds() & 0xff;
|
2020-07-09 21:31:39 +00:00
|
|
|
|
|
|
|
// Todo: change types so we don't have to cast
|
2020-07-11 11:43:21 +00:00
|
|
|
return bluetooth::hid::report::WriteFakeHidData(address, report);
|
2020-07-09 21:31:39 +00:00
|
|
|
}
|
|
|
|
|
2020-08-04 20:19:12 +00:00
|
|
|
/* Only used for < 7.0.0. Newer firmwares read straight from shared memory */
|
2020-07-11 11:43:21 +00:00
|
|
|
Result GetEventInfo(bluetooth::HidEventType *type, u8* buffer, size_t size) {
|
2020-06-22 21:14:54 +00:00
|
|
|
|
2020-07-09 19:26:22 +00:00
|
|
|
//BTDRV_LOG_FMT("!!! GetEventInfo Called");
|
2020-06-22 21:14:54 +00:00
|
|
|
|
2020-07-09 19:26:22 +00:00
|
|
|
while (true) {
|
|
|
|
|
2020-08-04 20:19:12 +00:00
|
|
|
auto packet = g_fakeBuffer->Read();
|
2020-07-09 19:26:22 +00:00
|
|
|
if (!packet)
|
2020-08-04 20:19:12 +00:00
|
|
|
return -1;
|
2020-07-09 19:26:22 +00:00
|
|
|
|
2020-07-27 21:34:31 +00:00
|
|
|
g_fakeBuffer->Free();
|
2020-07-09 19:26:22 +00:00
|
|
|
|
2020-08-04 20:19:12 +00:00
|
|
|
if (packet->header.type == 0xff) {
|
2020-07-09 19:26:22 +00:00
|
|
|
continue;
|
2020-08-04 20:19:12 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
auto eventData = reinterpret_cast<bluetooth::HidEventData *>(buffer);
|
2020-07-09 19:26:22 +00:00
|
|
|
|
2020-08-04 20:19:12 +00:00
|
|
|
*type = static_cast<bluetooth::HidEventType>(packet->header.type);
|
|
|
|
std::memcpy(&eventData->getReport.address, &packet->data.address, sizeof(bluetooth::Address));
|
|
|
|
eventData->getReport.status = HidStatus_Ok;
|
|
|
|
eventData->getReport.report_length = packet->header.size;
|
2020-07-09 19:26:22 +00:00
|
|
|
|
2020-08-04 20:19:12 +00:00
|
|
|
std::memcpy(&eventData->getReport.report_data, &packet->data, packet->header.size);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2020-07-09 19:26:22 +00:00
|
|
|
|
|
|
|
//BTDRV_LOG_DATA_MSG(&packet->data, packet->header.size, "btdrv-mitm: hid::report::GetEventInfo -> Read");
|
2020-06-22 21:14:54 +00:00
|
|
|
|
|
|
|
return ams::ResultSuccess();
|
|
|
|
}
|
|
|
|
|
2020-07-09 19:26:22 +00:00
|
|
|
void _HandleEvent() {
|
2020-06-28 15:22:37 +00:00
|
|
|
|
2020-06-13 17:15:59 +00:00
|
|
|
while (true) {
|
|
|
|
// Get packet from real buffer
|
2020-08-04 20:19:12 +00:00
|
|
|
auto realPacket = g_realBuffer->Read();
|
2020-06-13 17:15:59 +00:00
|
|
|
if (!realPacket)
|
|
|
|
break;
|
2020-06-13 00:08:43 +00:00
|
|
|
|
2020-07-27 21:34:31 +00:00
|
|
|
g_realBuffer->Free();
|
2020-06-13 17:15:59 +00:00
|
|
|
|
|
|
|
switch (realPacket->header.type) {
|
|
|
|
case 0xff:
|
|
|
|
// Skip over packet type 0xff. This packet indicates the buffer should wrap around on next read.
|
|
|
|
continue;
|
|
|
|
|
|
|
|
case 4:
|
|
|
|
{
|
|
|
|
// Locate the controller that sent the report
|
2020-07-11 19:10:38 +00:00
|
|
|
auto device = controller::locateController(hos::GetVersion() < hos::Version_9_0_0 ? &realPacket->data.address : &realPacket->data.v2.address);
|
|
|
|
if (!device) {
|
2020-06-13 17:15:59 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-07-11 19:10:38 +00:00
|
|
|
if (device->isSwitchController()) {
|
2020-07-27 21:34:31 +00:00
|
|
|
// Write unmodified packet directly to fake buffer
|
2020-06-14 11:10:09 +00:00
|
|
|
g_fakeBuffer->Write(realPacket->header.type, &realPacket->data, realPacket->header.size);
|
2020-06-13 17:15:59 +00:00
|
|
|
}
|
|
|
|
else {
|
2020-07-11 11:43:21 +00:00
|
|
|
const bluetooth::HidReport *inReport;
|
|
|
|
bluetooth::HidReport *outReport;
|
2020-06-13 17:15:59 +00:00
|
|
|
// copy address and stuff over
|
2020-06-14 23:52:31 +00:00
|
|
|
if (hos::GetVersion() < hos::Version_9_0_0) {
|
2020-07-09 19:26:22 +00:00
|
|
|
g_fakeReportData->size = 0x42;
|
2020-07-11 11:43:21 +00:00
|
|
|
std::memcpy(&g_fakeReportData->address, &realPacket->data.address, sizeof(bluetooth::Address));
|
2020-06-13 17:15:59 +00:00
|
|
|
inReport = &realPacket->data.report;
|
|
|
|
outReport = &g_fakeReportData->report;
|
|
|
|
}
|
|
|
|
else {
|
2020-07-11 11:43:21 +00:00
|
|
|
std::memcpy(&g_fakeReportData->v2.address, &realPacket->data.v2.address, sizeof(bluetooth::Address));
|
2020-06-13 17:15:59 +00:00
|
|
|
inReport = &realPacket->data.v2.report;
|
|
|
|
outReport = &g_fakeReportData->v2.report;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto switchData = reinterpret_cast<controller::SwitchReportData *>(&outReport->data);
|
2020-07-11 13:24:00 +00:00
|
|
|
switchData->input0x30.timer = os::ConvertToTimeSpan(realPacket->header.timestamp).GetMilliSeconds() & 0xff;
|
2020-06-13 17:15:59 +00:00
|
|
|
|
|
|
|
// Translate packet to switch pro format
|
2020-07-11 19:10:38 +00:00
|
|
|
device->convertReportFormat(inReport, outReport);
|
2020-06-13 17:15:59 +00:00
|
|
|
|
|
|
|
// Write the converted report to our fake buffer
|
2020-06-25 21:50:43 +00:00
|
|
|
g_fakeBuffer->Write(4, g_fakeReportData, sizeof(g_fakeReportBuffer));
|
2020-06-13 17:15:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2020-06-13 00:08:43 +00:00
|
|
|
|
2020-06-13 17:15:59 +00:00
|
|
|
default:
|
2020-06-28 15:22:37 +00:00
|
|
|
BTDRV_LOG_FMT("unknown packet received: %d", realPacket->header.type);
|
2020-07-27 21:34:31 +00:00
|
|
|
g_fakeBuffer->Write(realPacket->header.type, &realPacket->data, realPacket->header.size);
|
2020-06-13 17:15:59 +00:00
|
|
|
break;
|
|
|
|
}
|
2020-06-13 00:08:43 +00:00
|
|
|
|
2020-06-13 17:15:59 +00:00
|
|
|
}
|
2020-07-09 19:26:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _HandleEventDeprecated(void) {
|
|
|
|
|
|
|
|
std::scoped_lock lk(g_eventDataLock);
|
|
|
|
R_ABORT_UNLESS(btdrvGetHidReportEventInfo(&g_currentEventType, g_eventDataBuffer, sizeof(g_eventDataBuffer)));
|
|
|
|
|
2020-07-11 11:43:21 +00:00
|
|
|
auto eventData = reinterpret_cast<bluetooth::HidEventData *>(g_eventDataBuffer);
|
2020-07-09 19:26:22 +00:00
|
|
|
|
|
|
|
//BTDRV_LOG_FMT("hid report event [%02d]", g_currentEventType);
|
|
|
|
|
|
|
|
switch (g_currentEventType) {
|
|
|
|
|
|
|
|
case HidEvent_GetReport:
|
|
|
|
{
|
|
|
|
// Locate the controller that sent the report
|
2020-07-11 19:10:38 +00:00
|
|
|
auto device = controller::locateController(&eventData->getReport.address);
|
|
|
|
if (!device) {
|
2020-07-09 19:26:22 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-11 19:10:38 +00:00
|
|
|
if (device->isSwitchController()) {
|
2020-07-09 19:26:22 +00:00
|
|
|
//BTDRV_LOG_DATA_MSG(&eventData->getReport.report_data, eventData->getReport.report_length, "Switch controller -> Write");
|
2020-07-09 21:31:39 +00:00
|
|
|
g_fakeBuffer->Write(g_currentEventType, &eventData->getReport.report_data, eventData->getReport.report_length);
|
2020-07-09 19:26:22 +00:00
|
|
|
}
|
|
|
|
else {
|
2020-07-11 11:43:21 +00:00
|
|
|
const bluetooth::HidReport *inReport;
|
|
|
|
bluetooth::HidReport *outReport;
|
2020-07-09 19:26:22 +00:00
|
|
|
|
|
|
|
g_fakeReportData->size = 0x42; // Todo: check size is correct for report 0x30
|
2020-07-11 11:43:21 +00:00
|
|
|
std::memcpy(&g_fakeReportData->address, &eventData->getReport.address, sizeof(bluetooth::Address));
|
2020-07-09 19:26:22 +00:00
|
|
|
inReport = &eventData->getReport.report_data.report;
|
|
|
|
outReport = &g_fakeReportData->report;
|
|
|
|
|
|
|
|
auto switchData = reinterpret_cast<controller::SwitchReportData *>(&outReport->data);
|
2020-07-11 13:24:00 +00:00
|
|
|
switchData->input0x30.timer = os::ConvertToTimeSpan(os::GetSystemTick()).GetMilliSeconds() & 0xff;
|
2020-07-09 19:26:22 +00:00
|
|
|
|
|
|
|
// Translate packet to switch pro format
|
2020-07-11 19:10:38 +00:00
|
|
|
device->convertReportFormat(inReport, outReport);
|
2020-07-09 19:26:22 +00:00
|
|
|
|
|
|
|
// Write the converted report to our fake buffer
|
|
|
|
g_fakeBuffer->Write(4, g_fakeReportData, sizeof(g_fakeReportBuffer));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
BTDRV_LOG_FMT("unknown packet received: %d", g_currentEventType);
|
2020-07-27 21:34:31 +00:00
|
|
|
g_fakeBuffer->Write(g_currentEventType, &eventData->getReport.report_data, eventData->getReport.report_length);
|
2020-07-09 19:26:22 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void HandleEvent(void) {
|
|
|
|
|
2020-07-27 21:34:31 +00:00
|
|
|
if (hos::GetVersion() < hos::Version_7_0_0)
|
2020-07-09 19:26:22 +00:00
|
|
|
_HandleEventDeprecated();
|
2020-07-27 21:34:31 +00:00
|
|
|
else
|
2020-07-09 19:26:22 +00:00
|
|
|
_HandleEvent();
|
2020-06-13 00:08:43 +00:00
|
|
|
|
2020-07-27 21:34:31 +00:00
|
|
|
if (!g_redirectHidReportEvents)
|
2020-07-09 17:38:53 +00:00
|
|
|
os::SignalSystemEvent(&g_btHidReportSystemEventFwd);
|
2020-07-27 21:34:31 +00:00
|
|
|
else
|
2020-07-09 17:38:53 +00:00
|
|
|
os::SignalSystemEvent(&g_btHidReportSystemEventUser);
|
2020-07-27 21:34:31 +00:00
|
|
|
|
2020-06-13 00:08:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|