| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519 |
- //
- // System.Data.Odbc.OdbcParameter
- //
- // Authors:
- // Brian Ritchie ([email protected])
- // Sureshkumar T <[email protected]> 2004.
- //
- // Copyright (C) Brian Ritchie, 2002
- //
- //
- // Copyright (C) 2004 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.Text;
- using System.Data;
- using System.Data.Common;
- using System.Runtime.InteropServices;
- using System.Globalization;
- #if NET_2_0
- using System.Data.ProviderBase;
- #endif // NET_2_0
- using System.ComponentModel;
- namespace System.Data.Odbc
- {
- [TypeConverterAttribute (typeof (OdbcParameterConverter))]
- #if NET_2_0
- public sealed class OdbcParameter : DbParameterBase, ICloneable
- #else
- public sealed class OdbcParameter : MarshalByRefObject, IDbDataParameter, IDataParameter, ICloneable
- #endif // NET_2_0
- {
- #region Fields
- #if ONLY_1_1
- string name;
- ParameterDirection direction;
- bool isNullable;
- int size;
- DataRowVersion sourceVersion;
- string sourceColumn;
- byte _precision;
- byte _scale;
- object _value;
- #endif // ONLY_1_1
- private OdbcTypeMap _typeMap;
- private NativeBuffer _nativeBuffer = new NativeBuffer ();
- private NativeBuffer _cbLengthInd;
- private OdbcParameterCollection container = null;
-
- #endregion
- #region Constructors
-
- public OdbcParameter ()
- {
- _cbLengthInd = new NativeBuffer ();
- ParameterName = String.Empty;
- IsNullable = true;
- SourceColumn = String.Empty;
- Direction = ParameterDirection.Input;
- _typeMap = OdbcTypeConverter.GetTypeMap (OdbcType.VarChar);
- }
- public OdbcParameter (string name, object value)
- : this ()
- {
- this.ParameterName = name;
- Value = value;
- _typeMap = OdbcTypeConverter.InferFromValue (value);
- if (value != null && !value.GetType ().IsValueType) {
- Type type = value.GetType ();
- if (type.IsArray)
- Size = type.GetElementType () == typeof (byte) ?
- ((Array) value).Length : 0;
- else
- Size = value.ToString ().Length;
- }
- }
- public OdbcParameter (string name, OdbcType odbcType)
- : this ()
- {
- this.ParameterName = name;
- _typeMap = (OdbcTypeMap) OdbcTypeConverter.GetTypeMap (odbcType);
- }
- public OdbcParameter (string name, OdbcType odbcType, int size)
- : this (name, odbcType)
- {
- this.Size = size;
- }
- public OdbcParameter (string name, OdbcType odbcType, int size, string srcColumn)
- : this (name, odbcType, size)
- {
- this.SourceColumn = srcColumn;
- }
- [EditorBrowsable (EditorBrowsableState.Advanced)]
- public OdbcParameter(string name, OdbcType odbcType, int size,
- ParameterDirection direction, bool isNullable,
- byte precision, byte scale, string srcColumn,
- DataRowVersion srcVersion, object value)
- : this (name, odbcType, size, srcColumn)
- {
- this.Direction = direction;
- this.IsNullable = isNullable;
- this.SourceVersion = srcVersion;
- }
- #endregion
- #region Properties
- // Used to ensure that only one collection can contain this
- // parameter
- internal OdbcParameterCollection Container {
- get { return container; }
- set { container = value; }
- }
-
- [BrowsableAttribute (false)]
- [OdbcDescriptionAttribute ("The parameter generic type")]
- [RefreshPropertiesAttribute (RefreshProperties.All)]
- [DesignerSerializationVisibilityAttribute (DesignerSerializationVisibility.Hidden)]
- [OdbcCategory ("Data")]
- public
- #if NET_2_0
- override
- #endif // NET_2_0
- DbType DbType {
- get { return _typeMap.DbType; }
- set {
- if (value == _typeMap.DbType)
- return;
-
- _typeMap = OdbcTypeConverter.GetTypeMap (value);
- }
- }
-
- #if ONLY_1_1
- [OdbcCategory ("Data")]
- [OdbcDescriptionAttribute ("Input, output, or bidirectional parameter")]
- [DefaultValue (ParameterDirection.Input)]
- public ParameterDirection Direction {
- get { return direction; }
- set { direction = value; }
- }
-
- [BrowsableAttribute (false)]
- [OdbcDescriptionAttribute ("A design-time property used for strongly typed code generation")]
- [DesignOnlyAttribute (true)]
- [EditorBrowsableAttribute (EditorBrowsableState.Advanced)]
- [DefaultValue (false)]
- public bool IsNullable {
- get { return isNullable; }
- set { isNullable = value; }
- }
- #endif // ONLY_1_1
- [DefaultValue (OdbcType.NChar)]
- [OdbcDescriptionAttribute ("The parameter native type")]
- [RefreshPropertiesAttribute (RefreshProperties.All)]
- [OdbcCategory ("Data")]
- public OdbcType OdbcType {
- get { return _typeMap.OdbcType; }
- set {
- if (value == OdbcType)
- return;
- _typeMap = OdbcTypeConverter.GetTypeMap (value);
- }
- }
-
- #if ONLY_1_1
- [OdbcDescription ("DataParameter_ParameterName")]
- [DefaultValue ("")]
- public string ParameterName {
- get { return name; }
- set { name = value; }
- }
- [OdbcDescription ("DbDataParameter_Precision")]
- [OdbcCategory ("DataCategory_Data")]
- [DefaultValue (0)]
- public byte Precision {
- get { return _precision; }
- set { _precision = value; }
- }
-
- [OdbcDescription ("DbDataParameter_Scale")]
- [OdbcCategory ("DataCategory_Data")]
- [DefaultValue (0)]
- public byte Scale {
- get { return _scale; }
- set { _scale = value; }
- }
-
- [OdbcDescription ("DbDataParameter_Size")]
- [OdbcCategory ("DataCategory_Data")]
- [DefaultValue (0)]
- public int Size {
- get { return size; }
- set { size = value; }
- }
- [OdbcDescription ("DataParameter_SourceColumn")]
- [OdbcCategory ("DataCategory_Data")]
- [DefaultValue ("")]
- public string SourceColumn {
- get { return sourceColumn; }
- set { sourceColumn = value; }
- }
-
- [OdbcDescription ("DataParameter_SourceVersion")]
- [OdbcCategory ("DataCategory_Data")]
- [DefaultValue ("Current")]
- public DataRowVersion SourceVersion {
- get { return sourceVersion; }
- set { sourceVersion = value; }
- }
- [TypeConverter (typeof(StringConverter))]
- [OdbcDescription ("DataParameter_Value")]
- [OdbcCategory ("DataCategory_Data")]
- [DefaultValue (null)]
- public object Value {
- get {
- return _value;
- }
- set {
- _value = value;
- }
- }
- #endif // ONLY_1_1
- #endregion // Properties
- #region Methods
- internal void Bind(IntPtr hstmt, int ParamNum) {
- OdbcReturn ret;
-
- // Convert System.Data.ParameterDirection into odbc enum
- OdbcInputOutputDirection paramdir = libodbc.ConvertParameterDirection(this.Direction);
- _cbLengthInd.EnsureAlloc (Marshal.SizeOf (typeof (int)));
- Marshal.WriteInt32 (_cbLengthInd, GetNativeSize ());
- AllocateBuffer ();
- ret = libodbc.SQLBindParameter(hstmt, (ushort) ParamNum, (short) paramdir,
- _typeMap.NativeType, _typeMap.SqlType, Convert.ToUInt32(Size),
- 0, (IntPtr) _nativeBuffer, 0, _cbLengthInd);
- // Check for error condition
- if ((ret != OdbcReturn.Success) && (ret != OdbcReturn.SuccessWithInfo))
- throw new OdbcException(new OdbcError("SQLBindParam", OdbcHandleType.Stmt, hstmt));
- }
- [MonoTODO]
- object ICloneable.Clone ()
- {
- throw new NotImplementedException ();
- }
- public override string ToString ()
- {
- return ParameterName;
- }
-
- private int GetNativeSize ()
- {
- TextInfo ti = CultureInfo.InvariantCulture.TextInfo;
- Encoding enc = Encoding.GetEncoding (ti.ANSICodePage);
- switch (_typeMap.OdbcType) {
- case OdbcType.Binary:
- if (Value.GetType ().IsArray &&
- Value.GetType ().GetElementType () == typeof (byte))
- return ( (Array) Value).Length;
- else
- return Value.ToString ().Length;
- case OdbcType.Bit:
- return Marshal.SizeOf (typeof (byte));
- case OdbcType.Double:
- return Marshal.SizeOf (typeof (double));
- case OdbcType.Real:
- return Marshal.SizeOf (typeof (float));
- case OdbcType.Int:
- return Marshal.SizeOf (typeof (int));
- case OdbcType.BigInt:
- return Marshal.SizeOf (typeof (long));
- case OdbcType.Decimal:
- case OdbcType.Numeric:
- return 19;
- case OdbcType.SmallInt:
- return Marshal.SizeOf (typeof (Int16));
- case OdbcType.TinyInt:
- return Marshal.SizeOf (typeof (byte));
- case OdbcType.Char:
- case OdbcType.Text:
- case OdbcType.VarChar:
- return enc.GetByteCount (Convert.ToString (Value)) + 1;
- case OdbcType.NChar:
- case OdbcType.NText:
- case OdbcType.NVarChar:
- // FIXME: Change to unicode
- return enc.GetByteCount (Convert.ToString (Value)) + 1;
- case OdbcType.VarBinary:
- case OdbcType.Image:
- if (Value.GetType ().IsArray &&
- Value.GetType ().GetElementType () == typeof (byte))
- return ( (Array) Value).Length;
- throw new ArgumentException ("Unsupported Native Type!");
- case OdbcType.Date:
- case OdbcType.DateTime:
- case OdbcType.SmallDateTime:
- case OdbcType.Time:
- case OdbcType.Timestamp:
- return 18;
- case OdbcType.UniqueIdentifier:
- return Marshal.SizeOf (typeof (Guid));
- }
- if (Value.GetType ().IsArray &&
- Value.GetType ().GetElementType () == typeof (byte))
- return ( (Array) Value).Length;
-
- return Value.ToString ().Length;
- }
- private void AllocateBuffer ()
- {
- int size = GetNativeSize ();
- if (_nativeBuffer.Size == size)
- return;
- _nativeBuffer.AllocBuffer (size);
- }
- internal void CopyValue ()
- {
- if (_nativeBuffer.Handle == IntPtr.Zero)
- return;
- DateTime dt;
- TextInfo ti = CultureInfo.InvariantCulture.TextInfo;
- Encoding enc = Encoding.GetEncoding (ti.ANSICodePage);
- byte [] nativeBytes, buffer;
- switch (_typeMap.OdbcType) {
- case OdbcType.Binary:
- throw new NotImplementedException ();
- case OdbcType.Bit:
- Marshal.WriteByte (_nativeBuffer, Convert.ToByte (Value));
- return;
- case OdbcType.Double:
- Marshal.StructureToPtr (Convert.ToDouble (Value), _nativeBuffer, false);
- return;
- case OdbcType.Real:
- Marshal.StructureToPtr (Convert.ToSingle (Value), _nativeBuffer, false);
- return;
- case OdbcType.Int:
- Marshal.WriteInt32 (_nativeBuffer, Convert.ToInt32 (Value));
- return;
- case OdbcType.BigInt:
- Marshal.WriteInt64 (_nativeBuffer, Convert.ToInt64 (Value));
- return;
- case OdbcType.Decimal:
- case OdbcType.Numeric:
- // for numeric, the buffer is a packed decimal struct.
- // ref http://www.it-faq.pl/mskb/181/254.HTM
- int [] bits = Decimal.GetBits (Convert.ToDecimal (Value));
- buffer = new byte [19]; // ref sqltypes.h
- buffer [0] = Precision;
- buffer [1] = (byte) ((bits [3] & 0x00FF0000) >> 16); // scale
- buffer [2] = (byte) ((bits [3] & 0x80000000) > 0 ? 2 : 1); //sign
- Buffer.BlockCopy (bits, 0, buffer, 3, 12); // copy data
- for (int j = 16; j < 19; j++) // pad with 0
- buffer [j] = 0;
- Marshal.Copy (buffer, 0, _nativeBuffer, 19);
- return;
- case OdbcType.SmallInt:
- Marshal.WriteInt16 (_nativeBuffer, Convert.ToInt16 (Value));
- return;
- case OdbcType.TinyInt:
- Marshal.WriteByte (_nativeBuffer, Convert.ToByte (Value));
- return;
- case OdbcType.Char:
- case OdbcType.Text:
- case OdbcType.VarChar:
- buffer = new byte [GetNativeSize ()];
- nativeBytes = enc.GetBytes (Convert.ToString (Value));
- Array.Copy (nativeBytes, 0, buffer, 0, nativeBytes.Length);
- buffer [buffer.Length-1] = (byte) 0;
- Marshal.Copy (buffer, 0, _nativeBuffer, buffer.Length);
- Marshal.WriteInt32 (_cbLengthInd, -3);
- return;
- case OdbcType.NChar:
- case OdbcType.NText:
- case OdbcType.NVarChar:
- // FIXME : change to unicode
- buffer = new byte [GetNativeSize ()];
- nativeBytes = enc.GetBytes (Convert.ToString (Value));
- Array.Copy (nativeBytes, 0, buffer, 0, nativeBytes.Length);
- buffer [buffer.Length-1] = (byte) 0;
- Marshal.Copy (buffer, 0, _nativeBuffer, buffer.Length);
- Marshal.WriteInt32 (_cbLengthInd, -3);
- return;
- case OdbcType.VarBinary:
- case OdbcType.Image:
- if (Value.GetType ().IsArray &&
- Value.GetType ().GetElementType () == typeof (byte)) {
- Marshal.Copy ( (byte []) Value, 0, _nativeBuffer, ((byte []) Value).Length);
- }else
- throw new ArgumentException ("Unsupported Native Type!");
- return;
- case OdbcType.Date:
- dt = (DateTime) Value;
- Marshal.WriteInt16 (_nativeBuffer, 0, (short) dt.Year);
- Marshal.WriteInt16 (_nativeBuffer, 2, (short) dt.Month);
- Marshal.WriteInt16 (_nativeBuffer, 4, (short) dt.Day);
- return;
- case OdbcType.Time:
- dt = (DateTime) Value;
- Marshal.WriteInt16 (_nativeBuffer, 0, (short) dt.Hour);
- Marshal.WriteInt16 (_nativeBuffer, 2, (short) dt.Minute);
- Marshal.WriteInt16 (_nativeBuffer, 4, (short) dt.Second);
- return;
- case OdbcType.SmallDateTime:
- case OdbcType.Timestamp:
- case OdbcType.DateTime:
- dt = (DateTime) Value;
- Marshal.WriteInt16 (_nativeBuffer, 0, (short) dt.Year);
- Marshal.WriteInt16 (_nativeBuffer, 2, (short) dt.Month);
- Marshal.WriteInt16 (_nativeBuffer, 4, (short) dt.Day);
- Marshal.WriteInt16 (_nativeBuffer, 6, (short) dt.Hour);
- Marshal.WriteInt16 (_nativeBuffer, 8, (short) dt.Minute);
- Marshal.WriteInt16 (_nativeBuffer, 10, (short) dt.Second);
- Marshal.WriteInt32 (_nativeBuffer, 12, (int) (dt.Ticks % 10000000) * 100);
- return;
- case OdbcType.UniqueIdentifier:
- throw new NotImplementedException ();
- }
- if (Value.GetType ().IsArray &&
- Value.GetType ().GetElementType () == typeof (byte)) {
- Marshal.Copy ( (byte []) Value, 0, _nativeBuffer, ((byte []) Value).Length);
- }else
- throw new ArgumentException ("Unsupported Native Type!");
- }
- #if NET_2_0
- [MonoTODO]
- public override bool SourceColumnNullMapping {
- get {return false;}
- set {}
- }
- [MonoTODO]
- public override void PropertyChanging ()
- {
- }
-
- [MonoTODO]
- protected override byte ValuePrecision (object value)
- {
- throw new NotImplementedException ();
- }
- [MonoTODO]
- protected override byte ValueScale (object value)
- {
- throw new NotImplementedException ();
- }
- [MonoTODO]
- protected override int ValueSize (object value)
- {
- throw new NotImplementedException ();
- }
- [MonoTODO]
- public override void ResetDbType ()
- {
- throw new NotImplementedException ();
- }
- #endif // NET_2_0
- #endregion
- }
- }
|