5 // Zoltan Varga (vargaz@gmail.com)
7 // Copyright (C) 2009, Novell, Inc (http://www.novell.com)
9 // Permission is hereby granted, free of charge, to any person obtaining
10 // a copy of this software and associated documentation files (the
11 // "Software"), to deal in the Software without restriction, including
12 // without limitation the rights to use, copy, modify, merge, publish,
13 // distribute, sublicense, and/or sell copies of the Software, and to
14 // permit persons to whom the Software is furnished to do so, subject to
15 // the following conditions:
17 // The above copyright notice and this permission notice shall be
18 // included in all copies or substantial portions of the Software.
20 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
21 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
22 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
23 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
24 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
25 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
26 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
33 using System.Collections.Generic;
34 using Microsoft.Win32.SafeHandles;
35 using System.Runtime.ConstrainedExecution;
37 namespace System.Runtime.InteropServices
39 public abstract class SafeBuffer : SafeHandleZeroOrMinusOneIsInvalid, IDisposable {
41 unsafe byte *last_byte;
44 protected SafeBuffer (bool ownsHandle) : base (ownsHandle)
48 [CLSCompliant (false)]
49 public void Initialize (ulong numBytes)
52 throw new ArgumentOutOfRangeException ("numBytes");
55 byte_length = numBytes;
57 last_byte = (byte *) (((byte *) handle) + numBytes);
61 [CLSCompliant (false)]
62 public void Initialize (uint numElements, uint sizeOfEachElement)
64 Initialize (numElements * sizeOfEachElement);
67 [CLSCompliant (false)]
68 public void Initialize<T> (uint numElements) where T : struct
70 Initialize (numElements, (uint)Marshal.SizeOf (typeof (T)));
73 [CLSCompliant (false)]
74 [ReliabilityContract (Consistency.WillNotCorruptState, Cer.MayFail)]
75 public unsafe void AcquirePointer (ref byte* pointer) {
77 throw new InvalidOperationException ();
80 DangerousAddRef (ref success);
82 pointer = (byte*)handle;
85 [ReliabilityContract (Consistency.WillNotCorruptState, Cer.Success)]
86 public void ReleasePointer () {
88 throw new InvalidOperationException ();
92 [CLSCompliant (false)]
93 public ulong ByteLength {
99 [CLSCompliant (false)]
100 [ReliabilityContract (Consistency.WillNotCorruptState, Cer.MayFail)]
101 public T Read<T> (ulong byteOffset) where T : struct
104 throw new InvalidOperationException ();
107 byte *source = (((byte *) handle) + byteOffset);
108 if (source >= last_byte || source + Marshal.SizeOf (typeof (T)) > last_byte){
109 throw new ArgumentException ("byteOffset");
112 return (T) Marshal.PtrToStructure ((IntPtr) source, typeof (T));
116 [CLSCompliant (false)]
117 [ReliabilityContract (Consistency.WillNotCorruptState, Cer.MayFail)]
118 public void ReadArray<T> (ulong byteOffset, T[] array, int index, int count) where T : struct {
120 throw new InvalidOperationException ();
123 int size = Marshal.SizeOf (typeof (T)) * count;
124 byte *source = (((byte *) handle) + byteOffset);
125 if (source >= last_byte || source + size > last_byte)
126 throw new ArgumentException ("byteOffset");
128 Marshal.copy_from_unmanaged ((IntPtr) source, index, array, count);
132 [CLSCompliant (false)]
133 [ReliabilityContract (Consistency.WillNotCorruptState, Cer.MayFail)]
134 public void Write<T> (ulong byteOffset, T value) where T : struct {
136 throw new InvalidOperationException ();
139 byte *target = (((byte *) handle) + byteOffset);
140 if (target >= last_byte || target + Marshal.SizeOf (typeof (T)) > last_byte)
141 throw new ArgumentException ("byteOffset");
143 Marshal.StructureToPtr (value, (IntPtr) target, false);
147 [CLSCompliant (false)]
148 [ReliabilityContract (Consistency.WillNotCorruptState, Cer.MayFail)]
149 public void WriteArray<T> (ulong byteOffset, T[] array, int index, int count) where T : struct
152 throw new InvalidOperationException ();
155 byte *target = ((byte *) handle) + byteOffset;
156 int size = Marshal.SizeOf (typeof (T)) * count;
157 if (target >= last_byte || target + size > last_byte)
158 throw new ArgumentException ("would overrite");
160 Marshal.copy_to_unmanaged (array, index, (IntPtr) target, count);