#region Copyright notice and license // Copyright 2015 gRPC authors. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #endregion using System; using System.Runtime.InteropServices; using System.Text; using Grpc.Core; using Grpc.Core.Logging; using Grpc.Core.Utils; namespace Grpc.Core.Internal { internal interface IOpCompletionCallback { void OnComplete(bool success); } /// /// grpcsharp_batch_context /// internal class BatchContextSafeHandle : SafeHandleZeroIsInvalid, IOpCompletionCallback, IPooledObject { static readonly NativeMethods Native = NativeMethods.Get(); static readonly ILogger Logger = GrpcEnvironment.Logger.ForType(); Action returnToPoolAction; CompletionCallbackData completionCallbackData; private BatchContextSafeHandle() { } public static BatchContextSafeHandle Create() { var ctx = Native.grpcsharp_batch_context_create(); return ctx; } public IntPtr Handle { get { return handle; } } public void SetReturnToPoolAction(Action returnAction) { GrpcPreconditions.CheckState(returnToPoolAction == null); returnToPoolAction = returnAction; } public void SetCompletionCallback(BatchCompletionDelegate callback, object state) { GrpcPreconditions.CheckState(completionCallbackData.Callback == null); GrpcPreconditions.CheckNotNull(callback, nameof(callback)); completionCallbackData = new CompletionCallbackData(callback, state); } // Gets data of recv_initial_metadata completion. public Metadata GetReceivedInitialMetadata() { IntPtr metadataArrayPtr = Native.grpcsharp_batch_context_recv_initial_metadata(this); return MetadataArraySafeHandle.ReadMetadataFromPtrUnsafe(metadataArrayPtr); } // Gets data of recv_status_on_client completion. public ClientSideStatus GetReceivedStatusOnClient() { UIntPtr detailsLength; IntPtr detailsPtr = Native.grpcsharp_batch_context_recv_status_on_client_details(this, out detailsLength); string details = MarshalUtils.PtrToStringUTF8(detailsPtr, (int)detailsLength.ToUInt32()); var status = new Status(Native.grpcsharp_batch_context_recv_status_on_client_status(this), details); IntPtr metadataArrayPtr = Native.grpcsharp_batch_context_recv_status_on_client_trailing_metadata(this); var metadata = MetadataArraySafeHandle.ReadMetadataFromPtrUnsafe(metadataArrayPtr); return new ClientSideStatus(status, metadata); } // Gets data of recv_message completion. public byte[] GetReceivedMessage() { IntPtr len = Native.grpcsharp_batch_context_recv_message_length(this); if (len == new IntPtr(-1)) { return null; } byte[] data = new byte[(int)len]; Native.grpcsharp_batch_context_recv_message_to_buffer(this, data, new UIntPtr((ulong)data.Length)); return data; } // Gets data of receive_close_on_server completion. public bool GetReceivedCloseOnServerCancelled() { return Native.grpcsharp_batch_context_recv_close_on_server_cancelled(this) != 0; } public void Recycle() { if (returnToPoolAction != null) { Native.grpcsharp_batch_context_reset(this); var origReturnAction = returnToPoolAction; // Not clearing all the references to the pool could prevent garbage collection of the pool object // and thus cause memory leaks. returnToPoolAction = null; origReturnAction(this); } else { Dispose(); } } protected override bool ReleaseHandle() { Native.grpcsharp_batch_context_destroy(handle); return true; } void IOpCompletionCallback.OnComplete(bool success) { try { completionCallbackData.Callback(success, this, completionCallbackData.State); } catch (Exception e) { Logger.Error(e, "Exception occurred while invoking batch completion delegate."); } finally { completionCallbackData = default(CompletionCallbackData); Recycle(); } } struct CompletionCallbackData { public CompletionCallbackData(BatchCompletionDelegate callback, object state) { this.Callback = callback; this.State = state; } public BatchCompletionDelegate Callback { get; } public object State { get; } } } }