| 
									
										
										
										
											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.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-16 08:38:12 -04:00
										 |  |  | #include <cinttypes>
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | #include "core/core_timing.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/kernel.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-20 23:52:38 -07:00
										 |  |  | #include "core/hle/kernel/timer.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  | /// The event type of the generic timer callback event
 | 
					
						
							| 
									
										
										
										
											2015-04-27 22:12:35 -04:00
										 |  |  | static int timer_callback_event_type; | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  | Timer::Timer() {} | 
					
						
							|  |  |  | Timer::~Timer() {} | 
					
						
							| 
									
										
										
										
											2015-01-31 22:56:59 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-01 00:14:40 -02:00
										 |  |  | SharedPtr<Timer> Timer::Create(ResetType reset_type, std::string name) { | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  |     SharedPtr<Timer> timer(new 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-02-01 00:14:40 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return timer; | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:53:22 -05:00
										 |  |  | bool Timer::ShouldWait(Thread* thread) const { | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  |     return !signaled; | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 16:53:22 -05:00
										 |  |  | void Timer::Acquire(Thread* thread) { | 
					
						
							|  |  |  |     ASSERT_MSG(!ShouldWait(thread), "object unavailable!"); | 
					
						
							| 
									
										
										
										
											2015-12-29 20:35:25 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-12 15:06:31 -05:00
										 |  |  |     if (reset_type == ResetType::OneShot) | 
					
						
							| 
									
										
										
										
											2015-12-29 20:35:25 -05:00
										 |  |  |         signaled = false; | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-09 12:48:17 -05:00
										 |  |  |     if (initial == 0) { | 
					
						
							|  |  |  |         // Immediately invoke the callback
 | 
					
						
							|  |  |  |         Signal(0); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         u64 initial_microseconds = initial / 1000; | 
					
						
							|  |  |  |         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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 19:23:19 -05:00
										 |  |  | void Timer::WakeupAllWaitingThreads() { | 
					
						
							|  |  |  |     WaitObject::WakeupAllWaitingThreads(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (reset_type == ResetType::Pulse) | 
					
						
							|  |  |  |         signaled = false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-09 12:48:17 -05:00
										 |  |  | void Timer::Signal(int cycles_late) { | 
					
						
							|  |  |  |     LOG_TRACE(Kernel, "Timer %08" PRIx64 " fired", timer_handle); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Resume all waiting threads
 | 
					
						
							|  |  |  |     WakeupAllWaitingThreads(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (interval_delay != 0) { | 
					
						
							|  |  |  |         // Reschedule the timer with the interval delay
 | 
					
						
							|  |  |  |         u64 interval_microseconds = interval_delay / 1000; | 
					
						
							|  |  |  |         CoreTiming::ScheduleEvent(usToCycles(interval_microseconds) - cycles_late, | 
					
						
							|  |  |  |                                   timer_callback_event_type, callback_handle); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     SharedPtr<Timer> timer = | 
					
						
							|  |  |  |         timer_callback_handle_table.Get<Timer>(static_cast<Handle>(timer_handle)); | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (timer == nullptr) { | 
					
						
							| 
									
										
										
										
											2015-09-16 08:38:12 -04:00
										 |  |  |         LOG_CRITICAL(Kernel, "Callback fired for invalid timer %08" PRIx64, timer_handle); | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-09 12:48:17 -05:00
										 |  |  |     timer->Signal(cycles_late); | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TimersInit() { | 
					
						
							| 
									
										
										
										
											2015-04-27 22:12:35 -04:00
										 |  |  |     timer_callback_handle_table.Clear(); | 
					
						
							| 
									
										
										
										
											2015-01-23 02:19:33 -02:00
										 |  |  |     timer_callback_event_type = CoreTiming::RegisterEvent("TimerCallback", TimerCallback); | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  | void TimersShutdown() {} | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | } // namespace
 |