Jo Shields a575963da9 Imported Upstream version 3.6.0
Former-commit-id: da6be194a6b1221998fc28233f2503bd61dd9d14
2014-08-13 10:39:27 +01:00

1011 lines
25 KiB
C#

//
// Copyright (C) 2008 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.Reflection;
#if !FULL_AOT_RUNTIME
using System.Reflection.Emit;
#endif
namespace System.Data.Common
{
internal abstract class DataContainer {
BitArray null_values;
System.Type _type;
DataColumn _column;
// implementing class protocol
protected internal abstract object GetValue (int index);
internal abstract long GetInt64 (int index);
// used to set the array value to something neutral when the corresponding item is null (in the database sense)
// note: we don't actually ever look at the value written there, but the GC may like us to avoid keeping stale
// values in the array.
protected abstract void ZeroOut (int index);
protected abstract void SetValue (int index, object value);
protected abstract void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field);
protected abstract void DoCopyValue (DataContainer from, int from_index, int to_index);
protected abstract int DoCompareValues (int index1, int index2);
protected abstract void Resize (int length);
internal object this [int index] {
get { return IsNull (index) ? DBNull.Value : GetValue (index); }
set {
if (value == null) {
// Table might not have a default values row to copy from
if (Column.Table.DefaultValuesRowIndex == -1) {
ZeroOut (index);
null_values [index] = true;
} else {
CopyValue (Column.Table.DefaultValuesRowIndex, index);
}
return;
}
bool is_dbnull = value == DBNull.Value;
if (is_dbnull)
ZeroOut (index);
else
SetValue (index, value);
null_values [index] = is_dbnull;
}
}
internal int Capacity {
get { return null_values != null ? null_values.Count : 0; }
set {
int old_capacity = Capacity;
if (value == old_capacity)
return;
if (null_values == null)
null_values = new BitArray (value);
else
null_values.Length = value;
Resize (value);
}
}
internal Type Type {
get { return _type; }
}
protected DataColumn Column {
get { return _column; }
}
internal static DataContainer Create (Type type, DataColumn column)
{
DataContainer container;
switch (Type.GetTypeCode(type)) {
case TypeCode.Int16:
container = new Int16DataContainer ();
break;
case TypeCode.Int32:
container = new Int32DataContainer ();
break;
case TypeCode.Int64:
container = new Int64DataContainer ();
break;
case TypeCode.String:
container = new StringDataContainer ();
break;
case TypeCode.Boolean:
container = new BitDataContainer ();
break;
case TypeCode.Byte:
container = new ByteDataContainer ();
break;
case TypeCode.Char:
container = new CharDataContainer ();
break;
case TypeCode.Double:
container = new DoubleDataContainer ();
break;
case TypeCode.SByte:
container = new SByteDataContainer ();
break;
case TypeCode.Single:
container = new SingleDataContainer ();
break;
case TypeCode.UInt16:
container = new UInt16DataContainer ();
break;
case TypeCode.UInt32:
container = new UInt32DataContainer ();
break;
case TypeCode.UInt64:
container = new UInt64DataContainer ();
break;
case TypeCode.DateTime:
container = new DateTimeDataContainer ();
break;
case TypeCode.Decimal:
container = new DecimalDataContainer ();
break;
default:
container = new ObjectDataContainer ();
break;
}
container._type = type;
container._column = column;
return container;
}
internal static object GetExplicitValue (object value)
{
Type valueType = value.GetType ();
MethodInfo method = valueType.GetMethod ("op_Explicit", new Type[]{valueType});
if (method != null)
return (method.Invoke (value, new object[]{value}));
return null;
}
internal object GetContainerData (object value)
{
object obj;
TypeCode tc;
if (value == null)
return null;
if (_type.IsInstanceOfType (value)) {
return value;
} else if ((tc = Type.GetTypeCode (_type)) == TypeCode.String) {
return (Convert.ToString (value));
} else if (value is IConvertible) {
switch (tc) {
case TypeCode.Int16:
return (Convert.ToInt16 (value));
case TypeCode.Int32:
return (Convert.ToInt32 (value));
case TypeCode.Int64:
return (Convert.ToInt64 (value));
case TypeCode.Boolean:
return (Convert.ToBoolean (value));
case TypeCode.Byte:
return (Convert.ToByte (value));
case TypeCode.Char:
return (Convert.ToChar (value));
case TypeCode.Double:
return (Convert.ToDouble (value));
case TypeCode.SByte:
return (Convert.ToSByte (value));
case TypeCode.Single:
return (Convert.ToSingle (value));
case TypeCode.UInt16:
return (Convert.ToUInt16 (value));
case TypeCode.UInt32:
return (Convert.ToUInt32 (value));
case TypeCode.UInt64:
return (Convert.ToUInt64 (value));
case TypeCode.DateTime:
if (value == DBNull.Value)
return DBNull.Value;
return (Convert.ToDateTime (value));
case TypeCode.Decimal:
return (Convert.ToDecimal (value));
default:
throw new InvalidCastException (string.Format ("Cannot convert from {0} to {1}", value.GetType ().FullName, _type.FullName));
}
} else if ((obj = GetExplicitValue (value)) != null) {
return (obj);
} else {
throw new InvalidCastException ();
}
}
internal bool IsNull (int index)
{
return null_values == null || null_values [index];
}
internal void FillValues (int fromIndex)
{
for (int i = 0; i < Capacity; i++)
CopyValue (fromIndex, i);
}
internal void CopyValue (int from_index, int to_index)
{
CopyValue (this, from_index, to_index);
}
internal void CopyValue (DataContainer from, int from_index, int to_index)
{
DoCopyValue (from, from_index, to_index);
null_values [to_index] = from.null_values [from_index];
}
internal void SetItemFromDataRecord (int index, IDataRecord record, int field)
{
if (record.IsDBNull (field))
this [index] = DBNull.Value;
else if (record is ISafeDataRecord)
SetValueFromSafeDataRecord (index, (ISafeDataRecord) record, field);
else
this [index] = record.GetValue (field);
}
internal int CompareValues (int index1, int index2)
{
bool null1 = IsNull (index1);
bool null2 = IsNull (index2);
if (null1 == null2)
return null1 ? 0 : DoCompareValues (index1, index2);
return null1 ? -1 : 1;
}
}
sealed class BitDataContainer : DataContainer {
BitArray _values;
protected internal override object GetValue (int index)
{
return _values [index];
}
protected override void ZeroOut (int index)
{
_values [index] = false;
}
protected override void SetValue (int index, object value)
{
_values [index] = (bool) GetContainerData (value);
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
_values [index] = record.GetBooleanSafe (field);
}
protected override void DoCopyValue (DataContainer from, int from_index, int to_index)
{
_values [to_index] = (bool) GetContainerData (from.GetValue (from_index));
}
protected override int DoCompareValues (int index1, int index2)
{
bool val1 = _values [index1];
bool val2 = _values [index2];
return val1 == val2 ? 0 : val1 ? 1 : -1;
}
protected override void Resize (int size)
{
if (_values == null)
_values = new BitArray (size);
else
_values.Length = size;
}
internal override long GetInt64 (int index)
{
return Convert.ToInt64 (_values [index]);
}
}
sealed class CharDataContainer : DataContainer {
char [] _values;
protected internal override object GetValue (int index)
{
return _values [index];
}
protected override void ZeroOut (int index)
{
_values [index] = '\0';
}
protected override void SetValue (int index, object value)
{
_values [index] = (char) GetContainerData (value);
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
_values [index] = record.GetCharSafe (field);
}
protected override void DoCopyValue (DataContainer from, int from_index, int to_index)
{
_values [to_index] = (char) GetContainerData (from.GetValue (from_index));
}
protected override int DoCompareValues (int index1, int index2)
{
char val1 = _values [index1];
char val2 = _values [index2];
return val1 == val2 ? 0 : val1 < val2 ? -1 : 1;
}
protected override void Resize (int size)
{
if (_values == null) {
_values = new char [size];
return;
}
char[] tmp = new char [size];
Array.Copy (_values, 0, tmp, 0, _values.Length);
_values = tmp;
}
internal override long GetInt64 (int index)
{
return Convert.ToInt64 (_values [index]);
}
}
sealed class ByteDataContainer : DataContainer {
byte [] _values;
protected internal override object GetValue (int index)
{
return _values [index];
}
protected override void ZeroOut (int index)
{
_values [index] = 0;
}
protected override void SetValue (int index, object value)
{
_values [index] = (byte) GetContainerData (value);
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
_values [index] = record.GetByteSafe (field);
}
protected override void DoCopyValue (DataContainer from, int from_index, int to_index)
{
_values [to_index] = (byte) GetContainerData (from.GetValue (from_index));
}
protected override int DoCompareValues (int index1, int index2)
{
int val1 = _values [index1];
int val2 = _values [index2];
return val1 - val2;
}
protected override void Resize (int size)
{
if (_values == null) {
_values = new byte [size];
return;
}
byte[] tmp = new byte [size];
Array.Copy (_values, 0, tmp, 0, _values.Length);
_values = tmp;
}
internal override long GetInt64 (int index)
{
return _values [index];
}
}
sealed class SByteDataContainer : DataContainer {
sbyte [] _values;
protected internal override object GetValue (int index)
{
return _values [index];
}
protected override void ZeroOut (int index)
{
_values [index] = 0;
}
protected override void SetValue (int index, object value)
{
_values [index] = (sbyte) GetContainerData (value);
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
_values [index] = (sbyte) record.GetByteSafe (field);
}
protected override void DoCopyValue (DataContainer from, int from_index, int to_index)
{
_values [to_index] = (sbyte) GetContainerData (from.GetValue (from_index));
}
protected override int DoCompareValues (int index1, int index2)
{
int val1 = _values [index1];
int val2 = _values [index2];
return val1 - val2;
}
protected override void Resize (int size)
{
if (_values == null) {
_values = new sbyte [size];
return;
}
sbyte[] tmp = new sbyte [size];
Array.Copy (_values, 0, tmp, 0, _values.Length);
_values = tmp;
}
internal override long GetInt64 (int index)
{
return _values [index];
}
}
sealed class Int16DataContainer : DataContainer {
short [] _values;
protected internal override object GetValue (int index)
{
return _values [index];
}
protected override void ZeroOut (int index)
{
_values [index] = 0;
}
protected override void SetValue (int index, object value)
{
_values [index] = (short) GetContainerData (value);
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
_values [index] = record.GetInt16Safe (field);
}
protected override void DoCopyValue (DataContainer from, int from_index, int to_index)
{
_values [to_index] = (short) GetContainerData (from.GetValue (from_index));
}
protected override int DoCompareValues (int index1, int index2)
{
int val1 = _values [index1];
int val2 = _values [index2];
return val1 - val2;
}
protected override void Resize (int size)
{
if (_values == null) {
_values = new short [size];
return;
}
short[] tmp = new short [size];
Array.Copy (_values, 0, tmp, 0, _values.Length);
_values = tmp;
}
internal override long GetInt64 (int index)
{
return _values [index];
}
}
sealed class UInt16DataContainer : DataContainer {
ushort [] _values;
protected internal override object GetValue (int index)
{
return _values [index];
}
protected override void ZeroOut (int index)
{
_values [index] = 0;
}
protected override void SetValue (int index, object value)
{
_values [index] = (ushort) GetContainerData (value);
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
_values [index] = (ushort) record.GetInt16Safe (field);
}
protected override void DoCopyValue (DataContainer from, int from_index, int to_index)
{
_values [to_index] = (ushort) GetContainerData (from.GetValue (from_index));
}
protected override int DoCompareValues (int index1, int index2)
{
int val1 = _values [index1];
int val2 = _values [index2];
return val1 - val2;
}
protected override void Resize (int size)
{
if (_values == null) {
_values = new ushort [size];
return;
}
ushort[] tmp = new ushort [size];
Array.Copy (_values, 0, tmp, 0, _values.Length);
_values = tmp;
}
internal override long GetInt64 (int index)
{
return _values [index];
}
}
sealed class Int32DataContainer : DataContainer {
int [] _values;
protected internal override object GetValue (int index)
{
return _values [index];
}
protected override void ZeroOut (int index)
{
_values [index] = 0;
}
protected override void SetValue (int index, object value)
{
_values [index] = (int) GetContainerData (value);
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
_values [index] = record.GetInt32Safe (field);
}
protected override void DoCopyValue (DataContainer from, int from_index, int to_index)
{
_values [to_index] = (int) GetContainerData (from.GetValue (from_index));
}
protected override int DoCompareValues (int index1, int index2)
{
int val1 = _values [index1];
int val2 = _values [index2];
return val1 == val2 ? 0 : val1 < val2 ? -1 : 1;
}
protected override void Resize (int size)
{
if (_values == null) {
_values = new int [size];
return;
}
int[] tmp = new int [size];
Array.Copy (_values, 0, tmp, 0, _values.Length);
_values = tmp;
}
internal override long GetInt64 (int index)
{
return _values [index];
}
}
sealed class UInt32DataContainer : DataContainer {
uint [] _values;
protected internal override object GetValue (int index)
{
return _values [index];
}
protected override void ZeroOut (int index)
{
_values [index] = 0;
}
protected override void SetValue (int index, object value)
{
_values [index] = (uint) GetContainerData (value);
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
_values [index] = (uint) record.GetInt32Safe (field);
}
protected override void DoCopyValue (DataContainer from, int from_index, int to_index)
{
_values [to_index] = (uint) GetContainerData (from.GetValue (from_index));
}
protected override int DoCompareValues (int index1, int index2)
{
uint val1 = _values [index1];
uint val2 = _values [index2];
return val1 == val2 ? 0 : val1 < val2 ? -1 : 1;
}
protected override void Resize (int size)
{
if (_values == null) {
_values = new uint [size];
return;
}
uint[] tmp = new uint [size];
Array.Copy (_values, 0, tmp, 0, _values.Length);
_values = tmp;
}
internal override long GetInt64 (int index)
{
return _values [index];
}
}
sealed class Int64DataContainer : DataContainer {
long [] _values;
protected internal override object GetValue (int index)
{
return _values [index];
}
protected override void ZeroOut (int index)
{
_values [index] = 0;
}
protected override void SetValue (int index, object value)
{
_values [index] = (long) GetContainerData (value);
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
_values [index] = record.GetInt64Safe (field);
}
protected override void DoCopyValue (DataContainer from, int from_index, int to_index)
{
_values [to_index] = (long) GetContainerData (from.GetValue (from_index));
}
protected override int DoCompareValues (int index1, int index2)
{
long val1 = _values [index1];
long val2 = _values [index2];
return val1 == val2 ? 0 : val1 < val2 ? -1 : 1;
}
protected override void Resize (int size)
{
if (_values == null) {
_values = new long [size];
return;
}
long[] tmp = new long [size];
Array.Copy (_values, 0, tmp, 0, _values.Length);
_values = tmp;
}
internal override long GetInt64 (int index)
{
return _values [index];
}
}
sealed class UInt64DataContainer : DataContainer {
ulong [] _values;
protected internal override object GetValue (int index)
{
return _values [index];
}
protected override void ZeroOut (int index)
{
_values [index] = 0;
}
protected override void SetValue (int index, object value)
{
_values [index] = (ulong) GetContainerData (value);
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
_values [index] = (ulong) record.GetInt64Safe (field);
}
protected override void DoCopyValue (DataContainer from, int from_index, int to_index)
{
_values [to_index] = (ulong) GetContainerData (from.GetValue (from_index));
}
protected override int DoCompareValues (int index1, int index2)
{
ulong val1 = _values [index1];
ulong val2 = _values [index2];
return val1 == val2 ? 0 : val1 < val2 ? -1 : 1;
}
protected override void Resize (int size)
{
if (_values == null) {
_values = new ulong [size];
return;
}
ulong[] tmp = new ulong [size];
Array.Copy (_values, 0, tmp, 0, _values.Length);
_values = tmp;
}
internal override long GetInt64 (int index)
{
return Convert.ToInt64 (_values [index]);
}
}
sealed class SingleDataContainer : DataContainer {
float [] _values;
protected internal override object GetValue (int index)
{
return _values [index];
}
protected override void ZeroOut (int index)
{
_values [index] = 0;
}
protected override void SetValue (int index, object value)
{
_values [index] = (float) GetContainerData (value);
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
_values [index] = record.GetFloatSafe (field);
}
protected override void DoCopyValue (DataContainer from, int from_index, int to_index)
{
_values [to_index] = (float) GetContainerData (from.GetValue (from_index));
}
protected override int DoCompareValues (int index1, int index2)
{
float val1 = _values [index1];
float val2 = _values [index2];
return val1 == val2 ? 0 : val1 < val2 ? -1 : 1;
}
protected override void Resize (int size)
{
if (_values == null) {
_values = new float [size];
return;
}
float[] tmp = new float [size];
Array.Copy (_values, 0, tmp, 0, _values.Length);
_values = tmp;
}
internal override long GetInt64 (int index)
{
return Convert.ToInt64 (_values [index]);
}
}
sealed class DoubleDataContainer : DataContainer {
double [] _values;
protected internal override object GetValue (int index)
{
return _values [index];
}
protected override void ZeroOut (int index)
{
_values [index] = 0;
}
protected override void SetValue (int index, object value)
{
_values [index] = (double) GetContainerData (value);
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
_values [index] = record.GetDoubleSafe (field);
}
protected override void DoCopyValue (DataContainer from, int from_index, int to_index)
{
_values [to_index] = (double) GetContainerData (from.GetValue (from_index));
}
protected override int DoCompareValues (int index1, int index2)
{
double val1 = _values [index1];
double val2 = _values [index2];
return val1 == val2 ? 0 : val1 < val2 ? -1 : 1;
}
protected override void Resize (int size)
{
if (_values == null) {
_values = new double [size];
return;
}
double[] tmp = new double [size];
Array.Copy (_values, 0, tmp, 0, _values.Length);
_values = tmp;
}
internal override long GetInt64 (int index)
{
return Convert.ToInt64 (_values[index]);
}
}
class ObjectDataContainer : DataContainer {
object [] _values;
protected internal override object GetValue (int index)
{
return _values [index];
}
protected override void ZeroOut (int index)
{
_values [index] = null;
}
protected override void SetValue (int index, object value)
{
_values [index] = value;
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
_values [index] = record.GetValue (field);
}
protected override void DoCopyValue (DataContainer from, int from_index, int to_index)
{
_values [to_index] = GetContainerData (from.GetValue (from_index));
}
protected override int DoCompareValues (int index1, int index2)
{
object obj1 = _values [index1];
object obj2 = _values [index2];
if (obj1 == obj2)
return 0;
if (obj1 is IComparable) {
try {
return ((IComparable)obj1).CompareTo (obj2);
} catch {
if (obj2 is IComparable) {
obj2 = Convert.ChangeType (obj2, Type.GetTypeCode (obj1.GetType ()));
return ((IComparable)obj1).CompareTo (obj2);
}
}
}
return String.Compare (obj1.ToString (), obj2.ToString ());
}
protected override void Resize (int size)
{
if (_values == null) {
_values = new object [size];
return;
}
object[] tmp = new object [size];
Array.Copy (_values, 0, tmp, 0, _values.Length);
_values = tmp;
}
internal override long GetInt64 (int index)
{
return Convert.ToInt64 (_values [index]);
}
}
sealed class DateTimeDataContainer : ObjectDataContainer {
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
base.SetValue (index, record.GetDateTimeSafe (field));
}
protected override void SetValue (int index, object value)
{
base.SetValue (index, GetContainerData (value));
}
}
sealed class DecimalDataContainer : ObjectDataContainer {
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
base.SetValue (index, record.GetDecimalSafe (field));
}
protected override void SetValue (int index, object value)
{
base.SetValue (index, GetContainerData (value));
}
}
sealed class StringDataContainer : ObjectDataContainer {
private void SetValue (int index, string value)
{
if (value != null && Column.MaxLength >= 0 && Column.MaxLength < value.Length)
throw new ArgumentException ("Cannot set column '" + Column.ColumnName + "' to '" + value + "'. The value violates the MaxLength limit of this column.");
base.SetValue (index, value);
}
protected override void SetValue (int index, object value)
{
SetValue (index, (string) GetContainerData (value));
}
protected override void SetValueFromSafeDataRecord (int index, ISafeDataRecord record, int field)
{
SetValue (index, record.GetStringSafe (field));
}
protected override int DoCompareValues (int index1, int index2)
{
DataTable table = Column.Table;
return String.Compare ((string) this [index1], (string) this [index2], !table.CaseSensitive, table.Locale);
}
}
}