RpcProcessor.cs 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  1. using Mono.CecilX;
  2. using Mono.CecilX.Cil;
  3. namespace Mirror.Weaver
  4. {
  5. /// <summary>
  6. /// Processes [Rpc] methods in NetworkBehaviour
  7. /// </summary>
  8. public static class RpcProcessor
  9. {
  10. public static MethodDefinition ProcessRpcInvoke(TypeDefinition td, MethodDefinition md, MethodDefinition rpcCallFunc)
  11. {
  12. MethodDefinition rpc = new MethodDefinition(
  13. Weaver.InvokeRpcPrefix + md.Name,
  14. MethodAttributes.Family | MethodAttributes.Static | MethodAttributes.HideBySig,
  15. WeaverTypes.Import(typeof(void)));
  16. ILProcessor worker = rpc.Body.GetILProcessor();
  17. Instruction label = worker.Create(OpCodes.Nop);
  18. NetworkBehaviourProcessor.WriteClientActiveCheck(worker, md.Name, label, "RPC");
  19. // setup for reader
  20. worker.Emit(OpCodes.Ldarg_0);
  21. worker.Emit(OpCodes.Castclass, td);
  22. if (!NetworkBehaviourProcessor.ReadArguments(md, worker, RemoteCallType.ClientRpc))
  23. return null;
  24. // invoke actual command function
  25. worker.Emit(OpCodes.Callvirt, rpcCallFunc);
  26. worker.Emit(OpCodes.Ret);
  27. NetworkBehaviourProcessor.AddInvokeParameters(rpc.Parameters);
  28. td.Methods.Add(rpc);
  29. return rpc;
  30. }
  31. /*
  32. * generates code like:
  33. public void RpcTest (int param)
  34. {
  35. NetworkWriter writer = new NetworkWriter ();
  36. writer.WritePackedUInt32((uint)param);
  37. base.SendRPCInternal(typeof(class),"RpcTest", writer, 0);
  38. }
  39. public void CallRpcTest (int param)
  40. {
  41. // whatever the user did before
  42. }
  43. Originally HLAPI put the send message code inside the Call function
  44. and then proceeded to replace every call to RpcTest with CallRpcTest
  45. This method moves all the user's code into the "CallRpc" method
  46. and replaces the body of the original method with the send message code.
  47. This way we do not need to modify the code anywhere else, and this works
  48. correctly in dependent assemblies
  49. */
  50. public static MethodDefinition ProcessRpcCall(TypeDefinition td, MethodDefinition md, CustomAttribute clientRpcAttr)
  51. {
  52. MethodDefinition rpc = MethodProcessor.SubstituteMethod(td, md);
  53. ILProcessor worker = md.Body.GetILProcessor();
  54. NetworkBehaviourProcessor.WriteSetupLocals(worker);
  55. if (Weaver.GenerateLogErrors)
  56. {
  57. worker.Emit(OpCodes.Ldstr, "Call ClientRpc function " + md.Name);
  58. worker.Emit(OpCodes.Call, WeaverTypes.logErrorReference);
  59. }
  60. NetworkBehaviourProcessor.WriteCreateWriter(worker);
  61. // write all the arguments that the user passed to the Rpc call
  62. if (!NetworkBehaviourProcessor.WriteArguments(worker, md, RemoteCallType.ClientRpc))
  63. return null;
  64. string rpcName = md.Name;
  65. int channel = clientRpcAttr.GetField("channel", 0);
  66. bool includeOwner = clientRpcAttr.GetField("includeOwner", true);
  67. // invoke SendInternal and return
  68. // this
  69. worker.Emit(OpCodes.Ldarg_0);
  70. worker.Emit(OpCodes.Ldtoken, td);
  71. // invokerClass
  72. worker.Emit(OpCodes.Call, WeaverTypes.getTypeFromHandleReference);
  73. worker.Emit(OpCodes.Ldstr, rpcName);
  74. // writer
  75. worker.Emit(OpCodes.Ldloc_0);
  76. worker.Emit(OpCodes.Ldc_I4, channel);
  77. // includeOwner ? 1 : 0
  78. worker.Emit(includeOwner ? OpCodes.Ldc_I4_1 : OpCodes.Ldc_I4_0);
  79. worker.Emit(OpCodes.Callvirt, WeaverTypes.sendRpcInternal);
  80. NetworkBehaviourProcessor.WriteRecycleWriter(worker);
  81. worker.Emit(OpCodes.Ret);
  82. return rpc;
  83. }
  84. }
  85. }