2015-08-26 07:17:56 -04:00
|
|
|
// System.Net.Sockets.SocketAsyncResult.cs
|
|
|
|
//
|
|
|
|
// Authors:
|
|
|
|
// Ludovic Henry <ludovic@xamarin.com>
|
|
|
|
//
|
|
|
|
// Copyright (C) 2015 Xamarin, Inc. (https://www.xamarin.com)
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
|
|
// a copy of this software and associated documentation files (the
|
|
|
|
// "Software"), to deal in the Software without restriction, including
|
|
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
|
|
// the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be
|
|
|
|
// included in all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
//
|
|
|
|
|
|
|
|
using System.Collections;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
using System.Runtime.Remoting.Messaging;
|
|
|
|
using System.Threading;
|
|
|
|
|
|
|
|
namespace System.Net.Sockets
|
|
|
|
{
|
|
|
|
[StructLayout (LayoutKind.Sequential)]
|
2016-02-22 11:00:01 -05:00
|
|
|
internal sealed class SocketAsyncResult: IOAsyncResult
|
2015-08-26 07:17:56 -04:00
|
|
|
{
|
|
|
|
public Socket socket;
|
2016-02-22 11:00:01 -05:00
|
|
|
public SocketOperation operation;
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-02-22 11:00:01 -05:00
|
|
|
Exception DelayedException;
|
2015-08-26 07:17:56 -04:00
|
|
|
|
|
|
|
public EndPoint EndPoint; // Connect,ReceiveFrom,SendTo
|
|
|
|
public byte [] Buffer; // Receive,ReceiveFrom,Send,SendTo
|
|
|
|
public int Offset; // Receive,ReceiveFrom,Send,SendTo
|
|
|
|
public int Size; // Receive,ReceiveFrom,Send,SendTo
|
|
|
|
public SocketFlags SockFlags; // Receive,ReceiveFrom,Send,SendTo
|
|
|
|
public Socket AcceptSocket; // AcceptReceive
|
|
|
|
public IPAddress[] Addresses; // Connect
|
|
|
|
public int Port; // Connect
|
|
|
|
public IList<ArraySegment<byte>> Buffers; // Receive, Send
|
|
|
|
public bool ReuseSocket; // Disconnect
|
2016-02-22 11:00:01 -05:00
|
|
|
public int CurrentAddress; // Connect
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-02-22 11:00:01 -05:00
|
|
|
public Socket AcceptedSocket;
|
|
|
|
public int Total;
|
2015-08-26 07:17:56 -04:00
|
|
|
|
|
|
|
internal int error;
|
2016-02-22 11:00:01 -05:00
|
|
|
|
2015-08-26 07:17:56 -04:00
|
|
|
public int EndCalled;
|
|
|
|
|
2016-02-22 11:00:01 -05:00
|
|
|
public IntPtr Handle {
|
|
|
|
get { return socket != null ? socket.Handle : IntPtr.Zero; }
|
|
|
|
}
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-02-22 11:00:01 -05:00
|
|
|
/* Used by SocketAsyncEventArgs */
|
2015-08-26 07:17:56 -04:00
|
|
|
public SocketAsyncResult ()
|
2016-02-22 11:00:01 -05:00
|
|
|
: base ()
|
2015-08-26 07:17:56 -04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2016-02-22 11:00:01 -05:00
|
|
|
public void Init (Socket socket, AsyncCallback callback, object state, SocketOperation operation)
|
2015-08-26 07:17:56 -04:00
|
|
|
{
|
2016-02-22 11:00:01 -05:00
|
|
|
base.Init (callback, state);
|
2015-08-26 07:17:56 -04:00
|
|
|
|
|
|
|
this.socket = socket;
|
|
|
|
this.operation = operation;
|
|
|
|
|
2016-02-22 11:00:01 -05:00
|
|
|
DelayedException = null;
|
2015-08-26 07:17:56 -04:00
|
|
|
|
|
|
|
EndPoint = null;
|
|
|
|
Buffer = null;
|
|
|
|
Offset = 0;
|
|
|
|
Size = 0;
|
|
|
|
SockFlags = SocketFlags.None;
|
|
|
|
AcceptSocket = null;
|
|
|
|
Addresses = null;
|
|
|
|
Port = 0;
|
|
|
|
Buffers = null;
|
|
|
|
ReuseSocket = false;
|
2016-02-22 11:00:01 -05:00
|
|
|
CurrentAddress = 0;
|
|
|
|
|
|
|
|
AcceptedSocket = null;
|
|
|
|
Total = 0;
|
2015-08-26 07:17:56 -04:00
|
|
|
|
|
|
|
error = 0;
|
2016-02-22 11:00:01 -05:00
|
|
|
|
2015-08-26 07:17:56 -04:00
|
|
|
EndCalled = 0;
|
|
|
|
}
|
|
|
|
|
2016-02-22 11:00:01 -05:00
|
|
|
public SocketAsyncResult (Socket socket, AsyncCallback callback, object state, SocketOperation operation)
|
|
|
|
: base (callback, state)
|
2015-08-26 07:17:56 -04:00
|
|
|
{
|
2016-02-22 11:00:01 -05:00
|
|
|
this.socket = socket;
|
|
|
|
this.operation = operation;
|
2015-08-26 07:17:56 -04:00
|
|
|
}
|
|
|
|
|
2016-02-22 11:00:01 -05:00
|
|
|
public SocketError ErrorCode {
|
|
|
|
get {
|
|
|
|
SocketException ex = DelayedException as SocketException;
|
|
|
|
if (ex != null)
|
|
|
|
return ex.SocketErrorCode;
|
|
|
|
|
|
|
|
if (error != 0)
|
|
|
|
return (SocketError) error;
|
|
|
|
|
|
|
|
return SocketError.Success;
|
2015-08-26 07:17:56 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void CheckIfThrowDelayedException ()
|
|
|
|
{
|
2016-02-22 11:00:01 -05:00
|
|
|
if (DelayedException != null) {
|
2015-08-26 07:17:56 -04:00
|
|
|
socket.is_connected = false;
|
2016-02-22 11:00:01 -05:00
|
|
|
throw DelayedException;
|
2015-08-26 07:17:56 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (error != 0) {
|
|
|
|
socket.is_connected = false;
|
|
|
|
throw new SocketException (error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-22 11:00:01 -05:00
|
|
|
internal override void CompleteDisposed ()
|
2015-08-26 07:17:56 -04:00
|
|
|
{
|
|
|
|
Complete ();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Complete ()
|
|
|
|
{
|
|
|
|
if (operation != SocketOperation.Receive && socket.is_disposed)
|
2016-02-22 11:00:01 -05:00
|
|
|
DelayedException = new ObjectDisposedException (socket.GetType ().ToString ());
|
2015-08-26 07:17:56 -04:00
|
|
|
|
|
|
|
IsCompleted = true;
|
|
|
|
|
2016-02-22 11:00:01 -05:00
|
|
|
AsyncCallback callback = AsyncCallback;
|
|
|
|
if (callback != null) {
|
|
|
|
ThreadPool.UnsafeQueueUserWorkItem (_ => callback (this), null);
|
|
|
|
}
|
|
|
|
|
|
|
|
Queue<KeyValuePair<IntPtr, IOSelectorJob>> queue = null;
|
2015-08-26 07:17:56 -04:00
|
|
|
switch (operation) {
|
|
|
|
case SocketOperation.Receive:
|
|
|
|
case SocketOperation.ReceiveFrom:
|
|
|
|
case SocketOperation.ReceiveGeneric:
|
|
|
|
case SocketOperation.Accept:
|
|
|
|
queue = socket.readQ;
|
|
|
|
break;
|
|
|
|
case SocketOperation.Send:
|
|
|
|
case SocketOperation.SendTo:
|
|
|
|
case SocketOperation.SendGeneric:
|
|
|
|
queue = socket.writeQ;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (queue != null) {
|
|
|
|
lock (queue) {
|
|
|
|
/* queue.Count will only be 0 if the socket is closed while receive/send/accept
|
|
|
|
* operation(s) are pending and at least one call to this method is waiting
|
|
|
|
* on the lock while another one calls CompleteAllOnDispose() */
|
|
|
|
if (queue.Count > 0)
|
|
|
|
queue.Dequeue (); /* remove ourselves */
|
|
|
|
if (queue.Count > 0) {
|
|
|
|
if (!socket.is_disposed) {
|
2016-02-22 11:00:01 -05:00
|
|
|
IOSelector.Add (queue.Peek ().Key, queue.Peek ().Value);
|
2015-08-26 07:17:56 -04:00
|
|
|
} else {
|
|
|
|
/* CompleteAllOnDispose */
|
2016-02-22 11:00:01 -05:00
|
|
|
KeyValuePair<IntPtr, IOSelectorJob> [] jobs = queue.ToArray ();
|
|
|
|
for (int i = 0; i < jobs.Length; i++)
|
|
|
|
ThreadPool.QueueUserWorkItem (j => ((IOSelectorJob) j).MarkDisposed (), jobs [i].Value);
|
2015-08-26 07:17:56 -04:00
|
|
|
queue.Clear ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// IMPORTANT: 'callback', if any is scheduled from unmanaged code
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Complete (bool synch)
|
|
|
|
{
|
2016-02-22 11:00:01 -05:00
|
|
|
CompletedSynchronously = synch;
|
2015-08-26 07:17:56 -04:00
|
|
|
Complete ();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Complete (int total)
|
|
|
|
{
|
2016-02-22 11:00:01 -05:00
|
|
|
Total = total;
|
2015-08-26 07:17:56 -04:00
|
|
|
Complete ();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Complete (Exception e, bool synch)
|
|
|
|
{
|
2016-02-22 11:00:01 -05:00
|
|
|
DelayedException = e;
|
|
|
|
CompletedSynchronously = synch;
|
2015-08-26 07:17:56 -04:00
|
|
|
Complete ();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Complete (Exception e)
|
|
|
|
{
|
2016-02-22 11:00:01 -05:00
|
|
|
DelayedException = e;
|
2015-08-26 07:17:56 -04:00
|
|
|
Complete ();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Complete (Socket s)
|
|
|
|
{
|
2016-02-22 11:00:01 -05:00
|
|
|
AcceptedSocket = s;
|
2015-08-26 07:17:56 -04:00
|
|
|
Complete ();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Complete (Socket s, int total)
|
|
|
|
{
|
2016-02-22 11:00:01 -05:00
|
|
|
AcceptedSocket = s;
|
|
|
|
Total = total;
|
2015-08-26 07:17:56 -04:00
|
|
|
Complete ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|