mirror of
				https://github.com/gnh1201/welsonjs.git
				synced 2025-10-26 18:41:18 +00:00 
			
		
		
		
	Add the package ManagedEsent (microsoft/ManagedEsent@d358c07), and WelsonJS.Esent
		
			
				
	
	
		
			95 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			95 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //-----------------------------------------------------------------------
 | |
| // <copyright file="GCHandleCollection.cs" company="Microsoft Corporation">
 | |
| //     Copyright (c) Microsoft Corporation.
 | |
| // </copyright>
 | |
| //-----------------------------------------------------------------------
 | |
| 
 | |
| namespace Microsoft.Isam.Esent.Interop.Implementation
 | |
| {
 | |
|     using System;
 | |
|     using System.Collections.Generic;
 | |
|     using System.Diagnostics;
 | |
|     using System.Runtime.InteropServices;
 | |
| 
 | |
|     /// <summary>
 | |
|     /// A collection of GCHandles for pinned objects. The handles
 | |
|     /// are freed when this object is disposed.
 | |
|     /// </summary>
 | |
|     [StructLayout(LayoutKind.Auto)]
 | |
|     internal struct GCHandleCollection : IDisposable
 | |
|     {
 | |
|         /// <summary>
 | |
|         /// The handles of the objects being pinned.
 | |
|         /// </summary>
 | |
|         private GCHandle[] handles;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Handle count.
 | |
|         /// </summary>
 | |
|         private int count;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Disposes of the object.
 | |
|         /// </summary>
 | |
|         public void Dispose()
 | |
|         {
 | |
|             if (null != this.handles)
 | |
|             {
 | |
|                 for (int i = 0; i < this.count; i++)
 | |
|                 {
 | |
|                     this.handles[i].Free();
 | |
|                 }
 | |
| 
 | |
|                 this.handles = null;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Add an object to the handle collection. This automatically
 | |
|         /// pins the object.
 | |
|         /// </summary>
 | |
|         /// <param name="value">The object to pin.</param>
 | |
|         /// <returns>
 | |
|         /// The address of the pinned object. This is valid until the
 | |
|         /// GCHandleCollection is disposed.
 | |
|         /// </returns>
 | |
|         public IntPtr Add(object value)
 | |
|         {
 | |
|             if (null == value)
 | |
|             {
 | |
|                 return IntPtr.Zero;
 | |
|             }
 | |
| 
 | |
|             if (null == this.handles)
 | |
|             {
 | |
|                 this.handles = new GCHandle[4]; // same as List<T>
 | |
|             }
 | |
|             else if (this.count == this.handles.Length)
 | |
|             {
 | |
|                 Array.Resize(ref this.handles, this.count * 2);
 | |
|             }
 | |
| 
 | |
|             Debug.Assert(this.count < this.handles.Length, "Index out of bound");
 | |
| 
 | |
|             GCHandle handle = GCHandle.Alloc(value, GCHandleType.Pinned);
 | |
|             this.handles[this.count++] = handle;
 | |
| 
 | |
|             IntPtr pinned = handle.AddrOfPinnedObject();
 | |
|             Debug.Assert(IntPtr.Zero != pinned, "Pinned object has null address");
 | |
|             return pinned;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Set handle array capacity.
 | |
|         /// </summary>
 | |
|         /// <param name="capacity">Estimated handle count</param>
 | |
|         public void SetCapacity(int capacity)
 | |
|         {
 | |
|             if (null == this.handles)
 | |
|             {
 | |
|                 this.handles = new GCHandle[capacity];
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |