2015-08-26 07:17:56 -04:00
|
|
|
|
//
|
|
|
|
|
// System.Net.Sockets.SafeSocketHandle
|
|
|
|
|
//
|
|
|
|
|
// Authors:
|
|
|
|
|
// Marcos Henrich <marcos.henrich@xamarin.com>
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Threading;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using Microsoft.Win32.SafeHandles;
|
|
|
|
|
|
|
|
|
|
namespace System.Net.Sockets {
|
|
|
|
|
|
|
|
|
|
sealed class SafeSocketHandle : SafeHandleZeroOrMinusOneIsInvalid {
|
|
|
|
|
|
|
|
|
|
List<Thread> blocking_threads;
|
|
|
|
|
|
|
|
|
|
const int SOCKET_CLOSED = 10004;
|
|
|
|
|
|
|
|
|
|
const int ABORT_RETRIES = 10;
|
|
|
|
|
static bool THROW_ON_ABORT_RETRIES = Environment.GetEnvironmentVariable("MONO_TESTS_IN_PROGRESS") == "yes";
|
|
|
|
|
|
|
|
|
|
public SafeSocketHandle (IntPtr preexistingHandle, bool ownsHandle) : base (ownsHandle)
|
|
|
|
|
{
|
|
|
|
|
SetHandle (preexistingHandle);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// This is just for marshalling
|
|
|
|
|
internal SafeSocketHandle () : base (true)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override bool ReleaseHandle ()
|
|
|
|
|
{
|
|
|
|
|
int error = 0;
|
|
|
|
|
|
|
|
|
|
Socket.Blocking_internal (handle, false, out error);
|
2016-02-22 11:00:01 -05:00
|
|
|
|
#if MOBILE_STATIC
|
|
|
|
|
/* It's only for platforms that do not have working syscall abort mechanism, like WatchOS and TvOS */
|
|
|
|
|
Socket.Shutdown_internal (handle, SocketShutdown.Both, out error);
|
|
|
|
|
#endif
|
2015-08-26 07:17:56 -04:00
|
|
|
|
|
|
|
|
|
if (blocking_threads != null) {
|
|
|
|
|
int abort_attempts = 0;
|
|
|
|
|
while (blocking_threads.Count > 0) {
|
|
|
|
|
if (abort_attempts++ >= ABORT_RETRIES) {
|
|
|
|
|
if (THROW_ON_ABORT_RETRIES)
|
|
|
|
|
throw new Exception ("Could not abort registered blocking threads before closing socket.");
|
|
|
|
|
|
|
|
|
|
// Attempts to close the socket safely failed.
|
|
|
|
|
// We give up, and close the socket with pending blocking system calls.
|
|
|
|
|
// This should not occur, nonetheless if it does this avoids an endless loop.
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* This method can be called by the DangerousRelease inside RegisterForBlockingSyscall
|
|
|
|
|
* When this happens blocking_threads contains the current thread.
|
|
|
|
|
* We can safely close the socket and throw SocketException in RegisterForBlockingSyscall
|
|
|
|
|
* before the blocking system call.
|
|
|
|
|
*/
|
|
|
|
|
lock (blocking_threads) {
|
|
|
|
|
if (blocking_threads.Count == 1 && blocking_threads[0] == Thread.CurrentThread)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
AbortRegisteredThreads ();
|
|
|
|
|
// Sleep so other threads can resume
|
|
|
|
|
Thread.Sleep (1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Socket.Close_internal (handle, out error);
|
|
|
|
|
|
|
|
|
|
return error == 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void RegisterForBlockingSyscall ()
|
|
|
|
|
{
|
|
|
|
|
if (blocking_threads == null)
|
|
|
|
|
Interlocked.CompareExchange (ref blocking_threads, new List<Thread> (), null);
|
|
|
|
|
|
|
|
|
|
bool release = false;
|
|
|
|
|
try {
|
|
|
|
|
DangerousAddRef (ref release);
|
|
|
|
|
} finally {
|
|
|
|
|
/* We must use a finally block here to make this atomic. */
|
|
|
|
|
lock (blocking_threads) {
|
|
|
|
|
blocking_threads.Add (Thread.CurrentThread);
|
|
|
|
|
}
|
|
|
|
|
if (release)
|
|
|
|
|
DangerousRelease ();
|
|
|
|
|
|
|
|
|
|
// Handle can be closed by DangerousRelease
|
|
|
|
|
if (IsClosed)
|
|
|
|
|
throw new SocketException (SOCKET_CLOSED);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* This must be called from a finally block! */
|
|
|
|
|
public void UnRegisterForBlockingSyscall ()
|
|
|
|
|
{
|
|
|
|
|
//If this NRE, we're in deep problems because Register Must have
|
|
|
|
|
lock (blocking_threads) {
|
|
|
|
|
blocking_threads.Remove (Thread.CurrentThread);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void AbortRegisteredThreads () {
|
|
|
|
|
if (blocking_threads == null)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
lock (blocking_threads) {
|
|
|
|
|
foreach (var t in blocking_threads)
|
|
|
|
|
Socket.cancel_blocking_socket_operation (t);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|