1aba033ba7
* 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
83 lines
2.4 KiB
C#
83 lines
2.4 KiB
C#
using Ryujinx.HLE.HOS.Kernel.Threading;
|
|
|
|
namespace Ryujinx.HLE.HOS.Services.Time.Clock
|
|
{
|
|
class StandardSteadyClockCore : SteadyClockCore
|
|
{
|
|
private TimeSpanType _setupValue;
|
|
private TimeSpanType _testOffset;
|
|
private TimeSpanType _internalOffset;
|
|
private TimeSpanType _cachedRawTimePoint;
|
|
|
|
public StandardSteadyClockCore()
|
|
{
|
|
_setupValue = TimeSpanType.Zero;
|
|
_testOffset = TimeSpanType.Zero;
|
|
_internalOffset = TimeSpanType.Zero;
|
|
_cachedRawTimePoint = TimeSpanType.Zero;
|
|
}
|
|
|
|
public override SteadyClockTimePoint GetTimePoint(KThread thread)
|
|
{
|
|
SteadyClockTimePoint result = new SteadyClockTimePoint
|
|
{
|
|
TimePoint = GetCurrentRawTimePoint(thread).ToSeconds(),
|
|
ClockSourceId = GetClockSourceId()
|
|
};
|
|
|
|
return result;
|
|
}
|
|
|
|
public override TimeSpanType GetTestOffset()
|
|
{
|
|
return _testOffset;
|
|
}
|
|
|
|
public override void SetTestOffset(TimeSpanType testOffset)
|
|
{
|
|
_testOffset = testOffset;
|
|
}
|
|
|
|
public override TimeSpanType GetInternalOffset()
|
|
{
|
|
return _internalOffset;
|
|
}
|
|
|
|
public override void SetInternalOffset(TimeSpanType internalOffset)
|
|
{
|
|
_internalOffset = internalOffset;
|
|
}
|
|
|
|
public override TimeSpanType GetCurrentRawTimePoint(KThread thread)
|
|
{
|
|
TimeSpanType ticksTimeSpan;
|
|
|
|
// As this may be called before the guest code, we support passing a null thread to make this api usable.
|
|
if (thread == null)
|
|
{
|
|
ticksTimeSpan = TimeSpanType.FromSeconds(0);
|
|
}
|
|
else
|
|
{
|
|
ticksTimeSpan = TimeSpanType.FromTicks(thread.Context.CntpctEl0, thread.Context.CntfrqEl0);
|
|
}
|
|
|
|
TimeSpanType rawTimePoint = new TimeSpanType(_setupValue.NanoSeconds + ticksTimeSpan.NanoSeconds);
|
|
|
|
if (rawTimePoint.NanoSeconds < _cachedRawTimePoint.NanoSeconds)
|
|
{
|
|
rawTimePoint.NanoSeconds = _cachedRawTimePoint.NanoSeconds;
|
|
}
|
|
|
|
_cachedRawTimePoint = rawTimePoint;
|
|
|
|
return rawTimePoint;
|
|
}
|
|
|
|
public void SetSetupValue(TimeSpanType setupValue)
|
|
{
|
|
_setupValue = setupValue;
|
|
}
|
|
}
|
|
}
|