| 
									
										
										
										
											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-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/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
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-31 22:56:59 -02:00
										 |  |  | Timer::Timer() {} | 
					
						
							|  |  |  | Timer::~Timer() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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() { | 
					
						
							| 
									
										
										
										
											2015-01-20 17:16:47 -08:00
										 |  |  |     ASSERT_MSG( !ShouldWait(), "object unavailable!"); | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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-02-01 15:31:21 -05: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-04-14 21:16:59 +02:00
										 |  |  |         LOG_CRITICAL(Kernel, "Callback fired for invalid timer %08lX", 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-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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TimersShutdown() { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace
 |