Welcome to mirror list, hosted at ThFree Co, Russian Federation.

github.com/mono/corefx.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorStephen Toub <stoub@microsoft.com>2017-07-14 04:48:10 +0300
committerGitHub <noreply@github.com>2017-07-14 04:48:10 +0300
commita26de495ffa390a9d6b3a0d0f338e9d9f97ad5fe (patch)
tree1481acb15bae0d8849fecbf9e2e1c14eb09a2e37 /src/System.Net.Http
parent78db832b8aa934207c9655a9321d416185c35e60 (diff)
Use a shared MultiAgent in CurlHandler when using LibreSSL backend on macOS (#22242)
LibreSSL needs locks initialized in order to be used safely from multiple threads, which happens when multiple HttpClients are used. But we don't currently have a good way to perform that initialization as we do for OpenSSL. As a workaround, add a shared MultiAgent that's initialized when LibreSSL is used on macOS, and use that same MultiAgent from all CurlHandler instances rather than giving each their own. Since MultiAgent is 1:1 with a thread, all HttpClient's will then end up sharing the same libcurl event loop thread, and we'll serialize all of our interactions with libcurl and thus hopefully with LibreSSL.
Diffstat (limited to 'src/System.Net.Http')
-rw-r--r--src/System.Net.Http/src/System/Net/Http/OSX/CurlHandler.SslProvider.cs13
-rw-r--r--src/System.Net.Http/src/System/Net/Http/Unix/CurlHandler.MultiAgent.cs44
-rw-r--r--src/System.Net.Http/src/System/Net/Http/Unix/CurlHandler.cs20
3 files changed, 57 insertions, 20 deletions
diff --git a/src/System.Net.Http/src/System/Net/Http/OSX/CurlHandler.SslProvider.cs b/src/System.Net.Http/src/System/Net/Http/OSX/CurlHandler.SslProvider.cs
index 134cfec0dc..3e9b3e1f6d 100644
--- a/src/System.Net.Http/src/System/Net/Http/OSX/CurlHandler.SslProvider.cs
+++ b/src/System.Net.Http/src/System/Net/Http/OSX/CurlHandler.SslProvider.cs
@@ -12,6 +12,19 @@ namespace System.Net.Http
{
internal partial class CurlHandler : HttpMessageHandler
{
+ static partial void UseSingletonMultiAgent(ref bool result)
+ {
+ // Some backends other than OpenSSL need locks initialized in order to use them in a
+ // multithreaded context, which would happen with multiple HttpClients and thus multiple
+ // MultiAgents. Since we don't currently have the ability to do so initialization, instead we
+ // restrict all HttpClients to use the same MultiAgent instance in this case. We know LibreSSL
+ // is in this camp, so we currently special-case it.
+ string curlSslVersion = Interop.Http.GetSslVersionDescription();
+ result =
+ !string.IsNullOrEmpty(curlSslVersion) &&
+ curlSslVersion.StartsWith(Interop.Http.LibreSslDescription, StringComparison.OrdinalIgnoreCase);
+ }
+
private static class SslProvider
{
internal static void SetSslOptions(EasyRequest easy, ClientCertificateOption clientCertOption)
diff --git a/src/System.Net.Http/src/System/Net/Http/Unix/CurlHandler.MultiAgent.cs b/src/System.Net.Http/src/System/Net/Http/Unix/CurlHandler.MultiAgent.cs
index 63358976c2..db20c778a3 100644
--- a/src/System.Net.Http/src/System/Net/Http/Unix/CurlHandler.MultiAgent.cs
+++ b/src/System.Net.Http/src/System/Net/Http/Unix/CurlHandler.MultiAgent.cs
@@ -37,8 +37,8 @@ namespace System.Net.Http
private static readonly Interop.Http.ReadWriteCallback s_receiveBodyCallback = CurlReceiveBodyCallback;
private static readonly Interop.Http.DebugCallback s_debugCallback = CurlDebugFunction;
- /// <summary>CurlHandler that owns this MultiAgent.</summary>
- private readonly CurlHandler _associatedHandler;
+ /// <summary>CurlHandler that created this MultiAgent. If null, this is a shared handler.</summary>
+ private readonly CurlHandler _creatingHandler;
/// <summary>
/// A collection of not-yet-processed incoming requests for work to be done
@@ -79,18 +79,21 @@ namespace System.Net.Http
/// </summary>
private Interop.Http.SafeCurlMultiHandle _multiHandle;
+ /// <summary>Set when Dispose has been called.</summary>
+ private bool _disposed;
+
/// <summary>Initializes the MultiAgent.</summary>
- /// <param name="handler">The handler that owns this agent.</param>
+ /// <param name="handler">The handler that created this agent, or null if it's shared.</param>
public MultiAgent(CurlHandler handler)
{
- Debug.Assert(handler != null, "Expected non-null handler");
- _associatedHandler = handler;
+ _creatingHandler = handler;
}
/// <summary>Disposes of the agent.</summary>
public void Dispose()
{
EventSourceTrace(null);
+ _disposed = true;
QueueIfRunning(new IncomingRequest { Type = IncomingRequestType.Shutdown });
_multiHandle?.Dispose();
}
@@ -247,20 +250,25 @@ namespace System.Net.Http
EventSourceTrace("Set multiplexing on multi handle");
}
- // Configure max connections per host if it was changed from the default
- int maxConnections = _associatedHandler.MaxConnectionsPerServer;
- if (maxConnections < int.MaxValue) // int.MaxValue considered infinite, mapping to libcurl default of 0
+ // Configure max connections per host if it was changed from the default. In shared mode,
+ // this will be pulled from the handler that first created the agent; the setting from subsequent
+ // handlers that use this will be ignored.
+ if (_creatingHandler != null)
{
- CURLMcode code = Interop.Http.MultiSetOptionLong(multiHandle, Interop.Http.CURLMoption.CURLMOPT_MAX_HOST_CONNECTIONS, maxConnections);
- switch (code)
+ int maxConnections = _creatingHandler.MaxConnectionsPerServer;
+ if (maxConnections < int.MaxValue) // int.MaxValue considered infinite, mapping to libcurl default of 0
{
- case CURLMcode.CURLM_OK:
- EventSourceTrace("Set max host connections to {0}", maxConnections);
- break;
- default:
- // Treat failures as non-fatal in release; worst case is we employ more connections than desired.
- EventSourceTrace("Setting CURLMOPT_MAX_HOST_CONNECTIONS failed: {0}. Ignoring option.", code);
- break;
+ CURLMcode code = Interop.Http.MultiSetOptionLong(multiHandle, Interop.Http.CURLMoption.CURLMOPT_MAX_HOST_CONNECTIONS, maxConnections);
+ switch (code)
+ {
+ case CURLMcode.CURLM_OK:
+ EventSourceTrace("Set max host connections to {0}", maxConnections);
+ break;
+ default:
+ // Treat failures as non-fatal in release; worst case is we employ more connections than desired.
+ EventSourceTrace("Setting CURLMOPT_MAX_HOST_CONNECTIONS failed: {0}. Ignoring option.", code);
+ break;
+ }
}
}
@@ -306,7 +314,7 @@ namespace System.Net.Http
// more requests could have been added. If they were,
// kick off another processing loop.
_runningWorker = null;
- if (_incomingRequests.Count > 0 && !_associatedHandler._disposed)
+ if (_incomingRequests.Count > 0 && !_disposed)
{
EnsureWorkerIsRunning();
}
diff --git a/src/System.Net.Http/src/System/Net/Http/Unix/CurlHandler.cs b/src/System.Net.Http/src/System/Net/Http/Unix/CurlHandler.cs
index b962ca7529..a9e4e35791 100644
--- a/src/System.Net.Http/src/System/Net/Http/Unix/CurlHandler.cs
+++ b/src/System.Net.Http/src/System/Net/Http/Unix/CurlHandler.cs
@@ -126,6 +126,7 @@ namespace System.Net.Http
private static string s_curlVersionDescription;
private static string s_curlSslVersionDescription;
+ private static readonly MultiAgent s_singletonSharedAgent;
private readonly MultiAgent _agent;
private volatile bool _anyOperationStarted;
private volatile bool _disposed;
@@ -169,13 +170,28 @@ namespace System.Net.Http
{
EventSourceTrace($"libcurl: {CurlVersionDescription} {CurlSslVersionDescription} {features}");
}
+
+ // By default every CurlHandler gets its own MultiAgent. But for some backends,
+ // we need to restrict the number of threads involved in processing libcurl work,
+ // so we create a single MultiAgent that's used by all handlers.
+ bool useSingleton = false;
+ UseSingletonMultiAgent(ref useSingleton);
+ if (useSingleton)
+ {
+ s_singletonSharedAgent = new MultiAgent(null);
+ }
}
public CurlHandler()
{
- _agent = new MultiAgent(this);
+ // If the shared MultiAgent was initialized, use it.
+ // Otherwise, create a new MultiAgent for this handler.
+ _agent = s_singletonSharedAgent ?? new MultiAgent(this);
}
+ /// <summary>Overridden by another partial implementation to set <see cref="result"/> to true if a single MultiAgent should be used.</summary>
+ static partial void UseSingletonMultiAgent(ref bool result);
+
#region Properties
private static string CurlVersionDescription => s_curlVersionDescription ?? (s_curlVersionDescription = Interop.Http.GetVersionDescription() ?? string.Empty);
@@ -409,7 +425,7 @@ namespace System.Net.Http
protected override void Dispose(bool disposing)
{
_disposed = true;
- if (disposing)
+ if (disposing && _agent != s_singletonSharedAgent)
{
_agent.Dispose();
}