| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | // Copyright 2015 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | #include "core/hle/service/service.h"
 | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | #include "core/hle/service/hid/hid.h"
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | #include "core/hle/service/hid/hid_spvr.h"
 | 
					
						
							|  |  |  | #include "core/hle/service/hid/hid_user.h"
 | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-16 12:16:59 -04:00
										 |  |  | #include "core/core_timing.h"
 | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | #include "core/hle/kernel/event.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/shared_memory.h"
 | 
					
						
							|  |  |  | #include "core/hle/hle.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 00:14:59 -04:00
										 |  |  | #include "video_core/video_core.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | namespace Service { | 
					
						
							|  |  |  | namespace HID { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:03:24 -04:00
										 |  |  | static const int MAX_CIRCLEPAD_POS = 0x9C; ///< Max value for a circle pad position
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  | // Handle to shared memory region designated to HID_User service
 | 
					
						
							| 
									
										
										
										
											2015-04-27 22:01:48 -04:00
										 |  |  | static Kernel::SharedPtr<Kernel::SharedMemory> shared_mem; | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  | // Event handles
 | 
					
						
							| 
									
										
										
										
											2015-04-27 22:01:48 -04:00
										 |  |  | static Kernel::SharedPtr<Kernel::Event> event_pad_or_touch_1; | 
					
						
							|  |  |  | static Kernel::SharedPtr<Kernel::Event> event_pad_or_touch_2; | 
					
						
							|  |  |  | static Kernel::SharedPtr<Kernel::Event> event_accelerometer; | 
					
						
							|  |  |  | static Kernel::SharedPtr<Kernel::Event> event_gyroscope; | 
					
						
							|  |  |  | static Kernel::SharedPtr<Kernel::Event> event_debug_pad; | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-27 22:01:48 -04:00
										 |  |  | static u32 next_pad_index; | 
					
						
							|  |  |  | static u32 next_touch_index; | 
					
						
							| 
									
										
										
										
											2015-03-08 03:05:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | // TODO(peachum):
 | 
					
						
							|  |  |  | // Add a method for setting analog input from joystick device for the circle Pad.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // This method should:
 | 
					
						
							|  |  |  | //     * Be called after both PadButton<Press, Release>().
 | 
					
						
							|  |  |  | //     * Be called before PadUpdateComplete()
 | 
					
						
							|  |  |  | //     * Set current PadEntry.circle_pad_<axis> using analog data
 | 
					
						
							|  |  |  | //     * Set PadData.raw_circle_pad_data
 | 
					
						
							|  |  |  | //     * Set PadData.current_state.circle_right = 1 if current PadEntry.circle_pad_x >= 41
 | 
					
						
							|  |  |  | //     * Set PadData.current_state.circle_up = 1 if current PadEntry.circle_pad_y >= 41
 | 
					
						
							|  |  |  | //     * Set PadData.current_state.circle_left = 1 if current PadEntry.circle_pad_x <= -41
 | 
					
						
							|  |  |  | //     * Set PadData.current_state.circle_right = 1 if current PadEntry.circle_pad_y <= -41
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  | void Update() { | 
					
						
							| 
									
										
										
										
											2015-05-10 19:47:07 -03:00
										 |  |  |     SharedMem* mem = reinterpret_cast<SharedMem*>(shared_mem->GetPointer()); | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  |     const PadState state = VideoCore::g_emu_window->GetPadState(); | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  |     if (mem == nullptr) { | 
					
						
							|  |  |  |         LOG_DEBUG(Service_HID, "Cannot update HID prior to mapping shared memory!"); | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  |     mem->pad.current_state.hex = state.hex; | 
					
						
							|  |  |  |     mem->pad.index = next_pad_index; | 
					
						
							|  |  |  |     ++next_touch_index %= mem->pad.entries.size(); | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Get the previous Pad state
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  |     u32 last_entry_index = (mem->pad.index - 1) % mem->pad.entries.size(); | 
					
						
							|  |  |  |     PadState old_state = mem->pad.entries[last_entry_index].current_state; | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Compute bitmask with 1s for bits different from the old state
 | 
					
						
							| 
									
										
										
										
											2015-03-09 00:14:59 -04:00
										 |  |  |     PadState changed = { { (state.hex ^ old_state.hex) } }; | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Get the current Pad entry
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  |     PadDataEntry* pad_entry = &mem->pad.entries[mem->pad.index]; | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Update entry properties
 | 
					
						
							| 
									
										
										
										
											2015-03-09 00:14:59 -04:00
										 |  |  |     pad_entry->current_state.hex = state.hex; | 
					
						
							|  |  |  |     pad_entry->delta_additions.hex = changed.hex & state.hex; | 
					
						
							|  |  |  |     pad_entry->delta_removals.hex = changed.hex & old_state.hex;; | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Set circle Pad
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:03:24 -04:00
										 |  |  |     pad_entry->circle_pad_x = state.circle_left  ? -MAX_CIRCLEPAD_POS : | 
					
						
							|  |  |  |                               state.circle_right ?  MAX_CIRCLEPAD_POS : 0x0; | 
					
						
							|  |  |  |     pad_entry->circle_pad_y = state.circle_down  ? -MAX_CIRCLEPAD_POS : | 
					
						
							|  |  |  |                               state.circle_up    ?  MAX_CIRCLEPAD_POS : 0x0; | 
					
						
							| 
									
										
										
										
											2015-03-08 00:12:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // If we just updated index 0, provide a new timestamp
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  |     if (mem->pad.index == 0) { | 
					
						
							|  |  |  |         mem->pad.index_reset_ticks_previous = mem->pad.index_reset_ticks; | 
					
						
							| 
									
										
										
										
											2015-03-16 12:16:59 -04:00
										 |  |  |         mem->pad.index_reset_ticks = (s64)CoreTiming::GetTicks(); | 
					
						
							| 
									
										
										
										
											2015-03-08 00:12:47 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  |     mem->touch.index = next_touch_index; | 
					
						
							|  |  |  |     ++next_touch_index %= mem->touch.entries.size(); | 
					
						
							| 
									
										
										
										
											2015-03-08 03:05:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Get the current touch entry
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  |     TouchDataEntry* touch_entry = &mem->touch.entries[mem->touch.index]; | 
					
						
							| 
									
										
										
										
											2015-03-09 00:14:59 -04:00
										 |  |  |     bool pressed = false; | 
					
						
							| 
									
										
										
										
											2015-03-08 03:05:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 00:14:59 -04:00
										 |  |  |     std::tie(touch_entry->x, touch_entry->y, pressed) = VideoCore::g_emu_window->GetTouchState(); | 
					
						
							|  |  |  |     touch_entry->valid = pressed ? 1 : 0; | 
					
						
							| 
									
										
										
										
											2015-03-08 03:05:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // TODO(bunnei): We're not doing anything with offset 0xA8 + 0x18 of HID SharedMemory, which
 | 
					
						
							|  |  |  |     // supposedly is "Touch-screen entry, which contains the raw coordinate data prior to being
 | 
					
						
							|  |  |  |     // converted to pixel coordinates." (http://3dbrew.org/wiki/HID_Shared_Memory#Offset_0xA8).
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If we just updated index 0, provide a new timestamp
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  |     if (mem->touch.index == 0) { | 
					
						
							|  |  |  |         mem->touch.index_reset_ticks_previous = mem->touch.index_reset_ticks; | 
					
						
							| 
									
										
										
										
											2015-03-16 12:16:59 -04:00
										 |  |  |         mem->touch.index_reset_ticks = (s64)CoreTiming::GetTicks(); | 
					
						
							| 
									
										
										
										
											2015-03-08 03:05:56 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     // Signal both handles when there's an update to Pad or touch
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  |     event_pad_or_touch_1->Signal(); | 
					
						
							|  |  |  |     event_pad_or_touch_2->Signal(); | 
					
						
							| 
									
										
										
										
											2015-03-08 03:05:56 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | void GetIPCHandles(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = 0; // No error
 | 
					
						
							| 
									
										
										
										
											2015-03-13 14:36:19 -07:00
										 |  |  |     cmd_buff[2] = 0x14000000; // IPC Command Structure translate-header
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  |     // TODO(yuriks): Return error from SendSyncRequest is this fails (part of IPC marshalling)
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  |     cmd_buff[3] = Kernel::g_handle_table.Create(Service::HID::shared_mem).MoveFrom(); | 
					
						
							|  |  |  |     cmd_buff[4] = Kernel::g_handle_table.Create(Service::HID::event_pad_or_touch_1).MoveFrom(); | 
					
						
							|  |  |  |     cmd_buff[5] = Kernel::g_handle_table.Create(Service::HID::event_pad_or_touch_2).MoveFrom(); | 
					
						
							|  |  |  |     cmd_buff[6] = Kernel::g_handle_table.Create(Service::HID::event_accelerometer).MoveFrom(); | 
					
						
							|  |  |  |     cmd_buff[7] = Kernel::g_handle_table.Create(Service::HID::event_gyroscope).MoveFrom(); | 
					
						
							|  |  |  |     cmd_buff[8] = Kernel::g_handle_table.Create(Service::HID::event_debug_pad).MoveFrom(); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-13 14:36:19 -07:00
										 |  |  | void EnableAccelerometer(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     event_accelerometer->Signal(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LOG_WARNING(Service_HID, "(STUBBED) called"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EnableGyroscopeLow(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     event_gyroscope->Signal(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LOG_WARNING(Service_HID, "(STUBBED) called"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GetSoundVolume(Service::Interface* self) { | 
					
						
							|  |  |  |     u32* cmd_buff = Kernel::GetCommandBuffer(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const u8 volume = 0x3F; // TODO(purpasmart): Find out if this is the max value for the volume
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cmd_buff[1] = RESULT_SUCCESS.raw; | 
					
						
							|  |  |  |     cmd_buff[2] = volume; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LOG_WARNING(Service_HID, "(STUBBED) called"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  | void Init() { | 
					
						
							| 
									
										
										
										
											2015-01-23 03:11:25 -02:00
										 |  |  |     using namespace Kernel; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  |     AddService(new HID_U_Interface); | 
					
						
							|  |  |  |     AddService(new HID_SPVR_Interface); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-10 19:47:07 -03:00
										 |  |  |     using Kernel::MemoryPermission; | 
					
						
							|  |  |  |     shared_mem = SharedMemory::Create(0x1000, MemoryPermission::ReadWrite, | 
					
						
							|  |  |  |             MemoryPermission::Read, "HID:SharedMem"); | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 00:14:59 -04:00
										 |  |  |     next_pad_index = 0; | 
					
						
							|  |  |  |     next_touch_index = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  |     // Create event handles
 | 
					
						
							| 
									
										
										
										
											2015-03-09 23:38:52 -04:00
										 |  |  |     event_pad_or_touch_1 = Event::Create(RESETTYPE_ONESHOT, "HID:EventPadOrTouch1"); | 
					
						
							|  |  |  |     event_pad_or_touch_2 = Event::Create(RESETTYPE_ONESHOT, "HID:EventPadOrTouch2"); | 
					
						
							|  |  |  |     event_accelerometer  = Event::Create(RESETTYPE_ONESHOT, "HID:EventAccelerometer"); | 
					
						
							|  |  |  |     event_gyroscope      = Event::Create(RESETTYPE_ONESHOT, "HID:EventGyroscope"); | 
					
						
							|  |  |  |     event_debug_pad      = Event::Create(RESETTYPE_ONESHOT, "HID:EventDebugPad"); | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  | void Shutdown() { | 
					
						
							| 
									
										
										
										
											2015-04-27 22:01:48 -04:00
										 |  |  |     shared_mem = nullptr; | 
					
						
							|  |  |  |     event_pad_or_touch_1 = nullptr; | 
					
						
							|  |  |  |     event_pad_or_touch_2 = nullptr; | 
					
						
							|  |  |  |     event_accelerometer = nullptr; | 
					
						
							|  |  |  |     event_gyroscope = nullptr; | 
					
						
							|  |  |  |     event_debug_pad = nullptr; | 
					
						
							| 
									
										
										
										
											2015-01-18 15:07:48 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 00:14:59 -04:00
										 |  |  | } // namespace HID
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Service
 |