コード例 #1
0
        public void RefStringTest()
        {
            using (var ms = new NativeMemoryStream(20, 20))
            {
                Assert.That(() => ms.WriteReference(""), Throws.Nothing);
                Assert.That(ms.Length, Is.EqualTo(2 + IntPtr.Size));
                Assert.That(ms.Position, Is.EqualTo(IntPtr.Size));
                Assert.That(ms.Capacity, Is.EqualTo(20));
                Assert.That(() => ms.WriteReference(testStr), Throws.Nothing);
                Assert.That(ms.Length, Is.EqualTo(24 + IntPtr.Size * 2));
                Assert.That(ms.Position, Is.EqualTo(IntPtr.Size * 2));
                Assert.That(() => ms.WriteReference((string)null), Throws.Nothing);
                Assert.That(ms.Length, Is.EqualTo(24 + IntPtr.Size * 3));
                Assert.That(ms.Position, Is.EqualTo(IntPtr.Size * 3));
                var ptrs = ms.Pointer.ToArray <IntPtr>(3);
                Assert.That(ptrs, Is.EquivalentTo(new[] { IntPtr.Zero, IntPtr.Zero, IntPtr.Zero }));
                ms.Flush();
                Assert.That(ms.Capacity, Is.EqualTo(40));
                ptrs = ms.Pointer.ToArray <IntPtr>(3);
                Assert.That(ptrs, Is.Not.EquivalentTo(new[] { IntPtr.Zero, IntPtr.Zero, IntPtr.Zero }));
                Assert.That(StringHelper.GetString(ptrs[0], ms.CharSet), Is.EqualTo(""));
                Assert.That(StringHelper.GetString(ptrs[1], ms.CharSet), Is.EqualTo(testStr));
                Assert.That(StringHelper.GetString(ptrs[2], ms.CharSet), Is.Null);

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

                Assert.That(ms.ReadReference <string>(), Is.EqualTo(string.Empty));
                Assert.That(ms.Position, Is.EqualTo(IntPtr.Size));
                Assert.That(ms.ReadReference <string>(), Is.EqualTo(testStr));
                Assert.That(ms.ReadReference <string>(), Is.Null);
                Assert.That(ms.Read <string>(), Is.EqualTo(string.Empty));
                Assert.That(ms.Read <string>(), Is.EqualTo(testStr));
            }
        }
コード例 #2
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();
            }
        }
コード例 #3
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));
            }
        }
コード例 #4
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);
            }
        }
コード例 #5
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>());
            }
        }
コード例 #6
0
        public void RefStructTest()
        {
            using (var ms = new NativeMemoryStream(8, 16, 100))
            {
                Assert.That(() => ms.WriteReference(256), Throws.Nothing);
                var isz  = sizeof(int);
                var pcnt = 1;
                Assert.That(ms.Length, Is.EqualTo(Len()));
                Assert.That(ms.Position, Is.EqualTo(Pos()));
                Assert.That(ms.Capacity, Is.EqualTo(8));
                Assert.That(() => ms.WriteReference((ushort)256), Throws.Nothing);
                isz += sizeof(ushort); pcnt++;
                Assert.That(ms.Length, Is.EqualTo(Len()));
                Assert.That(ms.Position, Is.EqualTo(Pos()));
                Assert.That(ms.Capacity, Is.EqualTo(24));
                Assert.That(() => ms.WriteReference((long)1), Throws.Nothing);
                isz += sizeof(long); pcnt++;
                Assert.That(ms.Length, Is.EqualTo(Len()));
                Assert.That(ms.Position, Is.EqualTo(Pos()));
                Assert.That(ms.Capacity, Is.EqualTo(40));
                Assert.That(ms.Pointer.ToArray(IntPtr.Size == 4 ? typeof(uint) : typeof(ulong), pcnt), Is.EquivalentTo(new[] { 0, 0, 0 }));
                var newGuid = Guid.NewGuid();
                Assert.That(() => ms.WriteReference(newGuid), Throws.Nothing);
                isz += Marshal.SizeOf(typeof(Guid)); pcnt++;
                Assert.That(ms.Length, Is.EqualTo(Len()));
                Assert.That(ms.Position, Is.EqualTo(Pos()));
                Assert.That(ms.Capacity, Is.EqualTo(56));
                ms.Flush();
                Assert.That(ms.Length, Is.EqualTo(ms.Position));
                Assert.That(ms.Capacity, Is.GreaterThanOrEqualTo(ms.Length));
                var arr = ms.Pointer.ToArray(IntPtr.Size == 4 ? typeof(uint) : typeof(ulong), pcnt);
                Assert.That(arr, Is.Not.EquivalentTo(new[] { 0, 0, 0, 0 }));

                ms.Position = 0;

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

                int Len() => Pos() + isz;
                int Pos() => pcnt *IntPtr.Size;
            }
        }
コード例 #7
0
        /// <summary>
        /// Marshals data from a managed list of objects to an unmanaged block of memory allocated by the <paramref name="memAlloc"/> method.
        /// </summary>
        /// <param name="values">The enumerated list of objects to marshal.</param>
        /// <param name="memAlloc">
        /// The function that allocates the memory for the block of objects (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="referencePointers">
        /// if set to <see langword="true"/> the pointer will be processed by storing a reference to the value; if <see langword="false"/>,
        /// the pointer value will be directly inserted into the array of pointers.
        /// </param>
        /// <param name="charSet">The character set to use for strings.</param>
        /// <param name="prefixBytes">Number of bytes preceding the allocated objects.</param>
        /// <returns>Pointer to the allocated native (unmanaged) array of objects stored using the character set defined by <paramref name="charSet"/>.</returns>
        public static IntPtr MarshalObjectsToPtr(this IEnumerable <object> values, Func <int, IntPtr> memAlloc, out int bytesAllocated, bool referencePointers = false, CharSet charSet = CharSet.Auto, int prefixBytes = 0)
        {
            // Bail early if empty
            if (values is null || !values.Any())
            {
                bytesAllocated = prefixBytes + 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);
                foreach (var o in values)
                {
                    if (referencePointers)
                    {
                        ms.WriteReferenceObject(o);
                    }
                    else
                    {
                        ms.WriteObject(o);
                    }
                }
                if (referencePointers)
                {
                    ms.WriteReference(null);
                }
                ms.Flush();

                // Copy to newly allocated memory using memAlloc
                bytesAllocated = (int)ms.Length;
                var ret = memAlloc(bytesAllocated);
                ms.Pointer.CopyTo(ret, bytesAllocated);
                return(ret);
            }
        }
コード例 #8
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));
            }
        }
コード例 #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));
                }
            }
        }