2018-01-15 02:42:23 +00:00
|
|
|
// Copyright 2018 yuzu emulator team
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-01-17 19:33:38 +00:00
|
|
|
#include <chrono>
|
|
|
|
#include "common/logging/log.h"
|
2018-01-25 04:52:21 +00:00
|
|
|
#include "core/core_timing.h"
|
2018-01-17 19:33:38 +00:00
|
|
|
#include "core/hle/ipc_helpers.h"
|
|
|
|
#include "core/hle/kernel/client_port.h"
|
|
|
|
#include "core/hle/kernel/client_session.h"
|
2018-01-15 02:42:23 +00:00
|
|
|
#include "core/hle/service/time/time.h"
|
2018-01-18 16:58:29 +00:00
|
|
|
#include "core/hle/service/time/time_s.h"
|
|
|
|
#include "core/hle/service/time/time_u.h"
|
2018-01-15 02:42:23 +00:00
|
|
|
|
|
|
|
namespace Service {
|
|
|
|
namespace Time {
|
|
|
|
|
2018-01-17 19:33:38 +00:00
|
|
|
class ISystemClock final : public ServiceFramework<ISystemClock> {
|
|
|
|
public:
|
|
|
|
ISystemClock() : ServiceFramework("ISystemClock") {
|
|
|
|
static const FunctionInfo functions[] = {
|
|
|
|
{0, &ISystemClock::GetCurrentTime, "GetCurrentTime"},
|
2018-01-24 03:02:14 +00:00
|
|
|
{2, &ISystemClock::GetSystemClockContext, "GetSystemClockContext"}};
|
2018-01-17 19:33:38 +00:00
|
|
|
RegisterHandlers(functions);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
void GetCurrentTime(Kernel::HLERequestContext& ctx) {
|
2018-01-24 14:54:47 +00:00
|
|
|
const s64 time_since_epoch{std::chrono::duration_cast<std::chrono::seconds>(
|
2018-01-17 19:33:38 +00:00
|
|
|
std::chrono::system_clock::now().time_since_epoch())
|
|
|
|
.count()};
|
2018-01-24 03:02:14 +00:00
|
|
|
LOG_DEBUG(Service, "called");
|
2018-01-24 00:52:18 +00:00
|
|
|
IPC::ResponseBuilder rb{ctx, 4};
|
2018-01-17 19:33:38 +00:00
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push<u64>(time_since_epoch);
|
2018-01-24 03:02:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GetSystemClockContext(Kernel::HLERequestContext& ctx) {
|
|
|
|
LOG_WARNING(Service, "(STUBBED) called");
|
|
|
|
SystemClockContext system_clock_ontext{};
|
|
|
|
IPC::ResponseBuilder rb{ctx, (sizeof(SystemClockContext) / 4) + 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.PushRaw(system_clock_ontext);
|
2018-01-17 19:33:38 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class ISteadyClock final : public ServiceFramework<ISteadyClock> {
|
|
|
|
public:
|
2018-01-25 04:52:21 +00:00
|
|
|
ISteadyClock() : ServiceFramework("ISteadyClock") {
|
|
|
|
static const FunctionInfo functions[] = {
|
|
|
|
{0, &ISteadyClock::GetCurrentTimePoint, "GetCurrentTimePoint"},
|
|
|
|
};
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
void GetCurrentTimePoint(Kernel::HLERequestContext& ctx) {
|
|
|
|
LOG_DEBUG(Service, "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);
|
|
|
|
}
|
2018-01-17 19:33:38 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class ITimeZoneService final : public ServiceFramework<ITimeZoneService> {
|
|
|
|
public:
|
|
|
|
ITimeZoneService() : ServiceFramework("ITimeZoneService") {
|
|
|
|
static const FunctionInfo functions[] = {
|
|
|
|
{0, &ITimeZoneService::GetDeviceLocationName, "GetDeviceLocationName"},
|
2018-01-18 05:57:11 +00:00
|
|
|
{2, &ITimeZoneService::GetTotalLocationNameCount, "GetTotalLocationNameCount"},
|
2018-01-17 19:33:38 +00:00
|
|
|
{101, &ITimeZoneService::ToCalendarTimeWithMyRule, "ToCalendarTimeWithMyRule"},
|
|
|
|
};
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
void GetDeviceLocationName(Kernel::HLERequestContext& ctx) {
|
|
|
|
LOG_WARNING(Service, "(STUBBED) called");
|
2018-01-18 05:57:11 +00:00
|
|
|
LocationName location_name{};
|
2018-01-24 00:52:18 +00:00
|
|
|
IPC::ResponseBuilder rb{ctx, (sizeof(LocationName) / 4) + 2};
|
2018-01-17 19:33:38 +00:00
|
|
|
rb.Push(RESULT_SUCCESS);
|
2018-01-18 05:57:11 +00:00
|
|
|
rb.PushRaw(location_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GetTotalLocationNameCount(Kernel::HLERequestContext& ctx) {
|
|
|
|
LOG_WARNING(Service, "(STUBBED) called");
|
2018-01-24 00:52:18 +00:00
|
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
2018-01-18 05:57:11 +00:00
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push<u32>(0);
|
2018-01-17 19:33:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ToCalendarTimeWithMyRule(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
2018-01-18 17:02:05 +00:00
|
|
|
u64 posix_time = rp.Pop<u64>();
|
2018-01-17 19:33:38 +00:00
|
|
|
|
2018-01-18 17:02:05 +00:00
|
|
|
LOG_WARNING(Service, "(STUBBED) called, posix_time=0x%016llX", posix_time);
|
2018-01-17 19:33:38 +00:00
|
|
|
|
2018-01-18 17:02:05 +00:00
|
|
|
CalendarTime calendar_time{2018, 1, 1, 0, 0, 0};
|
|
|
|
CalendarAdditionalInfo additional_info{};
|
2018-01-24 00:52:18 +00:00
|
|
|
IPC::ResponseBuilder rb{ctx, 10};
|
2018-01-17 19:33:38 +00:00
|
|
|
rb.Push(RESULT_SUCCESS);
|
2018-01-18 17:02:05 +00:00
|
|
|
rb.PushRaw(calendar_time);
|
|
|
|
rb.PushRaw(additional_info);
|
2018-01-17 19:33:38 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-01-18 16:58:29 +00:00
|
|
|
void Module::Interface::GetStandardUserSystemClock(Kernel::HLERequestContext& ctx) {
|
2018-01-24 00:52:18 +00:00
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
2018-01-22 22:42:11 +00:00
|
|
|
rb.Push(RESULT_SUCCESS);
|
2018-01-24 00:43:59 +00:00
|
|
|
rb.PushIpcInterface<ISystemClock>();
|
|
|
|
LOG_DEBUG(Service, "called");
|
2018-01-17 19:33:38 +00:00
|
|
|
}
|
|
|
|
|
2018-01-18 16:58:29 +00:00
|
|
|
void Module::Interface::GetStandardNetworkSystemClock(Kernel::HLERequestContext& ctx) {
|
2018-01-24 00:52:18 +00:00
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
2018-01-22 22:42:11 +00:00
|
|
|
rb.Push(RESULT_SUCCESS);
|
2018-01-24 00:43:59 +00:00
|
|
|
rb.PushIpcInterface<ISystemClock>();
|
|
|
|
LOG_DEBUG(Service, "called");
|
2018-01-17 19:33:38 +00:00
|
|
|
}
|
|
|
|
|
2018-01-18 16:58:29 +00:00
|
|
|
void Module::Interface::GetStandardSteadyClock(Kernel::HLERequestContext& ctx) {
|
2018-01-24 00:52:18 +00:00
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
2018-01-22 22:42:11 +00:00
|
|
|
rb.Push(RESULT_SUCCESS);
|
2018-01-24 00:43:59 +00:00
|
|
|
rb.PushIpcInterface<ISteadyClock>();
|
|
|
|
LOG_DEBUG(Service, "called");
|
2018-01-17 19:33:38 +00:00
|
|
|
}
|
|
|
|
|
2018-01-18 16:58:29 +00:00
|
|
|
void Module::Interface::GetTimeZoneService(Kernel::HLERequestContext& ctx) {
|
2018-01-24 00:52:18 +00:00
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
2018-01-17 19:33:38 +00:00
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.PushIpcInterface<ITimeZoneService>();
|
|
|
|
LOG_DEBUG(Service, "called");
|
|
|
|
}
|
|
|
|
|
2018-01-18 16:58:29 +00:00
|
|
|
Module::Interface::Interface(std::shared_ptr<Module> time, const char* name)
|
|
|
|
: ServiceFramework(name), time(std::move(time)) {}
|
2018-01-17 19:33:38 +00:00
|
|
|
|
2018-01-15 02:42:23 +00:00
|
|
|
void InstallInterfaces(SM::ServiceManager& service_manager) {
|
2018-01-18 16:58:29 +00:00
|
|
|
auto time = std::make_shared<Module>();
|
|
|
|
std::make_shared<TIME_S>(time)->InstallAsService(service_manager);
|
|
|
|
std::make_shared<TIME_U>(time)->InstallAsService(service_manager);
|
2018-01-15 02:42:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Time
|
|
|
|
} // namespace Service
|