commit
28108ef92a
|
@ -369,6 +369,8 @@ namespace Emby.Server.Implementations.Library
|
||||||
{
|
{
|
||||||
await _liveStreamSemaphore.WaitAsync(cancellationToken).ConfigureAwait(false);
|
await _liveStreamSemaphore.WaitAsync(cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
enableAutoClose = false;
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var tuple = GetProvider(request.OpenToken);
|
var tuple = GetProvider(request.OpenToken);
|
||||||
|
|
|
@ -510,7 +510,7 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts.HdHomerun
|
||||||
// The UDP method is not working reliably on OSX, and on BSD it hasn't been tested yet
|
// The UDP method is not working reliably on OSX, and on BSD it hasn't been tested yet
|
||||||
var enableHttpStream = _environment.OperatingSystem == OperatingSystem.OSX ||
|
var enableHttpStream = _environment.OperatingSystem == OperatingSystem.OSX ||
|
||||||
_environment.OperatingSystem == OperatingSystem.BSD;
|
_environment.OperatingSystem == OperatingSystem.BSD;
|
||||||
|
enableHttpStream = true;
|
||||||
if (enableHttpStream)
|
if (enableHttpStream)
|
||||||
{
|
{
|
||||||
mediaSource.Protocol = MediaProtocol.Http;
|
mediaSource.Protocol = MediaProtocol.Http;
|
||||||
|
|
|
@ -130,7 +130,7 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts.HdHomerun
|
||||||
onStarted = () => openTaskCompletionSource.TrySetResult(true);
|
onStarted = () => openTaskCompletionSource.TrySetResult(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
await _multicastStream.CopyUntilCancelled(udpClient, onStarted, cancellationToken).ConfigureAwait(false);
|
await _multicastStream.CopyUntilCancelled(new UdpClientStream(udpClient), onStarted, cancellationToken).ConfigureAwait(false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (OperationCanceledException ex)
|
catch (OperationCanceledException ex)
|
||||||
|
@ -167,4 +167,127 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts.HdHomerun
|
||||||
return _multicastStream.CopyToAsync(stream);
|
return _multicastStream.CopyToAsync(stream);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// This handles the ReadAsync function only of a Stream object
|
||||||
|
// This is used to wrap a UDP socket into a stream for MulticastStream which only uses ReadAsync
|
||||||
|
public class UdpClientStream : Stream
|
||||||
|
{
|
||||||
|
private static int RtpHeaderBytes = 12;
|
||||||
|
private static int PacketSize = 1316;
|
||||||
|
private readonly ISocket _udpClient;
|
||||||
|
bool disposed;
|
||||||
|
|
||||||
|
public UdpClientStream(ISocket udpClient) : base()
|
||||||
|
{
|
||||||
|
_udpClient = udpClient;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override async Task<int> ReadAsync(byte[] buffer, int offset, int count, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
if (buffer == null)
|
||||||
|
throw new ArgumentNullException("buffer");
|
||||||
|
|
||||||
|
if (offset + count < 0)
|
||||||
|
throw new ArgumentOutOfRangeException("offset + count must not be negative", "offset+count");
|
||||||
|
|
||||||
|
if (offset + count > buffer.Length)
|
||||||
|
throw new ArgumentException("offset + count must not be greater than the length of buffer", "offset+count");
|
||||||
|
|
||||||
|
if (disposed)
|
||||||
|
throw new ObjectDisposedException(typeof(UdpClientStream).ToString());
|
||||||
|
|
||||||
|
// This will always receive a 1328 packet size (PacketSize + RtpHeaderSize)
|
||||||
|
// The RTP header will be stripped so see how many reads we need to make to fill the buffer.
|
||||||
|
int numReads = count / PacketSize;
|
||||||
|
int totalBytesRead = 0;
|
||||||
|
|
||||||
|
for (int i = 0; i < numReads; ++i)
|
||||||
|
{
|
||||||
|
var data = await _udpClient.ReceiveAsync(cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
var bytesRead = data.ReceivedBytes - RtpHeaderBytes;
|
||||||
|
|
||||||
|
// remove rtp header
|
||||||
|
Buffer.BlockCopy(data.Buffer, RtpHeaderBytes, buffer, offset, bytesRead);
|
||||||
|
offset += bytesRead;
|
||||||
|
totalBytesRead += bytesRead;
|
||||||
|
}
|
||||||
|
return totalBytesRead;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Dispose(bool disposing)
|
||||||
|
{
|
||||||
|
disposed = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override bool CanRead
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override bool CanSeek
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override bool CanWrite
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override long Length
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override long Position
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
|
||||||
|
set
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void Flush()
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
|
||||||
|
public override int Read(byte[] buffer, int offset, int count)
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
|
||||||
|
public override long Seek(long offset, SeekOrigin origin)
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void SetLength(long value)
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void Write(byte[] buffer, int offset, int count)
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -35,13 +35,21 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts
|
||||||
|
|
||||||
if (bytesRead > 0)
|
if (bytesRead > 0)
|
||||||
{
|
{
|
||||||
byte[] copy = new byte[bytesRead];
|
|
||||||
Buffer.BlockCopy(buffer, 0, copy, 0, bytesRead);
|
|
||||||
|
|
||||||
var allStreams = _outputStreams.ToList();
|
var allStreams = _outputStreams.ToList();
|
||||||
foreach (var stream in allStreams)
|
|
||||||
|
if (allStreams.Count == 1)
|
||||||
{
|
{
|
||||||
stream.Value.Queue(copy, 0, copy.Length);
|
await allStreams[0].Value.WriteAsync(buffer, 0, bytesRead).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
byte[] copy = new byte[bytesRead];
|
||||||
|
Buffer.BlockCopy(buffer, 0, copy, 0, bytesRead);
|
||||||
|
|
||||||
|
foreach (var stream in allStreams)
|
||||||
|
{
|
||||||
|
stream.Value.Queue(copy, 0, copy.Length);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (onStarted != null)
|
if (onStarted != null)
|
||||||
|
@ -79,14 +87,21 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts
|
||||||
|
|
||||||
if (bytesRead > 0)
|
if (bytesRead > 0)
|
||||||
{
|
{
|
||||||
byte[] copy = new byte[bytesRead];
|
|
||||||
Buffer.BlockCopy(data.Buffer, RtpHeaderBytes, copy, 0, bytesRead);
|
|
||||||
|
|
||||||
var allStreams = _outputStreams.ToList();
|
var allStreams = _outputStreams.ToList();
|
||||||
foreach (var stream in allStreams)
|
|
||||||
|
if (allStreams.Count == 1)
|
||||||
{
|
{
|
||||||
//stream.Value.Queue(data.Buffer, RtpHeaderBytes, bytesRead);
|
await allStreams[0].Value.WriteAsync(data.Buffer, 0, bytesRead).ConfigureAwait(false);
|
||||||
stream.Value.Queue(copy, 0, copy.Length);
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
byte[] copy = new byte[bytesRead];
|
||||||
|
Buffer.BlockCopy(data.Buffer, RtpHeaderBytes, copy, 0, bytesRead);
|
||||||
|
|
||||||
|
foreach (var stream in allStreams)
|
||||||
|
{
|
||||||
|
stream.Value.Queue(copy, 0, copy.Length);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (onStarted != null)
|
if (onStarted != null)
|
||||||
|
|
|
@ -13,7 +13,7 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts
|
||||||
public class QueueStream
|
public class QueueStream
|
||||||
{
|
{
|
||||||
private readonly Stream _outputStream;
|
private readonly Stream _outputStream;
|
||||||
private readonly ConcurrentQueue<Tuple<byte[],int,int>> _queue = new ConcurrentQueue<Tuple<byte[], int, int>>();
|
private readonly ConcurrentQueue<Tuple<byte[], int, int>> _queue = new ConcurrentQueue<Tuple<byte[], int, int>>();
|
||||||
private CancellationToken _cancellationToken;
|
private CancellationToken _cancellationToken;
|
||||||
public TaskCompletionSource<bool> TaskCompletion { get; private set; }
|
public TaskCompletionSource<bool> TaskCompletion { get; private set; }
|
||||||
|
|
||||||
|
@ -50,6 +50,38 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void OnClosed()
|
||||||
|
{
|
||||||
|
GC.Collect();
|
||||||
|
if (OnFinished != null)
|
||||||
|
{
|
||||||
|
OnFinished(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task WriteAsync(byte[] bytes, int offset, int count)
|
||||||
|
{
|
||||||
|
//return _outputStream.WriteAsync(bytes, offset, count, cancellationToken);
|
||||||
|
var cancellationToken = _cancellationToken;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await _outputStream.WriteAsync(bytes, offset, count, cancellationToken).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
catch (OperationCanceledException)
|
||||||
|
{
|
||||||
|
_logger.Debug("QueueStream cancelled");
|
||||||
|
TaskCompletion.TrySetCanceled();
|
||||||
|
OnClosed();
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.ErrorException("Error in QueueStream", ex);
|
||||||
|
TaskCompletion.TrySetException(ex);
|
||||||
|
OnClosed();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private async Task StartInternal()
|
private async Task StartInternal()
|
||||||
{
|
{
|
||||||
var cancellationToken = _cancellationToken;
|
var cancellationToken = _cancellationToken;
|
||||||
|
@ -81,10 +113,7 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
if (OnFinished != null)
|
OnClosed();
|
||||||
{
|
|
||||||
OnFinished(this);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -301,10 +301,12 @@ namespace Emby.Server.Implementations.Security
|
||||||
|
|
||||||
if (reg.registered)
|
if (reg.registered)
|
||||||
{
|
{
|
||||||
|
_logger.Info("Registered for feature {0}", feature);
|
||||||
LicenseFile.AddRegCheck(feature, reg.expDate);
|
LicenseFile.AddRegCheck(feature, reg.expDate);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
_logger.Info("Not registered for feature {0}", feature);
|
||||||
LicenseFile.RemoveRegCheck(feature);
|
LicenseFile.RemoveRegCheck(feature);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,3 +1,3 @@
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
|
|
||||||
[assembly: AssemblyVersion("3.2.8.15")]
|
[assembly: AssemblyVersion("3.2.8.16")]
|
||||||
|
|
Loading…
Reference in New Issue
Block a user