121 lines
3.5 KiB
C#
121 lines
3.5 KiB
C#
|
using System;
|
|||
|
using NAudio.CoreAudioApi.Interfaces;
|
|||
|
using System.Diagnostics;
|
|||
|
using System.Runtime.InteropServices;
|
|||
|
|
|||
|
namespace NAudio.CoreAudioApi
|
|||
|
{
|
|||
|
/// <summary>
|
|||
|
/// Audio Clock Client
|
|||
|
/// </summary>
|
|||
|
public class AudioClockClient : IDisposable
|
|||
|
{
|
|||
|
IAudioClock audioClockClientInterface;
|
|||
|
|
|||
|
internal AudioClockClient(IAudioClock audioClockClientInterface)
|
|||
|
{
|
|||
|
this.audioClockClientInterface = audioClockClientInterface;
|
|||
|
|
|||
|
//Stopwatch.GetTimestamp();
|
|||
|
//Stopwatch.Frequency
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Characteristics
|
|||
|
/// </summary>
|
|||
|
public int Characteristics
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
Marshal.ThrowExceptionForHR(audioClockClientInterface.GetCharacteristics(out var characteristics));
|
|||
|
return (int)characteristics;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Frequency
|
|||
|
/// </summary>
|
|||
|
public ulong Frequency
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
Marshal.ThrowExceptionForHR(audioClockClientInterface.GetFrequency(out var freq));
|
|||
|
return freq;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Get Position
|
|||
|
/// </summary>
|
|||
|
public bool GetPosition(out ulong position, out ulong qpcPosition)
|
|||
|
{
|
|||
|
var hr = audioClockClientInterface.GetPosition(out position, out qpcPosition);
|
|||
|
if (hr == -1) return false;
|
|||
|
Marshal.ThrowExceptionForHR(hr);
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Adjusted Position
|
|||
|
/// </summary>
|
|||
|
public ulong AdjustedPosition
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
ulong pos, qpos;
|
|||
|
int cnt = 0;
|
|||
|
while (!GetPosition(out pos, out qpos))
|
|||
|
{
|
|||
|
if (++cnt == 5)
|
|||
|
{
|
|||
|
// we've tried too many times, so now we have to just run with what we have...
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (Stopwatch.IsHighResolution)
|
|||
|
{
|
|||
|
// cool, we can adjust our position appropriately
|
|||
|
|
|||
|
// get the current qpc count (in ticks)
|
|||
|
var qposNow = (ulong)((Stopwatch.GetTimestamp() * 10000000M) / Stopwatch.Frequency);
|
|||
|
|
|||
|
// find out how many ticks have passed since the device reported the position
|
|||
|
var qposDiff = qposNow - qpos;
|
|||
|
|
|||
|
// find out how many device position units (usually bytes) would have played in that time span
|
|||
|
var posDiff = (qposDiff * Frequency) / TimeSpan.TicksPerSecond;
|
|||
|
|
|||
|
// add it to the position
|
|||
|
pos += posDiff;
|
|||
|
}
|
|||
|
return pos;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Can Adjust Position
|
|||
|
/// </summary>
|
|||
|
public bool CanAdjustPosition => Stopwatch.IsHighResolution;
|
|||
|
|
|||
|
#region IDisposable Members
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Dispose
|
|||
|
/// </summary>
|
|||
|
public void Dispose()
|
|||
|
{
|
|||
|
if (audioClockClientInterface != null)
|
|||
|
{
|
|||
|
// althugh GC would do this for us, we want it done now
|
|||
|
// to let us reopen WASAPI
|
|||
|
Marshal.ReleaseComObject(audioClockClientInterface);
|
|||
|
audioClockClientInterface = null;
|
|||
|
GC.SuppressFinalize(this);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
#endregion
|
|||
|
}
|
|||
|
}
|