| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | // Copyright 2015 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common/common.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "core/core_timing.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/timer.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  | /// The event type of the generic timer callback event
 | 
					
						
							|  |  |  | static int timer_callback_event_type = -1; | 
					
						
							| 
									
										
										
										
											2015-01-31 14:23:09 -02:00
										 |  |  | // TODO(yuriks): This can be removed if Timer objects are explicitly pooled in the future, allowing
 | 
					
						
							|  |  |  | //               us to simply use a pool index or similar.
 | 
					
						
							|  |  |  | static Kernel::HandleTable timer_callback_handle_table; | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-31 22:56:59 -02:00
										 |  |  | Timer::Timer() {} | 
					
						
							|  |  |  | Timer::~Timer() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  | ResultVal<SharedPtr<Timer>> Timer::Create(ResetType reset_type, std::string name) { | 
					
						
							|  |  |  |     SharedPtr<Timer> timer(new Timer); | 
					
						
							|  |  |  |     // TOOD(yuriks): Don't create Handle (see Thread::Create())
 | 
					
						
							|  |  |  |     CASCADE_RESULT(auto unused, Kernel::g_handle_table.Create(timer)); | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     timer->reset_type = reset_type; | 
					
						
							|  |  |  |     timer->signaled = false; | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  |     timer->name = std::move(name); | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  |     timer->initial_delay = 0; | 
					
						
							|  |  |  |     timer->interval_delay = 0; | 
					
						
							| 
									
										
										
										
											2015-01-31 14:23:09 -02:00
										 |  |  |     timer->callback_handle = timer_callback_handle_table.Create(timer).MoveFrom(); | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  |     return MakeResult<SharedPtr<Timer>>(timer); | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  | bool Timer::ShouldWait() { | 
					
						
							|  |  |  |     return !signaled; | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  | void Timer::Acquire() { | 
					
						
							|  |  |  |     _assert_msg_(Kernel, !ShouldWait(), "object unavailable!"); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  | void Timer::Set(s64 initial, s64 interval) { | 
					
						
							| 
									
										
										
										
											2015-01-31 10:40:16 -02:00
										 |  |  |     // Ensure we get rid of any previous scheduled event
 | 
					
						
							|  |  |  |     Cancel(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  |     initial_delay = initial; | 
					
						
							|  |  |  |     interval_delay = interval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u64 initial_microseconds = initial / 1000; | 
					
						
							| 
									
										
										
										
											2015-01-31 14:23:09 -02:00
										 |  |  |     CoreTiming::ScheduleEvent(usToCycles(initial_microseconds), | 
					
						
							|  |  |  |             timer_callback_event_type, callback_handle); | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  | void Timer::Cancel() { | 
					
						
							| 
									
										
										
										
											2015-01-31 14:23:09 -02:00
										 |  |  |     CoreTiming::UnscheduleEvent(timer_callback_event_type, callback_handle); | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Timer::Clear() { | 
					
						
							|  |  |  |     signaled = false; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | /// The timer callback event, called when a timer is fired
 | 
					
						
							|  |  |  | static void TimerCallback(u64 timer_handle, int cycles_late) { | 
					
						
							| 
									
										
										
										
											2015-01-31 14:23:09 -02:00
										 |  |  |     SharedPtr<Timer> timer = timer_callback_handle_table.Get<Timer>(timer_handle); | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (timer == nullptr) { | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  |         LOG_CRITICAL(Kernel, "Callback fired for invalid timer %08X", timer_handle); | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LOG_TRACE(Kernel, "Timer %u fired", timer_handle); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     timer->signaled = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Resume all waiting threads
 | 
					
						
							| 
									
										
										
										
											2015-01-20 18:20:47 -05:00
										 |  |  |     timer->WakeupAllWaitingThreads(); | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (timer->reset_type == RESETTYPE_ONESHOT) | 
					
						
							|  |  |  |         timer->signaled = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (timer->interval_delay != 0) { | 
					
						
							|  |  |  |         // Reschedule the timer with the interval delay
 | 
					
						
							|  |  |  |         u64 interval_microseconds = timer->interval_delay / 1000; | 
					
						
							|  |  |  |         CoreTiming::ScheduleEvent(usToCycles(interval_microseconds) - cycles_late,  | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  |                 timer_callback_event_type, timer_handle); | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TimersInit() { | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  |     timer_callback_event_type = CoreTiming::RegisterEvent("TimerCallback", TimerCallback); | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TimersShutdown() { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace
 |