492 lines
14 KiB
C#
Raw Normal View History

//
// System.Data.Odbc.OdbcParameter
//
// Authors:
// Brian Ritchie (brianlritchie@hotmail.com)
// Sureshkumar T <tsureshkumar@novell.com> 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;
using System.ComponentModel;
namespace System.Data.Odbc
{
[TypeConverterAttribute ("System.Data.Odbc.OdbcParameter+OdbcParameterConverter, " + Consts.AssemblySystem_Data)]
public sealed class OdbcParameter :
DbParameter,
ICloneable, IDbDataParameter, IDataParameter
{
#region Fields
string name;
ParameterDirection direction;
bool isNullable;
int size;
DataRowVersion sourceVersion;
string sourceColumn;
byte _precision;
byte _scale;
object _value;
private OdbcTypeMap _typeMap;
private NativeBuffer _nativeBuffer = new NativeBuffer ();
private NativeBuffer _cbLengthInd;
private OdbcParameterCollection container;
#endregion
#region Constructors
public OdbcParameter ()
{
_cbLengthInd = new NativeBuffer ();
ParameterName = String.Empty;
IsNullable = false;
SourceColumn = String.Empty;
Direction = ParameterDirection.Input;
_typeMap = OdbcTypeConverter.GetTypeMap (OdbcType.NVarChar);
}
public OdbcParameter (string name, object value)
: this ()
{
this.ParameterName = name;
Value = value;
//FIXME: MS.net does not infer OdbcType from value unless a type is provided
_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 type)
: this ()
{
this.ParameterName = name;
_typeMap = (OdbcTypeMap) OdbcTypeConverter.GetTypeMap (type);
}
public OdbcParameter (string name, OdbcType type, int size)
: this (name, type)
{
this.Size = size;
}
public OdbcParameter (string name, OdbcType type, int size, string sourcecolumn)
: this (name, type, size)
{
this.SourceColumn = sourcecolumn;
}
[EditorBrowsable (EditorBrowsableState.Advanced)]
public OdbcParameter (string parameterName, OdbcType odbcType, int size,
ParameterDirection parameterDirection, bool isNullable,
byte precision, byte scale, string srcColumn,
DataRowVersion srcVersion, object value)
: this (parameterName, odbcType, size, srcColumn)
{
this.Direction = parameterDirection;
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; }
}
[OdbcCategory ("Data")]
[OdbcDescriptionAttribute ("The parameter generic type")]
public
override
DbType DbType {
get { return _typeMap.DbType; }
set {
if (value == _typeMap.DbType)
return;
_typeMap = OdbcTypeConverter.GetTypeMap (value);
}
}
[OdbcCategory ("Data")]
[OdbcDescriptionAttribute ("Input, output, or bidirectional parameter")]
[RefreshPropertiesAttribute (RefreshProperties.All)]
public
override
ParameterDirection Direction {
get { return direction; }
set { direction = value; }
}
[OdbcDescriptionAttribute ("A design-time property used for strongly typed code generation")]
public
override
bool IsNullable {
get { return isNullable; }
set { isNullable = value; }
}
[DefaultValue (OdbcType.NChar)]
[OdbcDescriptionAttribute ("The parameter native type")]
[RefreshPropertiesAttribute (RefreshProperties.All)]
[OdbcCategory ("Data")]
[DbProviderSpecificTypeProperty (true)]
public OdbcType OdbcType {
get { return _typeMap.OdbcType; }
set {
if (value == _typeMap.OdbcType)
return;
_typeMap = OdbcTypeConverter.GetTypeMap (value);
}
}
[OdbcDescription ("DataParameter_ParameterName")]
public
override
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")]
public
override
int Size {
get { return size; }
set { size = value; }
}
[OdbcDescription ("DataParameter_SourceColumn")]
[OdbcCategory ("DataCategory_Data")]
public
override
string SourceColumn {
get { return sourceColumn; }
set { sourceColumn = value; }
}
[OdbcDescription ("DataParameter_SourceVersion")]
[OdbcCategory ("DataCategory_Data")]
public
override
DataRowVersion SourceVersion {
get { return sourceVersion; }
set { sourceVersion = value; }
}
[TypeConverter (typeof(StringConverter))]
[OdbcDescription ("DataParameter_Value")]
[OdbcCategory ("DataCategory_Data")]
[RefreshPropertiesAttribute (RefreshProperties.All)]
public
override
object Value {
get { return _value; }
set { _value = value; }
}
#endregion // Properties
#region Methods
internal void Bind (OdbcCommand command, IntPtr hstmt, int ParamNum)
{
OdbcReturn ret;
int len;
// Convert System.Data.ParameterDirection into odbc enum
OdbcInputOutputDirection paramdir = libodbc.ConvertParameterDirection (this.Direction);
_cbLengthInd.EnsureAlloc (Marshal.SizeOf (typeof (int)));
if (Value is DBNull)
len = (int)OdbcLengthIndicator.NullData;
else {
len = GetNativeSize ();
AllocateBuffer ();
}
Marshal.WriteInt32 (_cbLengthInd, len);
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 command.Connection.CreateOdbcException (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;
if (Value is DBNull)
return;
DateTime dt;
TextInfo ti = CultureInfo.InvariantCulture.TextInfo;
Encoding enc = Encoding.GetEncoding (ti.ANSICodePage);
byte [] nativeBytes, buffer;
switch (_typeMap.OdbcType) {
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:
case OdbcType.Binary:
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!");
}
public override bool SourceColumnNullMapping {
get { return false; }
set { }
}
public override void ResetDbType ()
{
_typeMap = OdbcTypeConverter.GetTypeMap (OdbcType.NVarChar);
}
public void ResetOdbcType ()
{
_typeMap = OdbcTypeConverter.GetTypeMap (OdbcType.NVarChar);
}
#endregion
}
}