Пример #1
0
        public void StructEnumTest()
        {
            using (var ms = new NativeMemoryStream(48, 16))
            {
                var arr = new[] { 1L, 2L, 3L, 4L, 5L, 6L };
                Assert.That(() => ms.Write(arr), Throws.Nothing);
                var sz = sizeof(long) * arr.Length;
                Assert.That(ms.Length, Is.EqualTo(sz));
                Assert.That(ms.Position, Is.EqualTo(sz));
                Assert.That(ms.Capacity, Is.EqualTo(48));

                var list = new List <int> {
                    1, 2, 3
                };
                Assert.That(() => ms.Write(list), Throws.Nothing);
                sz += sizeof(int) * list.Count;
                Assert.That(ms.Length, Is.EqualTo(sz));
                Assert.That(ms.Position, Is.EqualTo(sz));
                Assert.That(ms.Capacity, Is.EqualTo(64));

                var pos = ms.Position;
                Assert.That(() => ms.Write(arr, true), Throws.Nothing);
                sz  += (sizeof(long) + IntPtr.Size) * arr.Length;
                pos += IntPtr.Size * arr.Length;
                Assert.That(ms.Length, Is.EqualTo(sz));
                Assert.That(ms.Position, Is.EqualTo(pos));
                Assert.That(ms.Capacity, Is.EqualTo(144));

                Assert.That(() => ms.Write(list, true), Throws.Nothing);
                sz  += (sizeof(int) + IntPtr.Size) * list.Count;
                pos += IntPtr.Size * list.Count;
                Assert.That(ms.Length, Is.EqualTo(sz));
                Assert.That(ms.Position, Is.EqualTo(pos));
                Assert.That(ms.Capacity, Is.EqualTo(160));

                ms.Flush();
                ms.Position = 0;

                Assert.That(() => ms.ReadArray(null, 0, false), Throws.ArgumentNullException);
                Assert.That(() => ms.ReadArray(typeof(int), -1, false), Throws.InstanceOf <ArgumentOutOfRangeException>());
                Assert.That(() => ms.ReadArray(typeof(int), 0, false), Throws.Nothing);
                Assert.That(ms.ReadArray(typeof(int), 0, false).Length, Is.Zero);
                Assert.That(() => ms.ReadArray(typeof(Guid), 100, false), Throws.InstanceOf <ArgumentOutOfRangeException>());

                Assert.That(ms.ReadArray(typeof(long), arr.Length, false), Is.EquivalentTo(arr));
                Assert.That(ms.ReadArray <int>(list.Count, false), Is.EquivalentTo(list));
                Assert.That(ms.ReadArray <long>(arr.Length, true), Is.EquivalentTo(arr));
                Assert.That(ms.ReadArray(typeof(int), list.Count, true), Is.EquivalentTo(list));
            }
        }
Пример #2
0
 public void NativeMemoryStreamTest2()
 {
     using (var m = new SafeHGlobalHandle(10))
     {
         Assert.That(() => new NativeMemoryStream(null, 10, 20), Throws.ArgumentNullException);
         using (var ms = new NativeMemoryStream(m, 10, 20))
         {
             Assert.That(ms.Capacity, Is.EqualTo(10));
             Assert.That(ms.MaxCapacity, Is.EqualTo(20));
             Assert.That(ms.Pointer, Is.EqualTo((IntPtr)m));
             Assert.That(() => ms.Write(Guid.NewGuid()), Throws.Nothing);
             Assert.That(ms.Capacity, Is.EqualTo(20));
             Assert.That(() => ms.Write(Guid.NewGuid()), Throws.Exception);
         }
     }
 }
Пример #3
0
        /// <summary>
        /// Marshals data from a managed list of strings to an unmanaged block of memory allocated by the <paramref name="memAlloc"/> method.
        /// </summary>
        /// <param name="values">The enumerated list of strings to marshal.</param>
        /// <param name="packing">The packing type for the strings.</param>
        /// <param name="memAlloc">
        /// The function that allocates the memory for the block of strings (typically <see cref="Marshal.AllocCoTaskMem(int)"/> or <see cref="Marshal.AllocHGlobal(int)"/>.
        /// </param>
        /// <param name="bytesAllocated">The bytes allocated by the <paramref name="memAlloc"/> method.</param>
        /// <param name="charSet">The character set to use for the strings.</param>
        /// <param name="prefixBytes">Number of bytes preceding the trailing strings.</param>
        /// <returns>
        /// Pointer to the allocated native (unmanaged) array of strings stored using the <paramref name="packing"/> model and the character
        /// set defined by <paramref name="charSet"/>.
        /// </returns>
        public static IntPtr MarshalToPtr(this IEnumerable <string> values, StringListPackMethod packing, Func <int, IntPtr> memAlloc, out int bytesAllocated, CharSet charSet = CharSet.Auto, int prefixBytes = 0)
        {
            // Bail early if empty
            if (values is null || !values.Any())
            {
                bytesAllocated = prefixBytes + (packing == StringListPackMethod.Concatenated ? StringHelper.GetCharSize(charSet) : IntPtr.Size);
                var ret = memAlloc(bytesAllocated);
                ret.FillMemory(0, bytesAllocated);
                return(ret);
            }

            // Write to memory stream
            using (var ms = new NativeMemoryStream(1024, 1024)
            {
                CharSet = charSet
            })
            {
                ms.SetLength(ms.Position = prefixBytes);
                if (packing == StringListPackMethod.Packed)
                {
                    foreach (var s in values)
                    {
                        ms.WriteReference(s);
                    }
                    ms.WriteReference(null);
                }
                else
                {
                    foreach (var s in values)
                    {
                        if (string.IsNullOrEmpty(s))
                        {
                            throw new ArgumentException("Concatenated string arrays cannot contain empty or null strings.");
                        }
                        ms.Write(s);
                    }
                    ms.Write("");
                }
                ms.Flush();

                // Copy to newly allocated memory using memAlloc
                bytesAllocated = (int)ms.Length;
                var ret = memAlloc(bytesAllocated);
                ms.Pointer.CopyTo(ret, bytesAllocated);
                return(ret);
            }
        }
Пример #4
0
        public void StructTest()
        {
            using (var ms = new NativeMemoryStream(8, 16, 48))
            {
                Assert.That(() => ms.Write(256), Throws.Nothing);
                var isz = sizeof(int);
                Assert.That(ms.Length, Is.EqualTo(Len()));
                Assert.That(ms.Position, Is.EqualTo(Pos()));
                Assert.That(ms.Capacity, Is.EqualTo(8));
                Assert.That(() => ms.Write((ushort)256), Throws.Nothing);
                isz += sizeof(ushort);
                Assert.That(ms.Length, Is.EqualTo(Len()));
                Assert.That(ms.Position, Is.EqualTo(Pos()));
                Assert.That(ms.Capacity, Is.EqualTo(8));
                Assert.That(() => ms.Write((long)1), Throws.Nothing);
                isz += sizeof(long);
                Assert.That(ms.Length, Is.EqualTo(Len()));
                Assert.That(ms.Position, Is.EqualTo(Pos()));
                Assert.That(ms.Capacity, Is.EqualTo(24));
                var newGuid = Guid.NewGuid();
                Assert.That(() => ms.Write(newGuid), Throws.Nothing);
                isz += Marshal.SizeOf(typeof(Guid));
                Assert.That(ms.Length, Is.EqualTo(Len()));
                Assert.That(ms.Position, Is.EqualTo(Pos()));
                Assert.That(ms.Capacity, Is.EqualTo(40));

                Assert.That(ms.Length, Is.EqualTo(ms.Position));
                Assert.That(ms.Capacity, Is.GreaterThanOrEqualTo(ms.Length));

                Assert.That(() => ms.Write(new Unblittable()), Throws.Exception);

                TestContext.WriteLine(ms.Pointer.ToHexDumpString((int)ms.Length, 32));
                ms.Position = 0;

                Assert.That(() => ms.Read(null), Throws.ArgumentNullException);

                Assert.That(ms.Read <int>(), Is.EqualTo(256));
                Assert.That(ms.Read <ushort>(), Is.EqualTo(256));
                Assert.That(ms.Read <long>(), Is.EqualTo(1));
                Assert.That(ms.Read <Guid>(), Is.EqualTo(newGuid));

                int Len() => isz;
                int Pos() => isz;
            }
        }
Пример #5
0
        public void StringEnumTest()
        {
            var abc = new[] { "A", "B", "C" };
            var a_c = new[] { "A", null, "C" };

            using (var ms = new NativeMemoryStream(128, 128))
            {
                Assert.That(() => ms.Write(null, StringListPackMethod.Concatenated), Throws.Nothing);

                Assert.That(() => ms.Write(abc, StringListPackMethod.Concatenated), Throws.Nothing);
                var len = 14;
                var pos = len;
                Assert.That(len, Is.EqualTo(ms.Length).And.EqualTo(ms.Position));
                Assert.That(() => ms.Write(abc, StringListPackMethod.Packed), Throws.Nothing);
                Assert.That(len += 12 + IntPtr.Size * 3, Is.EqualTo(ms.Length));
                Assert.That(pos += IntPtr.Size * 3, Is.EqualTo(ms.Position));
                Assert.That(ms.Capacity, Is.EqualTo(128));
                Assert.That(() => ms.Write(a_c, StringListPackMethod.Concatenated), Throws.Exception);
                Assert.That(() => ms.Write(a_c, StringListPackMethod.Packed), Throws.Nothing);
                Assert.That(len += 8 + IntPtr.Size * 3, Is.EqualTo(ms.Length));
                Assert.That(pos += IntPtr.Size * 3, Is.EqualTo(ms.Position));
                Assert.That(ms.Capacity, Is.EqualTo(128));

                var testSz = 10;
                var strSz  = 50;
                var l      = new List <string>(testSz);
                for (var i = 0; i < testSz; i++)
                {
                    l.Add(new string('X', strSz));
                }
                Assert.That(() => ms.Write(l, StringListPackMethod.Concatenated), Throws.Nothing);
                var strLen = ((strSz + 1) * testSz + 1) * StringHelper.GetCharSize();
                Assert.That(len += strLen, Is.EqualTo(ms.Length));
                Assert.That(pos += strLen, Is.EqualTo(ms.Position));
                Assert.That(ms.Capacity, Is.GreaterThan(128));

                Assert.That(() => ms.Write(l, StringListPackMethod.Packed), Throws.Nothing);
                strLen           = (strSz + 1) * testSz * StringHelper.GetCharSize();
                Assert.That(len += strLen + IntPtr.Size * testSz, Is.EqualTo(ms.Length));
                Assert.That(pos += IntPtr.Size * testSz, Is.EqualTo(ms.Position));

                Assert.That(ms.Position, Is.LessThan(ms.Length));
                ms.Flush();
                Assert.That(ms.Position, Is.EqualTo(ms.Length));
                Assert.That(ms.Position, Is.LessThanOrEqualTo(ms.Capacity));

                ms.Position = 0;

                Assert.That(ms.ReadArray(typeof(string), abc.Length, false), Is.EquivalentTo(abc));
                Assert.That(ms.ReadArray(typeof(string), abc.Length, true), Is.EquivalentTo(abc));
                Assert.That(ms.ReadArray(typeof(string), a_c.Length, true), Is.EquivalentTo(a_c));
                Assert.That(ms.ReadArray <string>(l.Count, false), Is.EquivalentTo(l));
                Assert.That(ms.ReadArray <string>(l.Count, true), Is.EquivalentTo(l));

                l.Clear();
            }
        }
Пример #6
0
        public void StringTest()
        {
            using (var ms = new NativeMemoryStream(20))
            {
                Assert.That(() => ms.Write(""), Throws.Nothing);
                Assert.That(ms.Length, Is.EqualTo(2));
                Assert.That(ms.Position, Is.EqualTo(2));
                Assert.That(ms.Capacity, Is.EqualTo(20));
                Assert.That(() => ms.Write(testStr), Throws.Nothing);
                Assert.That(ms.Length, Is.EqualTo(24));
                Assert.That(ms.Position, Is.EqualTo(24));
                Assert.That(ms.Capacity, Is.GreaterThan(20));

                ms.Flush();
                ms.Position = 0;

                Assert.That(ms.Read <string>(), Is.EqualTo(string.Empty));
                Assert.That(ms.Position, Is.EqualTo(StringHelper.GetCharSize(ms.CharSet)));
                Assert.That(ms.Read <string>(), Is.EqualTo(testStr));
            }
        }
Пример #7
0
        public void DisposedTest()
        {
            var ms = new NativeMemoryStream(20);

            ms.Dispose();
            Assert.That(ms.CanRead, Is.False);
            Assert.That(ms.CanSeek, Is.False);
            Assert.That(ms.CanWrite, Is.False);
            Assert.That(() => ms.Capacity > 0, Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.Read <int>(), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.Read(typeof(int)), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.Read(new byte[3], 0, 3), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.ReadByte(), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.ReadArray(typeof(byte), 1, false), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.ReadReference <int>(), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.ReadReference <string>(), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.Seek(0, SeekOrigin.Begin), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.SetLength(20), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.Write("X"), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.Write("X", CharSet.Ansi), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.Write(new[] { "X", "Y" }), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.Write(new byte[3], 0, 3), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.Write(new[] { 1, 2 }), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.Write(256), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.WriteByte(1), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.WriteObject(Guid.NewGuid()), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.WriteReference("X"), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.WriteReference(256), Throws.InstanceOf <ObjectDisposedException>());
            Assert.That(() => ms.WriteReferenceObject(Guid.NewGuid()), Throws.InstanceOf <ObjectDisposedException>());
        }
Пример #8
0
        public void CapacityTest()
        {
            using (var ms = new NativeMemoryStream(10, 10, 20))
            {
                Assert.That(ms.Capacity, Is.EqualTo(10));
                ms.Write(Guid.NewGuid());
                Assert.That(ms.Capacity, Is.EqualTo(20));
                Assert.That(ms.Length, Is.EqualTo(16));

                Assert.That(() => ms.EnsureCapacity(-100), Throws.TypeOf <ArgumentOutOfRangeException>());
                Assert.That(() => ms.EnsureCapacity(40), Throws.TypeOf <ArgumentOutOfRangeException>());
            }
        }
Пример #9
0
        public void MixedReadWriteTest()
        {
            using (var m = new SafeHGlobalHandle(512))
            {
                var  str  = "Test1";
                var  guid = Guid.NewGuid();
                var  lVal = 1208L;
                byte b    = 18;
                using (var ms = new NativeMemoryStream(m)
                {
                    CharSet = CharSet.Unicode
                })
                {
                    Assert.That(() => ms.WriteReference(str), Throws.Nothing);
                    Assert.That(() => ms.Write(str), Throws.Nothing);
                    Assert.That(() => ms.WriteReference(guid), Throws.Nothing);
                    Assert.That(() => ms.Write(guid), Throws.Nothing);
                    Assert.That(() => ms.WriteReference(lVal), Throws.Nothing);
                    Assert.That(() => ms.Write(lVal), Throws.Nothing);
                    Assert.That(() => ms.WriteReference(b), Throws.Nothing);
                    Assert.That(() => ms.Write(b), Throws.Nothing);
                    Assert.That(() => ms.WriteReference(str), Throws.Nothing);

                    ms.Flush();
                    ms.Position = 0;

                    Assert.That(ms.ReadReference <string>(CharSet.Unicode), Is.EqualTo(str));
                    Assert.That(ms.Read <string>(CharSet.Unicode), Is.EqualTo(str));
                    Assert.That(ms.ReadReference <Guid>(), Is.EqualTo(guid));
                    Assert.That(ms.Read <Guid>(), Is.EqualTo(guid));
                    Assert.That(ms.ReadReference <long>(), Is.EqualTo(lVal));
                    Assert.That(ms.Read <long>(), Is.EqualTo(lVal));
                    Assert.That(ms.ReadReference <byte>(), Is.EqualTo(b));
                    Assert.That(ms.Read <byte>(), Is.EqualTo(b));
                    Assert.That(ms.ReadReference <string>(CharSet.Unicode), Is.EqualTo(str));
                }
            }
        }
Пример #10
0
        public void BytesTest()
        {
            using (var ms = new NativeMemoryStream(20, 10))
            {
                Assert.That(ms.Length, Is.EqualTo(0));
                Assert.That(() => ms.Write(null, 0, 0), Throws.ArgumentNullException);
                var bytes = new byte[] { 0, 0, 0, 0, 0, 0, 0, 3 };
                Assert.That(() => ms.Write(bytes, 1, 8), Throws.ArgumentException);
                Assert.That(() => ms.Write(bytes, -1, 8), Throws.TypeOf <ArgumentOutOfRangeException>());
                Assert.That(() => ms.Write(bytes, 1, -8), Throws.TypeOf <ArgumentOutOfRangeException>());
                Assert.That(() => ms.Write(bytes, 0, 8), Throws.Nothing);
                Assert.That(ms.Length, Is.EqualTo(8));
                Assert.That(ms.Position, Is.EqualTo(8));
                Assert.That(ms.Capacity, Is.EqualTo(20));
                var ones = new byte[22];
                for (var i = 0; i < ones.Length; i++)
                {
                    ones[i] = 1;
                }
                Array.ForEach(ones, b => b = 1);
                Assert.That(() => ms.Write(ones, 0, ones.Length), Throws.Nothing);
                Assert.That(ms.Length, Is.EqualTo(30));
                Assert.That(ms.Position, Is.EqualTo(30));
                Assert.That(ms.Capacity, Is.EqualTo(30));

                ms.Flush();
                ms.Position = 0;

                Assert.That(() => ms.Read(null, 0, 0), Throws.ArgumentNullException);
                var rbytes = new byte[bytes.Length];
                Assert.That(() => ms.Read(rbytes, 5, bytes.Length), Throws.ArgumentException);
                Assert.That(() => ms.Read(rbytes, -1, 1), Throws.InstanceOf <ArgumentOutOfRangeException>());
                Assert.That(() => ms.Read(rbytes, 0, -1), Throws.InstanceOf <ArgumentOutOfRangeException>());
                Assert.That(ms.Read(rbytes, 0, bytes.Length), Is.EqualTo(bytes.Length));
                Assert.That(rbytes, Is.EquivalentTo(bytes));
                Assert.That(ms.Position, Is.EqualTo(bytes.Length));
                rbytes = new byte[ms.Length];
                Assert.That(() => ms.Read(rbytes, (int)ms.Position, 40), Throws.ArgumentException);
                Assert.That(ms.Read(rbytes, (int)ms.Position, ones.Length), Is.EqualTo(ones.Length));
                for (var i = bytes.Length; i < ms.Position; i++)
                {
                    Assert.That(rbytes[i], Is.EqualTo(1));
                }
                Assert.That(() => ms.Read(rbytes, 0, 2), Throws.InstanceOf <ArgumentOutOfRangeException>());
            }
        }
Пример #11
0
        public void ReadWriteOnlyTest()
        {
            using (var m = new SafeHGlobalHandle(10))
                using (var ms = new NativeMemoryStream((IntPtr)m, m.Size))
                {
                    Assert.That(ms.CanWrite, Is.False);
                    Assert.That(ms.CanSeek, Is.True);
                    Assert.That(ms.CanRead, Is.True);

                    Assert.That(() => ms.Write("X"), Throws.InstanceOf <NotSupportedException>());
                    Assert.That(() => ms.Write("X", CharSet.Ansi), Throws.InstanceOf <NotSupportedException>());
                    Assert.That(() => ms.Write(new[] { "X", "Y" }), Throws.InstanceOf <NotSupportedException>());
                    Assert.That(() => ms.Write(new byte[3], 0, 3), Throws.InstanceOf <NotSupportedException>());
                    Assert.That(() => ms.Write(new[] { 1, 2 }), Throws.InstanceOf <NotSupportedException>());
                    Assert.That(() => ms.Write(256), Throws.InstanceOf <NotSupportedException>());
                    Assert.That(() => ms.WriteByte(1), Throws.InstanceOf <NotSupportedException>());
                    Assert.That(() => ms.WriteObject(Guid.NewGuid()), Throws.InstanceOf <NotSupportedException>());
                    Assert.That(() => ms.WriteReference("X"), Throws.InstanceOf <NotSupportedException>());
                    Assert.That(() => ms.WriteReference(256), Throws.InstanceOf <NotSupportedException>());
                    Assert.That(() => ms.WriteReferenceObject(Guid.NewGuid()), Throws.InstanceOf <NotSupportedException>());
                }

            using (var ms = new NativeMemoryStream(10, 10, 20, FileAccess.Write))
            {
                Assert.That(ms.CanWrite, Is.True);
                Assert.That(ms.CanSeek, Is.True);
                Assert.That(ms.CanRead, Is.False);

                Assert.That(() => ms.Read <int>(), Throws.InstanceOf <NotSupportedException>());
                Assert.That(() => ms.Read(typeof(int)), Throws.InstanceOf <NotSupportedException>());
                Assert.That(() => ms.Read(new byte[3], 0, 3), Throws.InstanceOf <NotSupportedException>());
                Assert.That(() => ms.ReadByte(), Throws.InstanceOf <NotSupportedException>());
                Assert.That(() => ms.ReadArray(typeof(byte), 1, false), Throws.InstanceOf <NotSupportedException>());
                Assert.That(() => ms.ReadReference <int>(), Throws.InstanceOf <NotSupportedException>());
                Assert.That(() => ms.ReadReference <string>(), Throws.InstanceOf <NotSupportedException>());
            }
        }