Xamarin Public Jenkins (auto-signing) 94b2861243 Imported Upstream version 4.8.0.309
Former-commit-id: 5f9c6ae75f295e057a7d2971f3a6df4656fa8850
2016-11-10 13:04:39 +00:00

143 lines
4.6 KiB
C#

//
// ClientRealProxy.cs
//
// Author:
// Atsushi Enomoto <atsushi@ximian.com>
//
// Copyright (C) 2011 Novell, Inc. http://www.novell.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.
//
#if !DISABLE_REAL_PROXY
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Runtime.Remoting;
using System.Runtime.Remoting.Messaging;
using System.Runtime.Remoting.Proxies;
using System.ServiceModel.Channels;
using System.ServiceModel.Description;
using System.ServiceModel.Dispatcher;
using System.ServiceModel.MonoInternal;
using System.Threading;
namespace System.ServiceModel
{
class ClientRealProxy : RealProxy, IRemotingTypeInfo
{
public ClientRealProxy (Type type, IInternalContextChannel channel, bool isDuplex)
: base (type)
{
this.channel = channel;
this.isDuplex = isDuplex;
}
bool isDuplex;
IInternalContextChannel channel;
Dictionary<object,object[]> saved_params = new Dictionary<object,object[]> ();
// It is used for such case that EndProcess() gets invoked
// before storing params is done after BeginProcess().
ManualResetEvent wait = new ManualResetEvent (false);
#region IRemotingTypeInfo
public virtual string TypeName { get; set; }
static bool CanCastTo<T> (Type type)
{
return typeof (T) == type || typeof (T).GetInterfaces ().Contains (type);
}
public virtual bool CanCastTo (Type t, object o)
{
if (CanCastTo<IClientChannel> (t))
return true;
#if !MOBILE
if (isDuplex && CanCastTo<IDuplexContextChannel> (t))
return true;
#endif
return false;
}
#endregion
public override IMessage Invoke (IMessage inputMessage)
{
try {
return DoInvoke (inputMessage);
} catch (TargetInvocationException ex) {
if (ex.InnerException != null)
throw ex.InnerException;
throw;
}
}
IMessage DoInvoke (IMessage inputMessage)
{
var inmsg = (IMethodCallMessage) inputMessage;
var od = channel.Contract.Operations.FirstOrDefault (o => inmsg.MethodBase.Equals (o.SyncMethod) || inmsg.MethodBase.Equals (o.BeginMethod) || inmsg.MethodBase.Equals (o.EndMethod));
if (od == null) {
// Then IContextChannel methods.
var ret = inmsg.MethodBase.Invoke (channel, inmsg.InArgs);
return new ReturnMessage (ret, null, 0, null, inmsg);
} else {
object [] pl;
MethodBase method = null;
List<object> outArgs = null;
object ret;
if (inmsg.MethodBase.Equals (od.SyncMethod)) {
// sync invocation
pl = new object [inmsg.MethodBase.GetParameters ().Length];
Array.Copy (inmsg.Args, pl, inmsg.ArgCount);
ret = channel.Process (inmsg.MethodBase, od.Name, pl, OperationContext.Current);
method = od.SyncMethod;
} else if (inmsg.MethodBase.Equals (od.BeginMethod)) {
// async invocation
pl = new object [inmsg.ArgCount - 2];
Array.Copy (inmsg.Args, 0, pl, 0, pl.Length);
ret = channel.BeginProcess (inmsg.MethodBase, od.Name, pl, (AsyncCallback) inmsg.Args [inmsg.ArgCount - 2], inmsg.Args [inmsg.ArgCount - 1]);
saved_params [ret] = pl;
wait.Set ();
} else {
var result = (IAsyncResult) inmsg.InArgs [0];
wait.WaitOne ();
pl = saved_params [result];
wait.Reset ();
saved_params.Remove (result);
ret = channel.EndProcess (inmsg.MethodBase, od.Name, pl, result);
method = od.BeginMethod;
}
if (method != null && method.GetParameters ().Any (pi => pi.IsOut || pi.ParameterType.IsByRef))
return new ReturnMessage (ret, pl, pl.Length, null, inmsg);
else
return new ReturnMessage (ret, outArgs != null ? outArgs.ToArray () : null, outArgs != null ? outArgs.Count : 0, null, inmsg);
}
}
}
}
#endif