Update time implementation to 9.0.0 (#783)
* Fix 9.0.0 related services bindings This was wrong because of a mistake on switchbrew. * Fix wronog cmdid for ISteadyClock::GetTestOffset/SetTestOffset * Update ClockCore logics to 9.0.0 Also apply 9.0.0 permissions and comment time:u, and time:a (as those are going to be moved) * Move every clocks instances + timezone to a global manager * Start implementing time:m Also prepare the skeleton of the shared memory * Implement SystemClockContextUpdateCallback and co * Update StaticService to 9.0.0 * Update ISystemClock to 9.0.0 * Rename IStaticService and add glue's IStaticService * Implement psc's ITimeZoneService * Integrate psc layer into glue for TimeZoneService * Rename TimeZoneManagerForPsc => TimeZoneManager * Use correct TimeZoneService interface for both StaticService implementations * Accurately implement time shared memory operations * Fix two critical flaws in TimeZone logic The first one was the month range being different fron Nintendo one (0-11 instead of 1-12) The other flaw was a bad incrementation order during days & months computation. * Follow Nintendo's abort logic for TimeManager * Avoid crashing when timezone sysarchive isn't present * Update Readme * Address comments * Correctly align fields in ISystemClock * Fix code style and some typos * Improve timezone system archive warning/error messages * Rearrange using definitions in Horizon.cs * Address comments
This commit is contained in:
parent
16869402bf
commit
1aba033ba7
37 changed files with 2202 additions and 716 deletions
182
Ryujinx.HLE/HOS/Services/Time/IStaticServiceForGlue.cs
Normal file
182
Ryujinx.HLE/HOS/Services/Time/IStaticServiceForGlue.cs
Normal file
|
@ -0,0 +1,182 @@
|
|||
using Ryujinx.Common;
|
||||
using Ryujinx.HLE.HOS.Services.Pcv.Bpc;
|
||||
using Ryujinx.HLE.HOS.Services.Settings;
|
||||
using Ryujinx.HLE.HOS.Services.Time.Clock;
|
||||
using Ryujinx.HLE.HOS.Services.Time.StaticService;
|
||||
using System;
|
||||
|
||||
namespace Ryujinx.HLE.HOS.Services.Time
|
||||
{
|
||||
[Service("time:a", TimePermissions.Admin)]
|
||||
[Service("time:r", TimePermissions.Repair)]
|
||||
[Service("time:u", TimePermissions.User)]
|
||||
class IStaticServiceForGlue : IpcService
|
||||
{
|
||||
private IStaticServiceForPsc _inner;
|
||||
private TimePermissions _permissions;
|
||||
|
||||
public IStaticServiceForGlue(ServiceCtx context, TimePermissions permissions)
|
||||
{
|
||||
_permissions = permissions;
|
||||
_inner = new IStaticServiceForPsc(context, permissions);
|
||||
}
|
||||
|
||||
[Command(0)]
|
||||
// GetStandardUserSystemClock() -> object<nn::timesrv::detail::service::ISystemClock>
|
||||
public ResultCode GetStandardUserSystemClock(ServiceCtx context)
|
||||
{
|
||||
return _inner.GetStandardUserSystemClock(context);
|
||||
}
|
||||
|
||||
[Command(1)]
|
||||
// GetStandardNetworkSystemClock() -> object<nn::timesrv::detail::service::ISystemClock>
|
||||
public ResultCode GetStandardNetworkSystemClock(ServiceCtx context)
|
||||
{
|
||||
return _inner.GetStandardNetworkSystemClock(context);
|
||||
}
|
||||
|
||||
[Command(2)]
|
||||
// GetStandardSteadyClock() -> object<nn::timesrv::detail::service::ISteadyClock>
|
||||
public ResultCode GetStandardSteadyClock(ServiceCtx context)
|
||||
{
|
||||
return _inner.GetStandardSteadyClock(context);
|
||||
}
|
||||
|
||||
[Command(3)]
|
||||
// GetTimeZoneService() -> object<nn::timesrv::detail::service::ITimeZoneService>
|
||||
public ResultCode GetTimeZoneService(ServiceCtx context)
|
||||
{
|
||||
MakeObject(context, new ITimeZoneServiceForGlue(TimeManager.Instance.TimeZone, (_permissions & TimePermissions.TimeZoneWritableMask) != 0));
|
||||
|
||||
return ResultCode.Success;
|
||||
}
|
||||
|
||||
[Command(4)]
|
||||
// GetStandardLocalSystemClock() -> object<nn::timesrv::detail::service::ISystemClock>
|
||||
public ResultCode GetStandardLocalSystemClock(ServiceCtx context)
|
||||
{
|
||||
return _inner.GetStandardLocalSystemClock(context);
|
||||
}
|
||||
|
||||
[Command(5)] // 4.0.0+
|
||||
// GetEphemeralNetworkSystemClock() -> object<nn::timesrv::detail::service::ISystemClock>
|
||||
public ResultCode GetEphemeralNetworkSystemClock(ServiceCtx context)
|
||||
{
|
||||
return _inner.GetEphemeralNetworkSystemClock(context);
|
||||
}
|
||||
|
||||
[Command(20)] // 6.0.0+
|
||||
// GetSharedMemoryNativeHandle() -> handle<copy>
|
||||
public ResultCode GetSharedMemoryNativeHandle(ServiceCtx context)
|
||||
{
|
||||
return _inner.GetSharedMemoryNativeHandle(context);
|
||||
}
|
||||
|
||||
[Command(50)] // 4.0.0+
|
||||
// SetStandardSteadyClockInternalOffset(nn::TimeSpanType internal_offset)
|
||||
public ResultCode SetStandardSteadyClockInternalOffset(ServiceCtx context)
|
||||
{
|
||||
if ((_permissions & TimePermissions.BypassUninitialized) == 0)
|
||||
{
|
||||
return ResultCode.PermissionDenied;
|
||||
}
|
||||
|
||||
TimeSpanType internalOffset = context.RequestData.ReadStruct<TimeSpanType>();
|
||||
|
||||
// TODO: set:sys SetExternalSteadyClockInternalOffset(internalOffset.ToSeconds())
|
||||
|
||||
return ResultCode.Success;
|
||||
}
|
||||
|
||||
[Command(51)] // 9.0.0+
|
||||
// GetStandardSteadyClockRtcValue() -> u64
|
||||
public ResultCode GetStandardSteadyClockRtcValue(ServiceCtx context)
|
||||
{
|
||||
ResultCode result = (ResultCode)IRtcManager.GetExternalRtcValue(out ulong rtcValue);
|
||||
|
||||
if (result == ResultCode.Success)
|
||||
{
|
||||
context.ResponseData.Write(rtcValue);
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
[Command(100)]
|
||||
// IsStandardUserSystemClockAutomaticCorrectionEnabled() -> bool
|
||||
public ResultCode IsStandardUserSystemClockAutomaticCorrectionEnabled(ServiceCtx context)
|
||||
{
|
||||
return _inner.IsStandardUserSystemClockAutomaticCorrectionEnabled(context);
|
||||
}
|
||||
|
||||
[Command(101)]
|
||||
// SetStandardUserSystemClockAutomaticCorrectionEnabled(b8)
|
||||
public ResultCode SetStandardUserSystemClockAutomaticCorrectionEnabled(ServiceCtx context)
|
||||
{
|
||||
return _inner.SetStandardUserSystemClockAutomaticCorrectionEnabled(context);
|
||||
}
|
||||
|
||||
[Command(102)] // 5.0.0+
|
||||
// GetStandardUserSystemClockInitialYear() -> u32
|
||||
public ResultCode GetStandardUserSystemClockInitialYear(ServiceCtx context)
|
||||
{
|
||||
if (!NxSettings.Settings.TryGetValue("time!standard_user_clock_initial_year", out object standardUserSystemClockInitialYear))
|
||||
{
|
||||
throw new InvalidOperationException("standard_user_clock_initial_year isn't defined in system settings!");
|
||||
}
|
||||
|
||||
context.ResponseData.Write((int)standardUserSystemClockInitialYear);
|
||||
|
||||
return ResultCode.Success;
|
||||
}
|
||||
|
||||
[Command(200)] // 3.0.0+
|
||||
// IsStandardNetworkSystemClockAccuracySufficient() -> bool
|
||||
public ResultCode IsStandardNetworkSystemClockAccuracySufficient(ServiceCtx context)
|
||||
{
|
||||
return _inner.IsStandardNetworkSystemClockAccuracySufficient(context);
|
||||
}
|
||||
|
||||
[Command(201)] // 6.0.0+
|
||||
// GetStandardUserSystemClockAutomaticCorrectionUpdatedTime() -> nn::time::SteadyClockTimePoint
|
||||
public ResultCode GetStandardUserSystemClockAutomaticCorrectionUpdatedTime(ServiceCtx context)
|
||||
{
|
||||
return _inner.GetStandardUserSystemClockAutomaticCorrectionUpdatedTime(context);
|
||||
}
|
||||
|
||||
[Command(300)] // 4.0.0+
|
||||
// CalculateMonotonicSystemClockBaseTimePoint(nn::time::SystemClockContext) -> s64
|
||||
public ResultCode CalculateMonotonicSystemClockBaseTimePoint(ServiceCtx context)
|
||||
{
|
||||
return _inner.CalculateMonotonicSystemClockBaseTimePoint(context);
|
||||
}
|
||||
|
||||
[Command(400)] // 4.0.0+
|
||||
// GetClockSnapshot(u8) -> buffer<nn::time::sf::ClockSnapshot, 0x1a>
|
||||
public ResultCode GetClockSnapshot(ServiceCtx context)
|
||||
{
|
||||
return _inner.GetClockSnapshot(context);
|
||||
}
|
||||
|
||||
[Command(401)] // 4.0.0+
|
||||
// GetClockSnapshotFromSystemClockContext(u8, nn::time::SystemClockContext, nn::time::SystemClockContext) -> buffer<nn::time::sf::ClockSnapshot, 0x1a>
|
||||
public ResultCode GetClockSnapshotFromSystemClockContext(ServiceCtx context)
|
||||
{
|
||||
return _inner.GetClockSnapshotFromSystemClockContext(context);
|
||||
}
|
||||
|
||||
[Command(500)] // 4.0.0+
|
||||
// CalculateStandardUserSystemClockDifferenceByUser(buffer<nn::time::sf::ClockSnapshot, 0x19>, buffer<nn::time::sf::ClockSnapshot, 0x19>) -> nn::TimeSpanType
|
||||
public ResultCode CalculateStandardUserSystemClockDifferenceByUser(ServiceCtx context)
|
||||
{
|
||||
return _inner.CalculateStandardUserSystemClockDifferenceByUser(context);
|
||||
}
|
||||
|
||||
[Command(501)] // 4.0.0+
|
||||
// CalculateSpanBetween(buffer<nn::time::sf::ClockSnapshot, 0x19>, buffer<nn::time::sf::ClockSnapshot, 0x19>) -> nn::TimeSpanType
|
||||
public ResultCode CalculateSpanBetween(ServiceCtx context)
|
||||
{
|
||||
return _inner.CalculateSpanBetween(context);
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue