| 
									
										
										
										
											2018-01-14 21:42:23 -05:00
										 |  |  | // Copyright 2018 yuzu emulator team
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  | #include <chrono>
 | 
					
						
							| 
									
										
										
										
											2018-05-31 15:33:30 +03:00
										 |  |  | #include <ctime>
 | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-24 23:52:21 -05:00
										 |  |  | #include "core/core_timing.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-24 11:03:24 +01:00
										 |  |  | #include "core/core_timing_util.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  | #include "core/hle/ipc_helpers.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/client_port.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/client_session.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-24 02:45:23 -04:00
										 |  |  | #include "core/hle/service/time/interface.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-14 21:42:23 -05:00
										 |  |  | #include "core/hle/service/time/time.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 21:41:44 -04:00
										 |  |  | namespace Service::Time { | 
					
						
							| 
									
										
										
										
											2018-01-14 21:42:23 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  | class ISystemClock final : public ServiceFramework<ISystemClock> { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     ISystemClock() : ServiceFramework("ISystemClock") { | 
					
						
							|  |  |  |         static const FunctionInfo functions[] = { | 
					
						
							|  |  |  |             {0, &ISystemClock::GetCurrentTime, "GetCurrentTime"}, | 
					
						
							| 
									
										
										
										
											2018-04-17 18:37:43 +03:00
										 |  |  |             {1, nullptr, "SetCurrentTime"}, | 
					
						
							|  |  |  |             {2, &ISystemClock::GetSystemClockContext, "GetSystemClockContext"}, | 
					
						
							|  |  |  |             {3, nullptr, "SetSystemClockContext"}, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         }; | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  |         RegisterHandlers(functions); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     void GetCurrentTime(Kernel::HLERequestContext& ctx) { | 
					
						
							| 
									
										
										
										
											2018-01-24 11:54:47 -03:00
										 |  |  |         const s64 time_since_epoch{std::chrono::duration_cast<std::chrono::seconds>( | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  |                                        std::chrono::system_clock::now().time_since_epoch()) | 
					
						
							|  |  |  |                                        .count()}; | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_DEBUG(Service_Time, "called"); | 
					
						
							| 
									
										
										
										
											2018-01-23 19:52:18 -05:00
										 |  |  |         IPC::ResponseBuilder rb{ctx, 4}; | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							|  |  |  |         rb.Push<u64>(time_since_epoch); | 
					
						
							| 
									
										
										
										
											2018-01-23 22:02:14 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void GetSystemClockContext(Kernel::HLERequestContext& ctx) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_WARNING(Service_Time, "(STUBBED) called"); | 
					
						
							| 
									
										
										
										
											2018-01-23 22:02:14 -05:00
										 |  |  |         SystemClockContext system_clock_ontext{}; | 
					
						
							|  |  |  |         IPC::ResponseBuilder rb{ctx, (sizeof(SystemClockContext) / 4) + 2}; | 
					
						
							|  |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							|  |  |  |         rb.PushRaw(system_clock_ontext); | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ISteadyClock final : public ServiceFramework<ISteadyClock> { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2018-01-24 23:52:21 -05:00
										 |  |  |     ISteadyClock() : ServiceFramework("ISteadyClock") { | 
					
						
							|  |  |  |         static const FunctionInfo functions[] = { | 
					
						
							|  |  |  |             {0, &ISteadyClock::GetCurrentTimePoint, "GetCurrentTimePoint"}, | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |         RegisterHandlers(functions); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     void GetCurrentTimePoint(Kernel::HLERequestContext& ctx) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_DEBUG(Service_Time, "called"); | 
					
						
							| 
									
										
										
										
											2018-04-30 03:24:27 -04:00
										 |  |  |         SteadyClockTimePoint steady_clock_time_point{ | 
					
						
							|  |  |  |             CoreTiming::cyclesToMs(CoreTiming::GetTicks()) / 1000}; | 
					
						
							| 
									
										
										
										
											2018-01-24 23:52:21 -05:00
										 |  |  |         IPC::ResponseBuilder rb{ctx, (sizeof(SteadyClockTimePoint) / 4) + 2}; | 
					
						
							|  |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							|  |  |  |         rb.PushRaw(steady_clock_time_point); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ITimeZoneService final : public ServiceFramework<ITimeZoneService> { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     ITimeZoneService() : ServiceFramework("ITimeZoneService") { | 
					
						
							|  |  |  |         static const FunctionInfo functions[] = { | 
					
						
							|  |  |  |             {0, &ITimeZoneService::GetDeviceLocationName, "GetDeviceLocationName"}, | 
					
						
							| 
									
										
										
										
											2018-02-07 15:11:17 +03:00
										 |  |  |             {1, nullptr, "SetDeviceLocationName"}, | 
					
						
							| 
									
										
										
										
											2018-01-18 00:57:11 -05:00
										 |  |  |             {2, &ITimeZoneService::GetTotalLocationNameCount, "GetTotalLocationNameCount"}, | 
					
						
							| 
									
										
										
										
											2018-02-07 15:11:17 +03:00
										 |  |  |             {3, nullptr, "LoadLocationNameList"}, | 
					
						
							|  |  |  |             {4, &ITimeZoneService::LoadTimeZoneRule, "LoadTimeZoneRule"}, | 
					
						
							|  |  |  |             {5, nullptr, "GetTimeZoneRuleVersion"}, | 
					
						
							| 
									
										
										
										
											2018-05-31 15:33:30 +03:00
										 |  |  |             {100, &ITimeZoneService::ToCalendarTime, "ToCalendarTime"}, | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  |             {101, &ITimeZoneService::ToCalendarTimeWithMyRule, "ToCalendarTimeWithMyRule"}, | 
					
						
							| 
									
										
										
										
											2018-02-07 15:11:17 +03:00
										 |  |  |             {200, nullptr, "ToPosixTime"}, | 
					
						
							|  |  |  |             {201, nullptr, "ToPosixTimeWithMyRule"}, | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  |         }; | 
					
						
							|  |  |  |         RegisterHandlers(functions); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2018-05-31 15:33:30 +03:00
										 |  |  |     LocationName location_name{"UTC"}; | 
					
						
							|  |  |  |     TimeZoneRule my_time_zone_rule{}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  |     void GetDeviceLocationName(Kernel::HLERequestContext& ctx) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_DEBUG(Service_Time, "called"); | 
					
						
							| 
									
										
										
										
											2018-01-23 19:52:18 -05:00
										 |  |  |         IPC::ResponseBuilder rb{ctx, (sizeof(LocationName) / 4) + 2}; | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							| 
									
										
										
										
											2018-01-18 00:57:11 -05:00
										 |  |  |         rb.PushRaw(location_name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void GetTotalLocationNameCount(Kernel::HLERequestContext& ctx) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_WARNING(Service_Time, "(STUBBED) called"); | 
					
						
							| 
									
										
										
										
											2018-01-23 19:52:18 -05:00
										 |  |  |         IPC::ResponseBuilder rb{ctx, 3}; | 
					
						
							| 
									
										
										
										
											2018-01-18 00:57:11 -05:00
										 |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							|  |  |  |         rb.Push<u32>(0); | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-07 15:11:17 +03:00
										 |  |  |     void LoadTimeZoneRule(Kernel::HLERequestContext& ctx) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_WARNING(Service_Time, "(STUBBED) called"); | 
					
						
							| 
									
										
										
										
											2018-05-31 15:33:30 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         ctx.WriteBuffer(&my_time_zone_rule, sizeof(TimeZoneRule)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-07 15:11:17 +03:00
										 |  |  |         IPC::ResponseBuilder rb{ctx, 2}; | 
					
						
							|  |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-31 15:33:30 +03:00
										 |  |  |     void ToCalendarTime(Kernel::HLERequestContext& ctx) { | 
					
						
							|  |  |  |         IPC::RequestParser rp{ctx}; | 
					
						
							|  |  |  |         const u64 posix_time = rp.Pop<u64>(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_WARNING(Service_Time, "(STUBBED) called, posix_time=0x{:016X}", posix_time); | 
					
						
							| 
									
										
										
										
											2018-05-31 15:33:30 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         TimeZoneRule time_zone_rule{}; | 
					
						
							|  |  |  |         auto buffer = ctx.ReadBuffer(); | 
					
						
							|  |  |  |         std::memcpy(&time_zone_rule, buffer.data(), buffer.size()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         CalendarTime calendar_time{2018, 1, 1, 0, 0, 0}; | 
					
						
							|  |  |  |         CalendarAdditionalInfo additional_info{}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         PosixToCalendar(posix_time, calendar_time, additional_info, time_zone_rule); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         IPC::ResponseBuilder rb{ctx, 10}; | 
					
						
							|  |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							|  |  |  |         rb.PushRaw(calendar_time); | 
					
						
							|  |  |  |         rb.PushRaw(additional_info); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  |     void ToCalendarTimeWithMyRule(Kernel::HLERequestContext& ctx) { | 
					
						
							|  |  |  |         IPC::RequestParser rp{ctx}; | 
					
						
							| 
									
										
										
										
											2018-05-31 15:33:30 +03:00
										 |  |  |         const u64 posix_time = rp.Pop<u64>(); | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_WARNING(Service_Time, "(STUBBED) called, posix_time=0x{:016X}", posix_time); | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-18 11:02:05 -06:00
										 |  |  |         CalendarTime calendar_time{2018, 1, 1, 0, 0, 0}; | 
					
						
							|  |  |  |         CalendarAdditionalInfo additional_info{}; | 
					
						
							| 
									
										
										
										
											2018-05-31 15:33:30 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         PosixToCalendar(posix_time, calendar_time, additional_info, my_time_zone_rule); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-23 19:52:18 -05:00
										 |  |  |         IPC::ResponseBuilder rb{ctx, 10}; | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							| 
									
										
										
										
											2018-01-18 11:02:05 -06:00
										 |  |  |         rb.PushRaw(calendar_time); | 
					
						
							|  |  |  |         rb.PushRaw(additional_info); | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-05-31 15:33:30 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     void PosixToCalendar(u64 posix_time, CalendarTime& calendar_time, | 
					
						
							|  |  |  |                          CalendarAdditionalInfo& additional_info, const TimeZoneRule& /*rule*/) { | 
					
						
							|  |  |  |         std::time_t t(posix_time); | 
					
						
							|  |  |  |         std::tm* tm = std::localtime(&t); | 
					
						
							|  |  |  |         if (!tm) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         calendar_time.year = tm->tm_year + 1900; | 
					
						
							|  |  |  |         calendar_time.month = tm->tm_mon + 1; | 
					
						
							|  |  |  |         calendar_time.day = tm->tm_mday; | 
					
						
							|  |  |  |         calendar_time.hour = tm->tm_hour; | 
					
						
							|  |  |  |         calendar_time.minute = tm->tm_min; | 
					
						
							|  |  |  |         calendar_time.second = tm->tm_sec; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         additional_info.day_of_week = tm->tm_wday; | 
					
						
							|  |  |  |         additional_info.day_of_year = tm->tm_yday; | 
					
						
							|  |  |  |         std::memcpy(additional_info.name.data(), "UTC", sizeof("UTC")); | 
					
						
							|  |  |  |         additional_info.utc_offset = 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-18 10:58:29 -06:00
										 |  |  | void Module::Interface::GetStandardUserSystemClock(Kernel::HLERequestContext& ctx) { | 
					
						
							| 
									
										
										
										
											2018-01-23 19:52:18 -05:00
										 |  |  |     IPC::ResponseBuilder rb{ctx, 2, 0, 1}; | 
					
						
							| 
									
										
										
										
											2018-01-22 17:42:11 -05:00
										 |  |  |     rb.Push(RESULT_SUCCESS); | 
					
						
							| 
									
										
										
										
											2018-01-23 19:43:59 -05:00
										 |  |  |     rb.PushIpcInterface<ISystemClock>(); | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |     LOG_DEBUG(Service_Time, "called"); | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-18 10:58:29 -06:00
										 |  |  | void Module::Interface::GetStandardNetworkSystemClock(Kernel::HLERequestContext& ctx) { | 
					
						
							| 
									
										
										
										
											2018-01-23 19:52:18 -05:00
										 |  |  |     IPC::ResponseBuilder rb{ctx, 2, 0, 1}; | 
					
						
							| 
									
										
										
										
											2018-01-22 17:42:11 -05:00
										 |  |  |     rb.Push(RESULT_SUCCESS); | 
					
						
							| 
									
										
										
										
											2018-01-23 19:43:59 -05:00
										 |  |  |     rb.PushIpcInterface<ISystemClock>(); | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |     LOG_DEBUG(Service_Time, "called"); | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-18 10:58:29 -06:00
										 |  |  | void Module::Interface::GetStandardSteadyClock(Kernel::HLERequestContext& ctx) { | 
					
						
							| 
									
										
										
										
											2018-01-23 19:52:18 -05:00
										 |  |  |     IPC::ResponseBuilder rb{ctx, 2, 0, 1}; | 
					
						
							| 
									
										
										
										
											2018-01-22 17:42:11 -05:00
										 |  |  |     rb.Push(RESULT_SUCCESS); | 
					
						
							| 
									
										
										
										
											2018-01-23 19:43:59 -05:00
										 |  |  |     rb.PushIpcInterface<ISteadyClock>(); | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |     LOG_DEBUG(Service_Time, "called"); | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-18 10:58:29 -06:00
										 |  |  | void Module::Interface::GetTimeZoneService(Kernel::HLERequestContext& ctx) { | 
					
						
							| 
									
										
										
										
											2018-01-23 19:52:18 -05:00
										 |  |  |     IPC::ResponseBuilder rb{ctx, 2, 0, 1}; | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  |     rb.Push(RESULT_SUCCESS); | 
					
						
							|  |  |  |     rb.PushIpcInterface<ITimeZoneService>(); | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |     LOG_DEBUG(Service_Time, "called"); | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-21 18:43:05 -07:00
										 |  |  | void Module::Interface::GetStandardLocalSystemClock(Kernel::HLERequestContext& ctx) { | 
					
						
							|  |  |  |     IPC::ResponseBuilder rb{ctx, 2, 0, 1}; | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); | 
					
						
							|  |  |  |     rb.PushIpcInterface<ISystemClock>(); | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |     LOG_DEBUG(Service_Time, "called"); | 
					
						
							| 
									
										
										
										
											2018-02-21 18:43:05 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-18 10:58:29 -06:00
										 |  |  | Module::Interface::Interface(std::shared_ptr<Module> time, const char* name) | 
					
						
							|  |  |  |     : ServiceFramework(name), time(std::move(time)) {} | 
					
						
							| 
									
										
										
										
											2018-01-17 13:33:38 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-14 21:42:23 -05:00
										 |  |  | void InstallInterfaces(SM::ServiceManager& service_manager) { | 
					
						
							| 
									
										
										
										
											2018-01-18 10:58:29 -06:00
										 |  |  |     auto time = std::make_shared<Module>(); | 
					
						
							| 
									
										
										
										
											2018-07-24 02:47:41 -04:00
										 |  |  |     std::make_shared<Time>(time, "time:a")->InstallAsService(service_manager); | 
					
						
							|  |  |  |     std::make_shared<Time>(time, "time:s")->InstallAsService(service_manager); | 
					
						
							|  |  |  |     std::make_shared<Time>(time, "time:u")->InstallAsService(service_manager); | 
					
						
							| 
									
										
										
										
											2018-01-14 21:42:23 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 21:41:44 -04:00
										 |  |  | } // namespace Service::Time
 |