e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
1862 lines
46 KiB
C#
1862 lines
46 KiB
C#
//
|
|
// Authors:
|
|
// Miguel de Icaza (miguel@novell.com)
|
|
//
|
|
// See the following url for documentation:
|
|
// http://www.mono-project.com/Mono_DataConvert
|
|
//
|
|
// Compilation Options:
|
|
// MONO_DATACONVERTER_PUBLIC:
|
|
// Makes the class public instead of the default internal.
|
|
//
|
|
// MONO_DATACONVERTER_STATIC_METHODS:
|
|
// Exposes the public static methods.
|
|
//
|
|
// TODO:
|
|
// Support for "DoubleWordsAreSwapped" for ARM devices
|
|
//
|
|
// Copyright (C) 2006 Novell, Inc (http://www.novell.com)
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
// a copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
// the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be
|
|
// included in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
//
|
|
using System;
|
|
using System.Collections;
|
|
using System.Text;
|
|
using System.Collections.Generic;
|
|
|
|
#pragma warning disable 3021
|
|
|
|
namespace Mono {
|
|
|
|
#if MONO_DATACONVERTER_PUBLIC
|
|
public
|
|
#endif
|
|
unsafe abstract class DataConverter {
|
|
|
|
// Disables the warning: CLS compliance checking will not be performed on
|
|
// `XXXX' because it is not visible from outside this assembly
|
|
#pragma warning disable 3019
|
|
static readonly DataConverter SwapConv = new SwapConverter ();
|
|
static readonly DataConverter CopyConv = new CopyConverter ();
|
|
|
|
public static readonly bool IsLittleEndian = BitConverter.IsLittleEndian;
|
|
|
|
public abstract double GetDouble (byte [] data, int index);
|
|
public abstract float GetFloat (byte [] data, int index);
|
|
public abstract long GetInt64 (byte [] data, int index);
|
|
public abstract int GetInt32 (byte [] data, int index);
|
|
|
|
public abstract short GetInt16 (byte [] data, int index);
|
|
|
|
[CLSCompliant (false)]
|
|
public abstract uint GetUInt32 (byte [] data, int index);
|
|
[CLSCompliant (false)]
|
|
public abstract ushort GetUInt16 (byte [] data, int index);
|
|
[CLSCompliant (false)]
|
|
public abstract ulong GetUInt64 (byte [] data, int index);
|
|
|
|
public abstract void PutBytes (byte [] dest, int destIdx, double value);
|
|
public abstract void PutBytes (byte [] dest, int destIdx, float value);
|
|
public abstract void PutBytes (byte [] dest, int destIdx, int value);
|
|
public abstract void PutBytes (byte [] dest, int destIdx, long value);
|
|
public abstract void PutBytes (byte [] dest, int destIdx, short value);
|
|
|
|
[CLSCompliant (false)]
|
|
public abstract void PutBytes (byte [] dest, int destIdx, ushort value);
|
|
[CLSCompliant (false)]
|
|
public abstract void PutBytes (byte [] dest, int destIdx, uint value);
|
|
[CLSCompliant (false)]
|
|
public abstract void PutBytes (byte [] dest, int destIdx, ulong value);
|
|
|
|
public byte[] GetBytes (double value)
|
|
{
|
|
byte [] ret = new byte [8];
|
|
PutBytes (ret, 0, value);
|
|
return ret;
|
|
}
|
|
|
|
public byte[] GetBytes (float value)
|
|
{
|
|
byte [] ret = new byte [4];
|
|
PutBytes (ret, 0, value);
|
|
return ret;
|
|
}
|
|
|
|
public byte[] GetBytes (int value)
|
|
{
|
|
byte [] ret = new byte [4];
|
|
PutBytes (ret, 0, value);
|
|
return ret;
|
|
}
|
|
|
|
public byte[] GetBytes (long value)
|
|
{
|
|
byte [] ret = new byte [8];
|
|
PutBytes (ret, 0, value);
|
|
return ret;
|
|
}
|
|
|
|
public byte[] GetBytes (short value)
|
|
{
|
|
byte [] ret = new byte [2];
|
|
PutBytes (ret, 0, value);
|
|
return ret;
|
|
}
|
|
|
|
[CLSCompliant (false)]
|
|
public byte[] GetBytes (ushort value)
|
|
{
|
|
byte [] ret = new byte [2];
|
|
PutBytes (ret, 0, value);
|
|
return ret;
|
|
}
|
|
|
|
[CLSCompliant (false)]
|
|
public byte[] GetBytes (uint value)
|
|
{
|
|
byte [] ret = new byte [4];
|
|
PutBytes (ret, 0, value);
|
|
return ret;
|
|
}
|
|
|
|
[CLSCompliant (false)]
|
|
public byte[] GetBytes (ulong value)
|
|
{
|
|
byte [] ret = new byte [8];
|
|
PutBytes (ret, 0, value);
|
|
return ret;
|
|
}
|
|
|
|
static public DataConverter LittleEndian {
|
|
get {
|
|
return BitConverter.IsLittleEndian ? CopyConv : SwapConv;
|
|
}
|
|
}
|
|
|
|
static public DataConverter BigEndian {
|
|
get {
|
|
return BitConverter.IsLittleEndian ? SwapConv : CopyConv;
|
|
}
|
|
}
|
|
|
|
static public DataConverter Native {
|
|
get {
|
|
return CopyConv;
|
|
}
|
|
}
|
|
|
|
static int Align (int current, int align)
|
|
{
|
|
return ((current + align - 1) / align) * align;
|
|
}
|
|
|
|
class PackContext {
|
|
// Buffer
|
|
public byte [] buffer;
|
|
int next;
|
|
|
|
public string description;
|
|
public int i; // position in the description
|
|
public DataConverter conv;
|
|
public int repeat;
|
|
|
|
//
|
|
// if align == -1, auto align to the size of the byte array
|
|
// if align == 0, do not do alignment
|
|
// Any other values aligns to that particular size
|
|
//
|
|
public int align;
|
|
|
|
public void Add (byte [] group)
|
|
{
|
|
//Console.WriteLine ("Adding {0} bytes to {1} (next={2}", group.Length,
|
|
// buffer == null ? "null" : buffer.Length.ToString (), next);
|
|
|
|
if (buffer == null){
|
|
buffer = group;
|
|
next = group.Length;
|
|
return;
|
|
}
|
|
if (align != 0){
|
|
if (align == -1)
|
|
next = Align (next, group.Length);
|
|
else
|
|
next = Align (next, align);
|
|
align = 0;
|
|
}
|
|
|
|
if (next + group.Length > buffer.Length){
|
|
byte [] nb = new byte [System.Math.Max (next, 16) * 2 + group.Length];
|
|
Array.Copy (buffer, nb, buffer.Length);
|
|
Array.Copy (group, 0, nb, next, group.Length);
|
|
next = next + group.Length;
|
|
buffer = nb;
|
|
} else {
|
|
Array.Copy (group, 0, buffer, next, group.Length);
|
|
next += group.Length;
|
|
}
|
|
}
|
|
|
|
public byte [] Get ()
|
|
{
|
|
if (buffer == null)
|
|
return new byte [0];
|
|
|
|
if (buffer.Length != next){
|
|
byte [] b = new byte [next];
|
|
Array.Copy (buffer, b, next);
|
|
return b;
|
|
}
|
|
return buffer;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Format includes:
|
|
// Control:
|
|
// ^ Switch to big endian encoding
|
|
// _ Switch to little endian encoding
|
|
// % Switch to host (native) encoding
|
|
// ! aligns the next data type to its natural boundary (for strings this is 4).
|
|
//
|
|
// Types:
|
|
// s Int16
|
|
// S UInt16
|
|
// i Int32
|
|
// I UInt32
|
|
// l Int64
|
|
// L UInt64
|
|
// f float
|
|
// d double
|
|
// b byte
|
|
// c 1-byte signed character
|
|
// C 1-byte unsigned character
|
|
// z8 string encoded as UTF8 with 1-byte null terminator
|
|
// z6 string encoded as UTF16 with 2-byte null terminator
|
|
// z7 string encoded as UTF7 with 1-byte null terminator
|
|
// zb string encoded as BigEndianUnicode with 2-byte null terminator
|
|
// z3 string encoded as UTF32 with 4-byte null terminator
|
|
// z4 string encoded as UTF32 big endian with 4-byte null terminator
|
|
// $8 string encoded as UTF8
|
|
// $6 string encoded as UTF16
|
|
// $7 string encoded as UTF7
|
|
// $b string encoded as BigEndianUnicode
|
|
// $3 string encoded as UTF32
|
|
// $4 string encoded as UTF-32 big endian encoding
|
|
// x null byte
|
|
//
|
|
// Repeats, these are prefixes:
|
|
// N a number between 1 and 9, indicates a repeat count (process N items
|
|
// with the following datatype
|
|
// [N] For numbers larger than 9, use brackets, for example [20]
|
|
// * Repeat the next data type until the arguments are exhausted
|
|
//
|
|
static public byte [] Pack (string description, params object [] args)
|
|
{
|
|
int argn = 0;
|
|
PackContext b = new PackContext ();
|
|
b.conv = CopyConv;
|
|
b.description = description;
|
|
|
|
for (b.i = 0; b.i < description.Length; ){
|
|
object oarg;
|
|
|
|
if (argn < args.Length)
|
|
oarg = args [argn];
|
|
else {
|
|
if (b.repeat != 0)
|
|
break;
|
|
|
|
oarg = null;
|
|
}
|
|
|
|
int save = b.i;
|
|
|
|
if (PackOne (b, oarg)){
|
|
argn++;
|
|
if (b.repeat > 0){
|
|
if (--b.repeat > 0)
|
|
b.i = save;
|
|
else
|
|
b.i++;
|
|
} else
|
|
b.i++;
|
|
} else
|
|
b.i++;
|
|
}
|
|
return b.Get ();
|
|
}
|
|
|
|
static public byte [] PackEnumerable (string description, IEnumerable args)
|
|
{
|
|
PackContext b = new PackContext ();
|
|
b.conv = CopyConv;
|
|
b.description = description;
|
|
|
|
IEnumerator enumerator = args.GetEnumerator ();
|
|
bool ok = enumerator.MoveNext ();
|
|
|
|
for (b.i = 0; b.i < description.Length; ){
|
|
object oarg;
|
|
|
|
if (ok)
|
|
oarg = enumerator.Current;
|
|
else {
|
|
if (b.repeat != 0)
|
|
break;
|
|
oarg = null;
|
|
}
|
|
|
|
int save = b.i;
|
|
|
|
if (PackOne (b, oarg)){
|
|
ok = enumerator.MoveNext ();
|
|
if (b.repeat > 0){
|
|
if (--b.repeat > 0)
|
|
b.i = save;
|
|
else
|
|
b.i++;
|
|
} else
|
|
b.i++;
|
|
} else
|
|
b.i++;
|
|
}
|
|
return b.Get ();
|
|
}
|
|
|
|
//
|
|
// Packs one datum `oarg' into the buffer `b', using the string format
|
|
// in `description' at position `i'
|
|
//
|
|
// Returns: true if we must pick the next object from the list
|
|
//
|
|
static bool PackOne (PackContext b, object oarg)
|
|
{
|
|
int n;
|
|
|
|
switch (b.description [b.i]){
|
|
case '^':
|
|
b.conv = BigEndian;
|
|
return false;
|
|
case '_':
|
|
b.conv = LittleEndian;
|
|
return false;
|
|
case '%':
|
|
b.conv = Native;
|
|
return false;
|
|
|
|
case '!':
|
|
b.align = -1;
|
|
return false;
|
|
|
|
case 'x':
|
|
b.Add (new byte [] { 0 });
|
|
return false;
|
|
|
|
// Type Conversions
|
|
case 'i':
|
|
b.Add (b.conv.GetBytes (Convert.ToInt32 (oarg)));
|
|
break;
|
|
|
|
case 'I':
|
|
b.Add (b.conv.GetBytes (Convert.ToUInt32 (oarg)));
|
|
break;
|
|
|
|
case 's':
|
|
b.Add (b.conv.GetBytes (Convert.ToInt16 (oarg)));
|
|
break;
|
|
|
|
case 'S':
|
|
b.Add (b.conv.GetBytes (Convert.ToUInt16 (oarg)));
|
|
break;
|
|
|
|
case 'l':
|
|
b.Add (b.conv.GetBytes (Convert.ToInt64 (oarg)));
|
|
break;
|
|
|
|
case 'L':
|
|
b.Add (b.conv.GetBytes (Convert.ToUInt64 (oarg)));
|
|
break;
|
|
|
|
case 'f':
|
|
b.Add (b.conv.GetBytes (Convert.ToSingle (oarg)));
|
|
break;
|
|
|
|
case 'd':
|
|
b.Add (b.conv.GetBytes (Convert.ToDouble (oarg)));
|
|
break;
|
|
|
|
case 'b':
|
|
b.Add (new byte [] { Convert.ToByte (oarg) });
|
|
break;
|
|
|
|
case 'c':
|
|
b.Add (new byte [] { (byte) (Convert.ToSByte (oarg)) });
|
|
break;
|
|
|
|
case 'C':
|
|
b.Add (new byte [] { Convert.ToByte (oarg) });
|
|
break;
|
|
|
|
// Repeat acount;
|
|
case '1': case '2': case '3': case '4': case '5':
|
|
case '6': case '7': case '8': case '9':
|
|
b.repeat = ((short) b.description [b.i]) - ((short) '0');
|
|
return false;
|
|
|
|
case '*':
|
|
b.repeat = Int32.MaxValue;
|
|
return false;
|
|
|
|
case '[':
|
|
int count = -1, j;
|
|
|
|
for (j = b.i+1; j < b.description.Length; j++){
|
|
if (b.description [j] == ']')
|
|
break;
|
|
n = ((short) b.description [j]) - ((short) '0');
|
|
if (n >= 0 && n <= 9){
|
|
if (count == -1)
|
|
count = n;
|
|
else
|
|
count = count * 10 + n;
|
|
}
|
|
}
|
|
if (count == -1)
|
|
throw new ArgumentException ("invalid size specification");
|
|
b.i = j;
|
|
b.repeat = count;
|
|
return false;
|
|
|
|
case '$': case 'z':
|
|
bool add_null = b.description [b.i] == 'z';
|
|
b.i++;
|
|
if (b.i >= b.description.Length)
|
|
throw new ArgumentException ("$ description needs a type specified", "description");
|
|
char d = b.description [b.i];
|
|
Encoding e;
|
|
|
|
switch (d){
|
|
case '8':
|
|
e = Encoding.UTF8;
|
|
n = 1;
|
|
break;
|
|
case '6':
|
|
e = Encoding.Unicode;
|
|
n = 2;
|
|
break;
|
|
case '7':
|
|
#if PCL
|
|
e = Encoding.GetEncoding ("utf-7");
|
|
#else
|
|
e = Encoding.UTF7;
|
|
#endif
|
|
n = 1;
|
|
break;
|
|
case 'b':
|
|
e = Encoding.BigEndianUnicode;
|
|
n = 2;
|
|
break;
|
|
case '3':
|
|
#if PCL
|
|
e = Encoding.GetEncoding ("utf-32");
|
|
#else
|
|
e = Encoding.GetEncoding (12000);
|
|
#endif
|
|
n = 4;
|
|
break;
|
|
case '4':
|
|
#if PCL
|
|
e = Encoding.GetEncoding ("utf-32BE");
|
|
#else
|
|
e = Encoding.GetEncoding (12001);
|
|
#endif
|
|
n = 4;
|
|
break;
|
|
|
|
default:
|
|
throw new ArgumentException ("Invalid format for $ specifier", "description");
|
|
}
|
|
if (b.align == -1)
|
|
b.align = 4;
|
|
b.Add (e.GetBytes (Convert.ToString (oarg)));
|
|
if (add_null)
|
|
b.Add (new byte [n]);
|
|
break;
|
|
default:
|
|
throw new ArgumentException (String.Format ("invalid format specified `{0}'",
|
|
b.description [b.i]));
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static bool Prepare (byte [] buffer, ref int idx, int size, ref bool align)
|
|
{
|
|
if (align){
|
|
idx = Align (idx, size);
|
|
align = false;
|
|
}
|
|
if (idx + size > buffer.Length){
|
|
idx = buffer.Length;
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static public IList Unpack (string description, byte [] buffer, int startIndex)
|
|
{
|
|
DataConverter conv = CopyConv;
|
|
var result = new List<object> ();
|
|
int idx = startIndex;
|
|
bool align = false;
|
|
int repeat = 0, n;
|
|
|
|
for (int i = 0; i < description.Length && idx < buffer.Length; ){
|
|
int save = i;
|
|
|
|
switch (description [i]){
|
|
case '^':
|
|
conv = BigEndian;
|
|
break;
|
|
case '_':
|
|
conv = LittleEndian;
|
|
break;
|
|
case '%':
|
|
conv = Native;
|
|
break;
|
|
case 'x':
|
|
idx++;
|
|
break;
|
|
|
|
case '!':
|
|
align = true;
|
|
break;
|
|
|
|
// Type Conversions
|
|
case 'i':
|
|
if (Prepare (buffer, ref idx, 4, ref align)){
|
|
result.Add (conv.GetInt32 (buffer, idx));
|
|
idx += 4;
|
|
}
|
|
break;
|
|
|
|
case 'I':
|
|
if (Prepare (buffer, ref idx, 4, ref align)){
|
|
result.Add (conv.GetUInt32 (buffer, idx));
|
|
idx += 4;
|
|
}
|
|
break;
|
|
|
|
case 's':
|
|
if (Prepare (buffer, ref idx, 2, ref align)){
|
|
result.Add (conv.GetInt16 (buffer, idx));
|
|
idx += 2;
|
|
}
|
|
break;
|
|
|
|
case 'S':
|
|
if (Prepare (buffer, ref idx, 2, ref align)){
|
|
result.Add (conv.GetUInt16 (buffer, idx));
|
|
idx += 2;
|
|
}
|
|
break;
|
|
|
|
case 'l':
|
|
if (Prepare (buffer, ref idx, 8, ref align)){
|
|
result.Add (conv.GetInt64 (buffer, idx));
|
|
idx += 8;
|
|
}
|
|
break;
|
|
|
|
case 'L':
|
|
if (Prepare (buffer, ref idx, 8, ref align)){
|
|
result.Add (conv.GetUInt64 (buffer, idx));
|
|
idx += 8;
|
|
}
|
|
break;
|
|
|
|
case 'f':
|
|
if (Prepare (buffer, ref idx, 4, ref align)){
|
|
result.Add (conv.GetFloat (buffer, idx));
|
|
idx += 4;
|
|
}
|
|
break;
|
|
|
|
case 'd':
|
|
if (Prepare (buffer, ref idx, 8, ref align)){
|
|
result.Add (conv.GetDouble (buffer, idx));
|
|
idx += 8;
|
|
}
|
|
break;
|
|
|
|
case 'b':
|
|
if (Prepare (buffer, ref idx, 1, ref align)){
|
|
result.Add (buffer [idx]);
|
|
idx++;
|
|
}
|
|
break;
|
|
|
|
case 'c': case 'C':
|
|
if (Prepare (buffer, ref idx, 1, ref align)){
|
|
char c;
|
|
|
|
if (description [i] == 'c')
|
|
c = ((char) ((sbyte)buffer [idx]));
|
|
else
|
|
c = ((char) ((byte)buffer [idx]));
|
|
|
|
result.Add (c);
|
|
idx++;
|
|
}
|
|
break;
|
|
|
|
// Repeat acount;
|
|
case '1': case '2': case '3': case '4': case '5':
|
|
case '6': case '7': case '8': case '9':
|
|
repeat = ((short) description [i]) - ((short) '0');
|
|
save = i + 1;
|
|
break;
|
|
|
|
case '*':
|
|
repeat = Int32.MaxValue;
|
|
break;
|
|
|
|
case '[':
|
|
int count = -1, j;
|
|
|
|
for (j = i+1; j < description.Length; j++){
|
|
if (description [j] == ']')
|
|
break;
|
|
n = ((short) description [j]) - ((short) '0');
|
|
if (n >= 0 && n <= 9){
|
|
if (count == -1)
|
|
count = n;
|
|
else
|
|
count = count * 10 + n;
|
|
}
|
|
}
|
|
if (count == -1)
|
|
throw new ArgumentException ("invalid size specification");
|
|
i = j;
|
|
save = i + 1;
|
|
repeat = count;
|
|
break;
|
|
|
|
case '$': case 'z':
|
|
// bool with_null = description [i] == 'z';
|
|
i++;
|
|
if (i >= description.Length)
|
|
throw new ArgumentException ("$ description needs a type specified", "description");
|
|
char d = description [i];
|
|
Encoding e;
|
|
if (align){
|
|
idx = Align (idx, 4);
|
|
align = false;
|
|
}
|
|
if (idx >= buffer.Length)
|
|
break;
|
|
|
|
switch (d){
|
|
case '8':
|
|
e = Encoding.UTF8;
|
|
n = 1;
|
|
break;
|
|
case '6':
|
|
e = Encoding.Unicode;
|
|
n = 2;
|
|
break;
|
|
case '7':
|
|
#if PCL
|
|
e = Encoding.GetEncoding ("utf-7");
|
|
#else
|
|
e = Encoding.UTF7;
|
|
#endif
|
|
n = 1;
|
|
break;
|
|
case 'b':
|
|
e = Encoding.BigEndianUnicode;
|
|
n = 2;
|
|
break;
|
|
case '3':
|
|
#if PCL
|
|
e = Encoding.GetEncoding ("utf-32");
|
|
#else
|
|
e = Encoding.GetEncoding (12000);
|
|
#endif
|
|
n = 4;
|
|
break;
|
|
case '4':
|
|
#if PCL
|
|
e = Encoding.GetEncoding ("utf-32BE");
|
|
#else
|
|
e = Encoding.GetEncoding (12001);
|
|
#endif
|
|
n = 4;
|
|
break;
|
|
|
|
default:
|
|
throw new ArgumentException ("Invalid format for $ specifier", "description");
|
|
}
|
|
int k = idx;
|
|
switch (n){
|
|
case 1:
|
|
for (; k < buffer.Length && buffer [k] != 0; k++)
|
|
;
|
|
result.Add (e.GetChars (buffer, idx, k-idx));
|
|
if (k == buffer.Length)
|
|
idx = k;
|
|
else
|
|
idx = k+1;
|
|
break;
|
|
|
|
case 2:
|
|
for (; k < buffer.Length; k++){
|
|
if (k+1 == buffer.Length){
|
|
k++;
|
|
break;
|
|
}
|
|
if (buffer [k] == 0 && buffer [k+1] == 0)
|
|
break;
|
|
}
|
|
result.Add (e.GetChars (buffer, idx, k-idx));
|
|
if (k == buffer.Length)
|
|
idx = k;
|
|
else
|
|
idx = k+2;
|
|
break;
|
|
|
|
case 4:
|
|
for (; k < buffer.Length; k++){
|
|
if (k+3 >= buffer.Length){
|
|
k = buffer.Length;
|
|
break;
|
|
}
|
|
if (buffer[k]==0 && buffer[k+1] == 0 && buffer[k+2] == 0 && buffer[k+3]== 0)
|
|
break;
|
|
}
|
|
result.Add (e.GetChars (buffer, idx, k-idx));
|
|
if (k == buffer.Length)
|
|
idx = k;
|
|
else
|
|
idx = k+4;
|
|
break;
|
|
}
|
|
break;
|
|
default:
|
|
throw new ArgumentException (String.Format ("invalid format specified `{0}'",
|
|
description [i]));
|
|
}
|
|
|
|
if (repeat > 0){
|
|
if (--repeat > 0)
|
|
i = save;
|
|
} else
|
|
i++;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
internal void Check (byte [] dest, int destIdx, int size)
|
|
{
|
|
if (dest == null)
|
|
throw new ArgumentNullException ("dest");
|
|
if (destIdx < 0 || destIdx > dest.Length - size)
|
|
throw new ArgumentException ("destIdx");
|
|
}
|
|
|
|
class CopyConverter : DataConverter {
|
|
public override double GetDouble (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
double ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 8; i++)
|
|
b [i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override ulong GetUInt64 (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
ulong ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 8; i++)
|
|
b [i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override long GetInt64 (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
long ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 8; i++)
|
|
b [i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override float GetFloat (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
float ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
b [i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override int GetInt32 (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
int ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
b [i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override uint GetUInt32 (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
uint ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
b [i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override short GetInt16 (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 2)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
short ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 2; i++)
|
|
b [i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override ushort GetUInt16 (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 2)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
ushort ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 2; i++)
|
|
b [i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, double value)
|
|
{
|
|
Check (dest, destIdx, 8);
|
|
fixed (byte *target = &dest [destIdx]){
|
|
long *source = (long *) &value;
|
|
|
|
*((long *)target) = *source;
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, float value)
|
|
{
|
|
Check (dest, destIdx, 4);
|
|
fixed (byte *target = &dest [destIdx]){
|
|
uint *source = (uint *) &value;
|
|
|
|
*((uint *)target) = *source;
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, int value)
|
|
{
|
|
Check (dest, destIdx, 4);
|
|
fixed (byte *target = &dest [destIdx]){
|
|
uint *source = (uint *) &value;
|
|
|
|
*((uint *)target) = *source;
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, uint value)
|
|
{
|
|
Check (dest, destIdx, 4);
|
|
fixed (byte *target = &dest [destIdx]){
|
|
uint *source = (uint *) &value;
|
|
|
|
*((uint *)target) = *source;
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, long value)
|
|
{
|
|
Check (dest, destIdx, 8);
|
|
fixed (byte *target = &dest [destIdx]){
|
|
long *source = (long *) &value;
|
|
|
|
*((long*)target) = *source;
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, ulong value)
|
|
{
|
|
Check (dest, destIdx, 8);
|
|
fixed (byte *target = &dest [destIdx]){
|
|
ulong *source = (ulong *) &value;
|
|
|
|
*((ulong *) target) = *source;
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, short value)
|
|
{
|
|
Check (dest, destIdx, 2);
|
|
fixed (byte *target = &dest [destIdx]){
|
|
ushort *source = (ushort *) &value;
|
|
|
|
*((ushort *)target) = *source;
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, ushort value)
|
|
{
|
|
Check (dest, destIdx, 2);
|
|
fixed (byte *target = &dest [destIdx]){
|
|
ushort *source = (ushort *) &value;
|
|
|
|
*((ushort *)target) = *source;
|
|
}
|
|
}
|
|
}
|
|
|
|
class SwapConverter : DataConverter {
|
|
public override double GetDouble (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
double ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 8; i++)
|
|
b [7-i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override ulong GetUInt64 (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
ulong ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 8; i++)
|
|
b [7-i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override long GetInt64 (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
long ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 8; i++)
|
|
b [7-i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override float GetFloat (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
float ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
b [3-i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override int GetInt32 (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
int ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
b [3-i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override uint GetUInt32 (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
uint ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
b [3-i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override short GetInt16 (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 2)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
short ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 2; i++)
|
|
b [1-i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override ushort GetUInt16 (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 2)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
ushort ret;
|
|
byte *b = (byte *)&ret;
|
|
|
|
for (int i = 0; i < 2; i++)
|
|
b [1-i] = data [index+i];
|
|
|
|
return ret;
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, double value)
|
|
{
|
|
Check (dest, destIdx, 8);
|
|
|
|
fixed (byte *target = &dest [destIdx]){
|
|
byte *source = (byte *) &value;
|
|
|
|
for (int i = 0; i < 8; i++)
|
|
target [i] = source [7-i];
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, float value)
|
|
{
|
|
Check (dest, destIdx, 4);
|
|
|
|
fixed (byte *target = &dest [destIdx]){
|
|
byte *source = (byte *) &value;
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
target [i] = source [3-i];
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, int value)
|
|
{
|
|
Check (dest, destIdx, 4);
|
|
|
|
fixed (byte *target = &dest [destIdx]){
|
|
byte *source = (byte *) &value;
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
target [i] = source [3-i];
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, uint value)
|
|
{
|
|
Check (dest, destIdx, 4);
|
|
|
|
fixed (byte *target = &dest [destIdx]){
|
|
byte *source = (byte *) &value;
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
target [i] = source [3-i];
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, long value)
|
|
{
|
|
Check (dest, destIdx, 8);
|
|
|
|
fixed (byte *target = &dest [destIdx]){
|
|
byte *source = (byte *) &value;
|
|
|
|
for (int i = 0; i < 8; i++)
|
|
target [i] = source [7-i];
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, ulong value)
|
|
{
|
|
Check (dest, destIdx, 8);
|
|
|
|
fixed (byte *target = &dest [destIdx]){
|
|
byte *source = (byte *) &value;
|
|
|
|
for (int i = 0; i < 8; i++)
|
|
target [i] = source [7-i];
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, short value)
|
|
{
|
|
Check (dest, destIdx, 2);
|
|
|
|
fixed (byte *target = &dest [destIdx]){
|
|
byte *source = (byte *) &value;
|
|
|
|
for (int i = 0; i < 2; i++)
|
|
target [i] = source [1-i];
|
|
}
|
|
}
|
|
|
|
public override void PutBytes (byte [] dest, int destIdx, ushort value)
|
|
{
|
|
Check (dest, destIdx, 2);
|
|
|
|
fixed (byte *target = &dest [destIdx]){
|
|
byte *source = (byte *) &value;
|
|
|
|
for (int i = 0; i < 2; i++)
|
|
target [i] = source [1-i];
|
|
}
|
|
}
|
|
}
|
|
|
|
#if MONO_DATACONVERTER_STATIC_METHODS
|
|
static unsafe void PutBytesLE (byte *dest, byte *src, int count)
|
|
{
|
|
int i = 0;
|
|
|
|
if (BitConverter.IsLittleEndian){
|
|
for (; i < count; i++)
|
|
*dest++ = *src++;
|
|
} else {
|
|
dest += count;
|
|
for (; i < count; i++)
|
|
*(--dest) = *src++;
|
|
}
|
|
}
|
|
|
|
static unsafe void PutBytesBE (byte *dest, byte *src, int count)
|
|
{
|
|
int i = 0;
|
|
|
|
if (BitConverter.IsLittleEndian){
|
|
dest += count;
|
|
for (; i < count; i++)
|
|
*(--dest) = *src++;
|
|
} else {
|
|
for (; i < count; i++)
|
|
*dest++ = *src++;
|
|
}
|
|
}
|
|
|
|
static unsafe void PutBytesNative (byte *dest, byte *src, int count)
|
|
{
|
|
int i = 0;
|
|
|
|
for (; i < count; i++)
|
|
dest [i-count] = *src++;
|
|
}
|
|
|
|
static public unsafe double DoubleFromLE (byte[] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
double ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesLE ((byte *) &ret, src, 8);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe float FloatFromLE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
float ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesLE ((byte *) &ret, src, 4);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe long Int64FromLE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
long ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesLE ((byte *) &ret, src, 8);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe ulong UInt64FromLE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
ulong ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesLE ((byte *) &ret, src, 8);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe int Int32FromLE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
int ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesLE ((byte *) &ret, src, 4);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe uint UInt32FromLE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
uint ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesLE ((byte *) &ret, src, 4);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe short Int16FromLE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 2)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
short ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesLE ((byte *) &ret, src, 2);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe ushort UInt16FromLE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 2)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
ushort ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesLE ((byte *) &ret, src, 2);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe double DoubleFromBE (byte[] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
double ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesBE ((byte *) &ret, src, 8);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe float FloatFromBE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
float ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesBE ((byte *) &ret, src, 4);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe long Int64FromBE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
long ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesBE ((byte *) &ret, src, 8);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe ulong UInt64FromBE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
ulong ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesBE ((byte *) &ret, src, 8);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe int Int32FromBE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
int ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesBE ((byte *) &ret, src, 4);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe uint UInt32FromBE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
uint ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesBE ((byte *) &ret, src, 4);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe short Int16FromBE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 2)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
short ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesBE ((byte *) &ret, src, 2);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe ushort UInt16FromBE (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 2)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
ushort ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesBE ((byte *) &ret, src, 2);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe double DoubleFromNative (byte[] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
double ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesNative ((byte *) &ret, src, 8);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe float FloatFromNative (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
float ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesNative ((byte *) &ret, src, 4);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe long Int64FromNative (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
long ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesNative ((byte *) &ret, src, 8);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe ulong UInt64FromNative (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 8)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
ulong ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesNative ((byte *) &ret, src, 8);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe int Int32FromNative (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
int ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesNative ((byte *) &ret, src, 4);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe uint UInt32FromNative (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 4)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
uint ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesNative ((byte *) &ret, src, 4);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe short Int16FromNative (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 2)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
short ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesNative ((byte *) &ret, src, 2);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static public unsafe ushort UInt16FromNative (byte [] data, int index)
|
|
{
|
|
if (data == null)
|
|
throw new ArgumentNullException ("data");
|
|
if (data.Length - index < 2)
|
|
throw new ArgumentException ("index");
|
|
if (index < 0)
|
|
throw new ArgumentException ("index");
|
|
|
|
ushort ret;
|
|
fixed (byte *src = &data[index]){
|
|
PutBytesNative ((byte *) &ret, src, 2);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
unsafe static byte[] GetBytesPtr (byte *ptr, int count)
|
|
{
|
|
byte [] ret = new byte [count];
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
ret [i] = ptr [i];
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
unsafe static byte[] GetBytesSwap (bool swap, byte *ptr, int count)
|
|
{
|
|
byte [] ret = new byte [count];
|
|
|
|
if (swap){
|
|
int t = count-1;
|
|
for (int i = 0; i < count; i++) {
|
|
ret [t-i] = ptr [i];
|
|
}
|
|
} else {
|
|
for (int i = 0; i < count; i++) {
|
|
ret [i] = ptr [i];
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesNative (bool value)
|
|
{
|
|
return GetBytesPtr ((byte *) &value, 1);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesNative (char value)
|
|
{
|
|
return GetBytesPtr ((byte *) &value, 2);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesNative (short value)
|
|
{
|
|
return GetBytesPtr ((byte *) &value, 2);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesNative (int value)
|
|
{
|
|
return GetBytesPtr ((byte *) &value, 4);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesNative (long value)
|
|
{
|
|
return GetBytesPtr ((byte *) &value, 8);
|
|
}
|
|
|
|
[CLSCompliant (false)]
|
|
unsafe public static byte[] GetBytesNative (ushort value)
|
|
{
|
|
return GetBytesPtr ((byte *) &value, 2);
|
|
}
|
|
|
|
[CLSCompliant (false)]
|
|
unsafe public static byte[] GetBytesNative (uint value)
|
|
{
|
|
return GetBytesPtr ((byte *) &value, 4);
|
|
}
|
|
|
|
[CLSCompliant (false)]
|
|
unsafe public static byte[] GetBytesNative (ulong value)
|
|
{
|
|
return GetBytesPtr ((byte *) &value, 8);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesNative (float value)
|
|
{
|
|
return GetBytesPtr ((byte *) &value, 4);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesNative (double value)
|
|
{
|
|
return GetBytesPtr ((byte *) &value, 8);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesLE (bool value)
|
|
{
|
|
return GetBytesSwap (!BitConverter.IsLittleEndian, (byte *) &value, 1);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesLE (char value)
|
|
{
|
|
return GetBytesSwap (!BitConverter.IsLittleEndian, (byte *) &value, 2);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesLE (short value)
|
|
{
|
|
return GetBytesSwap (!BitConverter.IsLittleEndian, (byte *) &value, 2);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesLE (int value)
|
|
{
|
|
return GetBytesSwap (!BitConverter.IsLittleEndian, (byte *) &value, 4);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesLE (long value)
|
|
{
|
|
return GetBytesSwap (!BitConverter.IsLittleEndian, (byte *) &value, 8);
|
|
}
|
|
|
|
[CLSCompliant (false)]
|
|
unsafe public static byte[] GetBytesLE (ushort value)
|
|
{
|
|
return GetBytesSwap (!BitConverter.IsLittleEndian, (byte *) &value, 2);
|
|
}
|
|
|
|
[CLSCompliant (false)]
|
|
unsafe public static byte[] GetBytesLE (uint value)
|
|
{
|
|
return GetBytesSwap (!BitConverter.IsLittleEndian, (byte *) &value, 4);
|
|
}
|
|
|
|
[CLSCompliant (false)]
|
|
unsafe public static byte[] GetBytesLE (ulong value)
|
|
{
|
|
return GetBytesSwap (!BitConverter.IsLittleEndian, (byte *) &value, 8);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesLE (float value)
|
|
{
|
|
return GetBytesSwap (!BitConverter.IsLittleEndian, (byte *) &value, 4);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesLE (double value)
|
|
{
|
|
return GetBytesSwap (!BitConverter.IsLittleEndian, (byte *) &value, 8);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesBE (bool value)
|
|
{
|
|
return GetBytesSwap (BitConverter.IsLittleEndian, (byte *) &value, 1);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesBE (char value)
|
|
{
|
|
return GetBytesSwap (BitConverter.IsLittleEndian, (byte *) &value, 2);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesBE (short value)
|
|
{
|
|
return GetBytesSwap (BitConverter.IsLittleEndian, (byte *) &value, 2);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesBE (int value)
|
|
{
|
|
return GetBytesSwap (BitConverter.IsLittleEndian, (byte *) &value, 4);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesBE (long value)
|
|
{
|
|
return GetBytesSwap (BitConverter.IsLittleEndian, (byte *) &value, 8);
|
|
}
|
|
|
|
[CLSCompliant (false)]
|
|
unsafe public static byte[] GetBytesBE (ushort value)
|
|
{
|
|
return GetBytesSwap (BitConverter.IsLittleEndian, (byte *) &value, 2);
|
|
}
|
|
|
|
[CLSCompliant (false)]
|
|
unsafe public static byte[] GetBytesBE (uint value)
|
|
{
|
|
return GetBytesSwap (BitConverter.IsLittleEndian, (byte *) &value, 4);
|
|
}
|
|
|
|
[CLSCompliant (false)]
|
|
unsafe public static byte[] GetBytesBE (ulong value)
|
|
{
|
|
return GetBytesSwap (BitConverter.IsLittleEndian, (byte *) &value, 8);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesBE (float value)
|
|
{
|
|
return GetBytesSwap (BitConverter.IsLittleEndian, (byte *) &value, 4);
|
|
}
|
|
|
|
unsafe public static byte[] GetBytesBE (double value)
|
|
{
|
|
return GetBytesSwap (BitConverter.IsLittleEndian, (byte *) &value, 8);
|
|
}
|
|
#endif
|
|
|
|
}
|
|
}
|