forked from eden-emu/eden
		
	
		
			
				
	
	
		
			159 lines
		
	
	
	
		
			5.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			159 lines
		
	
	
	
		
			5.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2018 yuzu emulator team
 | |
| // Licensed under GPLv2 or any later version
 | |
| // Refer to the license.txt file included.
 | |
| 
 | |
| #include <chrono>
 | |
| #include "common/logging/log.h"
 | |
| #include "core/core_timing.h"
 | |
| #include "core/hle/ipc_helpers.h"
 | |
| #include "core/hle/kernel/client_port.h"
 | |
| #include "core/hle/kernel/client_session.h"
 | |
| #include "core/hle/service/time/time.h"
 | |
| #include "core/hle/service/time/time_s.h"
 | |
| #include "core/hle/service/time/time_u.h"
 | |
| 
 | |
| namespace Service {
 | |
| namespace Time {
 | |
| 
 | |
| class ISystemClock final : public ServiceFramework<ISystemClock> {
 | |
| public:
 | |
|     ISystemClock() : ServiceFramework("ISystemClock") {
 | |
|         static const FunctionInfo functions[] = {
 | |
|             {0, &ISystemClock::GetCurrentTime, "GetCurrentTime"},
 | |
|             {2, &ISystemClock::GetSystemClockContext, "GetSystemClockContext"}};
 | |
|         RegisterHandlers(functions);
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     void GetCurrentTime(Kernel::HLERequestContext& ctx) {
 | |
|         const s64 time_since_epoch{std::chrono::duration_cast<std::chrono::seconds>(
 | |
|                                        std::chrono::system_clock::now().time_since_epoch())
 | |
|                                        .count()};
 | |
|         LOG_DEBUG(Service_Time, "called");
 | |
|         IPC::ResponseBuilder rb{ctx, 4};
 | |
|         rb.Push(RESULT_SUCCESS);
 | |
|         rb.Push<u64>(time_since_epoch);
 | |
|     }
 | |
| 
 | |
|     void GetSystemClockContext(Kernel::HLERequestContext& ctx) {
 | |
|         LOG_WARNING(Service_Time, "(STUBBED) called");
 | |
|         SystemClockContext system_clock_ontext{};
 | |
|         IPC::ResponseBuilder rb{ctx, (sizeof(SystemClockContext) / 4) + 2};
 | |
|         rb.Push(RESULT_SUCCESS);
 | |
|         rb.PushRaw(system_clock_ontext);
 | |
|     }
 | |
| };
 | |
| 
 | |
| class ISteadyClock final : public ServiceFramework<ISteadyClock> {
 | |
| public:
 | |
|     ISteadyClock() : ServiceFramework("ISteadyClock") {
 | |
|         static const FunctionInfo functions[] = {
 | |
|             {0, &ISteadyClock::GetCurrentTimePoint, "GetCurrentTimePoint"},
 | |
|         };
 | |
|         RegisterHandlers(functions);
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     void GetCurrentTimePoint(Kernel::HLERequestContext& ctx) {
 | |
|         LOG_DEBUG(Service_Time, "called");
 | |
|         SteadyClockTimePoint steady_clock_time_point{cyclesToMs(CoreTiming::GetTicks()) / 1000};
 | |
|         IPC::ResponseBuilder rb{ctx, (sizeof(SteadyClockTimePoint) / 4) + 2};
 | |
|         rb.Push(RESULT_SUCCESS);
 | |
|         rb.PushRaw(steady_clock_time_point);
 | |
|     }
 | |
| };
 | |
| 
 | |
| class ITimeZoneService final : public ServiceFramework<ITimeZoneService> {
 | |
| public:
 | |
|     ITimeZoneService() : ServiceFramework("ITimeZoneService") {
 | |
|         static const FunctionInfo functions[] = {
 | |
|             {0, &ITimeZoneService::GetDeviceLocationName, "GetDeviceLocationName"},
 | |
|             {1, nullptr, "SetDeviceLocationName"},
 | |
|             {2, &ITimeZoneService::GetTotalLocationNameCount, "GetTotalLocationNameCount"},
 | |
|             {3, nullptr, "LoadLocationNameList"},
 | |
|             {4, &ITimeZoneService::LoadTimeZoneRule, "LoadTimeZoneRule"},
 | |
|             {5, nullptr, "GetTimeZoneRuleVersion"},
 | |
|             {100, nullptr, "ToCalendarTime"},
 | |
|             {101, &ITimeZoneService::ToCalendarTimeWithMyRule, "ToCalendarTimeWithMyRule"},
 | |
|             {200, nullptr, "ToPosixTime"},
 | |
|             {201, nullptr, "ToPosixTimeWithMyRule"},
 | |
|         };
 | |
|         RegisterHandlers(functions);
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     void GetDeviceLocationName(Kernel::HLERequestContext& ctx) {
 | |
|         LOG_WARNING(Service_Time, "(STUBBED) called");
 | |
|         LocationName location_name{};
 | |
|         IPC::ResponseBuilder rb{ctx, (sizeof(LocationName) / 4) + 2};
 | |
|         rb.Push(RESULT_SUCCESS);
 | |
|         rb.PushRaw(location_name);
 | |
|     }
 | |
| 
 | |
|     void GetTotalLocationNameCount(Kernel::HLERequestContext& ctx) {
 | |
|         LOG_WARNING(Service_Time, "(STUBBED) called");
 | |
|         IPC::ResponseBuilder rb{ctx, 3};
 | |
|         rb.Push(RESULT_SUCCESS);
 | |
|         rb.Push<u32>(0);
 | |
|     }
 | |
| 
 | |
|     void LoadTimeZoneRule(Kernel::HLERequestContext& ctx) {
 | |
|         LOG_WARNING(Service_Time, "(STUBBED) called");
 | |
|         IPC::ResponseBuilder rb{ctx, 2};
 | |
|         rb.Push(RESULT_SUCCESS);
 | |
|     }
 | |
| 
 | |
|     void ToCalendarTimeWithMyRule(Kernel::HLERequestContext& ctx) {
 | |
|         IPC::RequestParser rp{ctx};
 | |
|         u64 posix_time = rp.Pop<u64>();
 | |
| 
 | |
|         LOG_WARNING(Service_Time, "(STUBBED) called, posix_time=0x%016llX", posix_time);
 | |
| 
 | |
|         CalendarTime calendar_time{2018, 1, 1, 0, 0, 0};
 | |
|         CalendarAdditionalInfo additional_info{};
 | |
|         IPC::ResponseBuilder rb{ctx, 10};
 | |
|         rb.Push(RESULT_SUCCESS);
 | |
|         rb.PushRaw(calendar_time);
 | |
|         rb.PushRaw(additional_info);
 | |
|     }
 | |
| };
 | |
| 
 | |
| void Module::Interface::GetStandardUserSystemClock(Kernel::HLERequestContext& ctx) {
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|     rb.Push(RESULT_SUCCESS);
 | |
|     rb.PushIpcInterface<ISystemClock>();
 | |
|     LOG_DEBUG(Service_Time, "called");
 | |
| }
 | |
| 
 | |
| void Module::Interface::GetStandardNetworkSystemClock(Kernel::HLERequestContext& ctx) {
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|     rb.Push(RESULT_SUCCESS);
 | |
|     rb.PushIpcInterface<ISystemClock>();
 | |
|     LOG_DEBUG(Service_Time, "called");
 | |
| }
 | |
| 
 | |
| void Module::Interface::GetStandardSteadyClock(Kernel::HLERequestContext& ctx) {
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|     rb.Push(RESULT_SUCCESS);
 | |
|     rb.PushIpcInterface<ISteadyClock>();
 | |
|     LOG_DEBUG(Service_Time, "called");
 | |
| }
 | |
| 
 | |
| void Module::Interface::GetTimeZoneService(Kernel::HLERequestContext& ctx) {
 | |
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1};
 | |
|     rb.Push(RESULT_SUCCESS);
 | |
|     rb.PushIpcInterface<ITimeZoneService>();
 | |
|     LOG_DEBUG(Service_Time, "called");
 | |
| }
 | |
| 
 | |
| Module::Interface::Interface(std::shared_ptr<Module> time, const char* name)
 | |
|     : ServiceFramework(name), time(std::move(time)) {}
 | |
| 
 | |
| void InstallInterfaces(SM::ServiceManager& service_manager) {
 | |
|     auto time = std::make_shared<Module>();
 | |
|     std::make_shared<TIME_S>(time)->InstallAsService(service_manager);
 | |
|     std::make_shared<TIME_U>(time)->InstallAsService(service_manager);
 | |
| }
 | |
| 
 | |
| } // namespace Time
 | |
| } // namespace Service
 | 
