19 Star 33 Fork 26

openLooKeng / hetu-odbc-driver

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
ma_typeconv.c 21.35 KB
一键复制 编辑 原始数据 按行查看 历史
wangmingyue 提交于 2020-06-29 21:58 . openlookeng first commit
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784
/************************************************************************************
Copyright (C) 2017 MariaDB Corporation AB
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Library General Public License for more details.
You should have received a copy of the GNU Library General Public
License along with this library; if not see <http://www.gnu.org/licenses>
or write to the Free Software Foundation, Inc.,
51 Franklin St., Fifth Floor, Boston, MA 02110, USA
*************************************************************************************/
/* ODBC C->SQL and SQL->C type conversion functions */
#include <ma_odbc.h>
/* Borrowed from C/C and adapted */
SQLRETURN MADB_Str2Ts(const char *Str, size_t Length, MYSQL_TIME *Tm, BOOL Interval, MADB_Error *Error, BOOL *isTime)
{
char *Start= MADB_ALLOC(Length + 1), *Frac, *End= Start + Length;
my_bool isDate= 0;
if (Start == NULL)
{
return MADB_SetError(Error, MADB_ERR_HY001, NULL, 0);
}
memset(Tm, 0, sizeof(MYSQL_TIME));
memcpy(Start, Str, Length);
Start[Length]= '\0';
while (Length && isspace(*Start)) Start++, Length--;
if (Length == 0)
{
return SQL_SUCCESS;//MADB_SetError(Error, MADB_ERR_22008, NULL, 0);
}
/* Determine time type:
MYSQL_TIMESTAMP_DATE: [-]YY[YY].MM.DD
MYSQL_TIMESTAMP_DATETIME: [-]YY[YY].MM.DD hh:mm:ss.mmmmmm
MYSQL_TIMESTAMP_TIME: [-]hh:mm:ss.mmmmmm
*/
if (strchr(Start, '-'))
{
if (sscanf(Start, "%d-%u-%u", &Tm->year, &Tm->month, &Tm->day) < 3)
{
return MADB_SetError(Error, MADB_ERR_22008, NULL, 0);
}
isDate= 1;
if (!(Start= strchr(Start, ' ')))
{
goto check;
}
}
if (!strchr(Start, ':'))
{
goto check;
}
if (isDate == 0)
{
*isTime= 1;
}
if ((Frac= strchr(Start, '.')) != NULL) /* fractional seconds */
{
size_t FracMulIdx= End - (Frac + 1) - 1/*to get index array index */;
/* ODBC - nano-seconds */
if (sscanf(Start, "%d:%u:%u.%6lu", &Tm->hour, &Tm->minute,
&Tm->second, &Tm->second_part) < 4)
{
return MADB_SetError(Error, MADB_ERR_22008, NULL, 0);
}
/* 9 digits up to nano-seconds, and -1 since comparing with arr idx */
if (FracMulIdx < 6 - 1)
{
static unsigned long Mul[]= {100000, 10000, 1000, 100, 10};
Tm->second_part*= Mul[FracMulIdx];
}
}
else
{
if (sscanf(Start, "%d:%u:%u", &Tm->hour, &Tm->minute,
&Tm->second) < 3)
{
return MADB_SetError(Error, MADB_ERR_22008, NULL, 0);
}
}
check:
if (Interval == FALSE)
{
if (isDate)
{
if (Tm->year > 0)
{
if (Tm->year < 70)
{
Tm->year+= 2000;
}
else if (Tm->year < 100)
{
Tm->year+= 1900;
}
}
}
}
return SQL_SUCCESS;
}
/* {{{ MADB_ConversionSupported */
BOOL MADB_ConversionSupported(MADB_DescRecord *From, MADB_DescRecord *To)
{
switch (From->ConciseType)
{
case SQL_C_TIMESTAMP:
case SQL_C_TYPE_TIMESTAMP:
case SQL_C_TIME:
case SQL_C_TYPE_TIME:
case SQL_C_DATE:
case SQL_C_TYPE_DATE:
if (To->Type == SQL_INTERVAL)
{
return FALSE;
}
}
return TRUE;
}
/* }}} */
/* {{{ MADB_ConvertCharToBit */
char MADB_ConvertCharToBit(MADB_Stmt *Stmt, char *src)
{
char *EndPtr= NULL;
float asNumber= strtof(src, &EndPtr);
if (asNumber < 0 || asNumber > 1)
{
/* 22003 */
}
else if (asNumber != 0 && asNumber != 1)
{
/* 22001 */
}
else if (EndPtr != NULL && *EndPtr != '\0')
{
/* 22018. TODO: check if condition is correct */
}
return asNumber != 0 ? '\1' : '\0';
}
/* }}} */
/* {{{ MADB_ConvertNumericToChar */
size_t MADB_ConvertNumericToChar(SQL_NUMERIC_STRUCT *Numeric, char *Buffer, int *ErrorCode)
{
long long Numerator= 0;
long long Denominator= 1;
unsigned long long Left= 0, Right= 0;
int Scale= 0;
int ppos= 0;
long ByteDenominator= 1;
int i;
char *p;
my_bool hasDot= FALSE;
Buffer[0]= 0;
*ErrorCode= 0;
Scale+= (Numeric->scale < 0) ? -Numeric->scale : Numeric->scale;
for (i=0; i < SQL_MAX_NUMERIC_LEN; ++i)
{
Numerator+= Numeric->val[i] * ByteDenominator;
ByteDenominator<<= 8;
}
if (!Numeric->sign)
Numerator= -Numerator;
Denominator= (long long)pow(10, Scale);
Left= Numerator / Denominator;
//_i64toa_s(Numerator, Buffer, 38, 10);
if (Numeric->scale > 0)
{
char tmp[38];
_snprintf(tmp, 38, "%%.%df", Numeric->scale);
_snprintf(Buffer, 38, tmp, Numerator / pow(10, Scale));
}
else
{
_snprintf(Buffer, 38, "%lld", Numerator);
while (strlen(Buffer) < (size_t)(Numeric->precision - Numeric->scale))
strcat(Buffer, "0");
}
if (Buffer[0] == '-')
Buffer++;
/* Truncation checks:
1st ensure, that the digits before decimal point will fit */
if ((p= strchr(Buffer, '.')))
{
if (p - Buffer - 1 > Numeric->precision)
{
*ErrorCode= MADB_ERR_22003;
Buffer[Numeric->precision]= 0;
goto end;
}
if (Numeric->scale > 0 && Left > 0 && (p - Buffer) + strlen(p) > Numeric->precision)
{
*ErrorCode= MADB_ERR_01S07;
Buffer[Numeric->precision + 1]= 0;
goto end;
}
}
while (Numeric->scale < 0 && strlen(Buffer) < (size_t)(Numeric->precision - Numeric->scale))
strcat(Buffer, "0");
if (strlen(Buffer) > (size_t)(Numeric->precision + Scale) && Numeric->scale > 0)
*ErrorCode= MADB_ERR_01S07;
end:
/* check if last char is decimal point */
if (strlen(Buffer) && Buffer[strlen(Buffer)-1] == '.')
Buffer[strlen(Buffer)-1] = 0;
if (!Numeric->sign)
Buffer--;
return strlen(Buffer);
}
/* }}} */
/* {{{ MADB_ConvertNullValue */
SQLRETURN MADB_ConvertNullValue(MADB_Stmt *Stmt, MYSQL_BIND *MaBind)
{
MaBind->buffer_type= MYSQL_TYPE_NULL;
MaBind->buffer_length= 0;
return SQL_SUCCESS;
}
/* }}} */
/* {{{ MADB_ProcessIndicator */
/* Returns TRUE if indicator contains some special value, and thus no further type conversion is needed */
BOOL MADB_ProcessIndicator(MADB_Stmt *Stmt, SQLLEN Indicator, char * DefaultValue, MYSQL_BIND *MaBind)
{
switch (Indicator)
{
case SQL_COLUMN_IGNORE:
if (DefaultValue == NULL)
{
MADB_ConvertNullValue(Stmt, MaBind);
}
else
{
MaBind->buffer= DefaultValue;
MaBind->buffer_length= (unsigned long)strlen(DefaultValue);
MaBind->buffer_type= MYSQL_TYPE_STRING;
}
return TRUE;
case SQL_NULL_DATA:
MADB_ConvertNullValue(Stmt, MaBind);
return TRUE;
}
return FALSE;
}
/* }}} */
/* {{{ MADB_CalculateLength */
SQLLEN MADB_CalculateLength(MADB_Stmt *Stmt, SQLLEN *OctetLengthPtr, MADB_DescRecord *CRec, void* DataPtr)
{
/* If no OctetLengthPtr was specified, or OctetLengthPtr is SQL_NTS character
are considered to be NULL binary data are null terminated */
if (!OctetLengthPtr || *OctetLengthPtr == SQL_NTS)
{
/* Meaning of Buffer Length is not quite clear in specs. Thus we treat in the way, that does not break
(old) testcases. i.e. we neglect its value if Length Ptr is specified */
SQLLEN BufferLen= OctetLengthPtr ? -1 : CRec->OctetLength;
switch (CRec->ConciseType)
{
case SQL_C_WCHAR:
/* CRec->OctetLength eq 0 means not 0-length buffer, but that this value is not specified. Thus -1, for SqlwcsLen
and SafeStrlen that means buffer len is not specified */
return SqlwcsLen((SQLWCHAR *)DataPtr, BufferLen/sizeof(SQLWCHAR) - test(BufferLen == 0)) * sizeof(SQLWCHAR);
break;
case SQL_C_BINARY:
case SQL_VARBINARY:
case SQL_LONGVARBINARY:
case SQL_C_CHAR:
return SafeStrlen((SQLCHAR *)DataPtr, BufferLen != 0 ? BufferLen : -1);
}
}
else
{
return *OctetLengthPtr;
}
return CRec->OctetLength;
}
/* }}} */
/* {{{ MADB_GetBufferForSqlValue */
void* MADB_GetBufferForSqlValue(MADB_Stmt *Stmt, MADB_DescRecord *CRec, size_t Size)
{
if (Stmt->RebindParams || CRec->InternalBuffer == NULL)
{
MADB_FREE(CRec->InternalBuffer);
CRec->InternalBuffer= MADB_CALLOC(Size);
if (CRec->InternalBuffer == NULL)
{
MADB_SetError(&Stmt->Error, MADB_ERR_HY001, NULL, 0);
return NULL;
}
}
return (void *)CRec->InternalBuffer;
}
/* }}} */
/* {{{ MADB_Wchar2Sql */
SQLRETURN MADB_Wchar2Sql(MADB_Stmt *Stmt, MADB_DescRecord *CRec, void* DataPtr, SQLLEN Length,
MADB_DescRecord *SqlRec, MYSQL_BIND *MaBind, void **Buffer, unsigned long *LengthPtr)
{
SQLULEN mbLength=0;
MADB_FREE(CRec->InternalBuffer);
/* conn cs ? */
CRec->InternalBuffer= MADB_ConvertFromWChar((SQLWCHAR *)DataPtr, (SQLINTEGER)(Length / sizeof(SQLWCHAR)),
&mbLength, &Stmt->Connection->Charset, NULL);
if (CRec->InternalBuffer == NULL)
{
return MADB_SetError(&Stmt->Error, MADB_ERR_HY001, NULL, 0);
}
*LengthPtr= (unsigned long)mbLength;
*Buffer= CRec->InternalBuffer;
MaBind->buffer_type= MYSQL_TYPE_STRING;
return SQL_SUCCESS;
}
/* }}} */
/* {{{ MADB_Char2Sql */
SQLRETURN MADB_Char2Sql(MADB_Stmt *Stmt, MADB_DescRecord *CRec, void* DataPtr, SQLLEN Length,
MADB_DescRecord *SqlRec, MYSQL_BIND *MaBind, void **Buffer, unsigned long *LengthPtr)
{
switch (SqlRec->Type)
{
case SQL_BIT:
if (*Buffer == NULL)
{
CRec->InternalBuffer= (char *)MADB_GetBufferForSqlValue(Stmt, CRec, MaBind->buffer_length);
if (CRec->InternalBuffer == NULL)
{
return Stmt->Error.ReturnValue;
}
*Buffer= CRec->InternalBuffer;
}
*LengthPtr= 1;
**(char**)Buffer= MADB_ConvertCharToBit(Stmt, DataPtr);
MaBind->buffer_type= MYSQL_TYPE_TINY;
break;
case SQL_DATETIME:
{
MYSQL_TIME Tm;
SQL_TIMESTAMP_STRUCT Ts;
BOOL isTime;
/* Enforcing constraints on date/time values */
RETURN_ERROR_OR_CONTINUE(MADB_Str2Ts(DataPtr, Length, &Tm, FALSE, &Stmt->Error, &isTime));
MADB_CopyMadbTimeToOdbcTs(&Tm, &Ts);
RETURN_ERROR_OR_CONTINUE(MADB_TsConversionIsPossible(&Ts, SqlRec->ConciseType, &Stmt->Error, MADB_ERR_22018, isTime));
/* To stay on the safe side - still sending as string in the default branch */
}
default:
/* Bulk shouldn't get here, thus logic for single paramset execution */
*LengthPtr= (unsigned long)Length;
*Buffer= DataPtr;
MaBind->buffer_type= MYSQL_TYPE_STRING;
}
return SQL_SUCCESS;
}
/* }}} */
/* {{{ MADB_Numeric2Sql */
SQLRETURN MADB_Numeric2Sql(MADB_Stmt *Stmt, MADB_DescRecord *CRec, void* DataPtr, SQLLEN Length,
MADB_DescRecord *SqlRec, MYSQL_BIND *MaBind, void **Buffer, unsigned long *LengthPtr)
{
SQL_NUMERIC_STRUCT *p;
int ErrorCode= 0;
/* We might need to preserve this pointer to be able to later release the memory */
CRec->InternalBuffer= (char *)MADB_GetBufferForSqlValue(Stmt, CRec, MADB_CHARSIZE_FOR_NUMERIC);
if (CRec->InternalBuffer == NULL)
{
return Stmt->Error.ReturnValue;
}
p= (SQL_NUMERIC_STRUCT *)DataPtr;
p->scale= (SQLSCHAR)SqlRec->Scale;
p->precision= (SQLSCHAR)SqlRec->Precision;
*LengthPtr= (unsigned long)MADB_ConvertNumericToChar((SQL_NUMERIC_STRUCT *)p, CRec->InternalBuffer, &ErrorCode);;
*Buffer= CRec->InternalBuffer;
MaBind->buffer_type= MYSQL_TYPE_STRING;
if (ErrorCode)
{
/*TODO: I guess this parameters row should be skipped */
return MADB_SetError(&Stmt->Error, ErrorCode, NULL, 0);
}
return SQL_SUCCESS;
}
/* }}} */
/* {{{ MADB_TsConversionIsPossible */
SQLRETURN MADB_TsConversionIsPossible(SQL_TIMESTAMP_STRUCT *ts, SQLSMALLINT SqlType, MADB_Error *Error, enum enum_madb_error SqlState, int isTime)
{
/* I think instead of MADB_ERR_22008 there should be also SqlState */
switch (SqlType)
{
case SQL_TIME:
case SQL_TYPE_TIME:
if (ts->fraction)
{
return MADB_SetError(Error, MADB_ERR_22008, NULL, 0);
}
break;
case SQL_DATE:
case SQL_TYPE_DATE:
if (ts->hour + ts->minute + ts->second + ts->fraction)
{
return MADB_SetError(Error, MADB_ERR_22008, NULL, 0);
}
default:
/* This only would be good for SQL_TYPE_TIME. If C type is time(isTime!=0), and SQL type is timestamp, date fields may be NULL - driver should set them to current date */
if ((isTime == 0 && ts->year == 0) || ts->month == 0 || ts->day == 0)
{
return MADB_SetError(Error, SqlState, NULL, 0);
}
}
return SQL_SUCCESS;
}
/* }}} */
/* {{{ MADB_Timestamp2Sql */
SQLRETURN MADB_Timestamp2Sql(MADB_Stmt *Stmt, MADB_DescRecord *CRec, void* DataPtr, SQLLEN Length,
MADB_DescRecord *SqlRec, MYSQL_BIND *MaBind, void **Buffer, unsigned long *LengthPtr)
{
MYSQL_TIME *tm= NULL;
SQL_TIMESTAMP_STRUCT *ts= (SQL_TIMESTAMP_STRUCT *)DataPtr;
RETURN_ERROR_OR_CONTINUE(MADB_TsConversionIsPossible(ts, SqlRec->ConciseType, &Stmt->Error, MADB_ERR_22007, 0));
if (*Buffer == NULL)
{
tm= (MYSQL_TIME*)MADB_GetBufferForSqlValue(Stmt, CRec, sizeof(MYSQL_TIME));
if (tm == NULL)
{
/* Error is set in function responsible for allocation */
return Stmt->Error.ReturnValue;
}
*Buffer= tm;
}
else
{
tm= *Buffer;
}
/* Default types. Not quite clear if time_type has any effect */
tm->time_type= MYSQL_TIMESTAMP_DATETIME;
MaBind->buffer_type= MYSQL_TYPE_TIMESTAMP;
switch (SqlRec->ConciseType) {
case SQL_TYPE_DATE:
if (ts->hour + ts->minute + ts->second + ts->fraction != 0)
{
return MADB_SetError(&Stmt->Error, MADB_ERR_22008, "Time fields are nonzero", 0);
}
MaBind->buffer_type= MYSQL_TYPE_DATE;
tm->time_type= MYSQL_TIMESTAMP_DATE;
tm->year= ts->year;
tm->month= ts->month;
tm->day= ts->day;
break;
case SQL_TYPE_TIME:
if (ts->fraction != 0)
{
return MADB_SetError(&Stmt->Error, MADB_ERR_22008, "Fractional seconds fields are nonzero", 0);
}
if (!VALID_TIME(ts))
{
return MADB_SetError(&Stmt->Error, MADB_ERR_22007, "Invalid time", 0);
}
MaBind->buffer_type= MYSQL_TYPE_TIME;
tm->time_type= MYSQL_TIMESTAMP_TIME;
tm->hour= ts->hour;
tm->minute= ts->minute;
tm->second= ts->second;
break;
default:
MADB_CopyOdbcTsToMadbTime(ts, tm);
}
*LengthPtr= sizeof(MYSQL_TIME);
return SQL_SUCCESS;
}
/* }}} */
/* {{{ MADB_Time2Sql */
SQLRETURN MADB_Time2Sql(MADB_Stmt *Stmt, MADB_DescRecord *CRec, void* DataPtr, SQLLEN Length,
MADB_DescRecord *SqlRec, MYSQL_BIND *MaBind, void **Buffer, unsigned long *LengthPtr)
{
MYSQL_TIME *tm= NULL;
SQL_TIME_STRUCT *ts= (SQL_TIME_STRUCT *)DataPtr;
if ((SqlRec->ConciseType == SQL_TYPE_TIME || SqlRec->ConciseType == SQL_TYPE_TIMESTAMP ||
SqlRec->ConciseType == SQL_TIME || SqlRec->ConciseType == SQL_TIMESTAMP || SqlRec->ConciseType == SQL_DATETIME) &&
!VALID_TIME(ts))
{
return MADB_SetError(&Stmt->Error, MADB_ERR_22007, NULL, 0);
}
if (*Buffer == NULL)
{
tm= (MYSQL_TIME*)MADB_GetBufferForSqlValue(Stmt, CRec, sizeof(MYSQL_TIME));
if (tm == NULL)
{
/* Error is set in function responsible for allocation */
return Stmt->Error.ReturnValue;
}
*Buffer= tm;
}
else
{
tm= *Buffer;
}
if(SqlRec->ConciseType == SQL_TYPE_TIMESTAMP ||
SqlRec->ConciseType == SQL_TIMESTAMP || SqlRec->ConciseType == SQL_DATETIME)
{
time_t sec_time;
struct tm * cur_tm;
sec_time= time(NULL);
cur_tm= localtime(&sec_time);
tm->year= 1900 + cur_tm->tm_year;
tm->month= cur_tm->tm_mon + 1;
tm->day= cur_tm->tm_mday;
tm->second_part= 0;
}
else
{
tm->year= 0;
tm->month= 0;
tm->day= 0;
}
tm->hour= ts->hour;
tm->minute= ts->minute;
tm->second= ts->second;
tm->second_part= 0;
tm->time_type= MYSQL_TIMESTAMP_DATETIME;
MaBind->buffer_type= MYSQL_TYPE_DATETIME;
*LengthPtr= sizeof(MYSQL_TIME);
return SQL_SUCCESS;
}
/* }}} */
/* {{{ MADB_IntervalHtoMS2Sql */
SQLRETURN MADB_IntervalHtoMS2Sql(MADB_Stmt *Stmt, MADB_DescRecord *CRec, void* DataPtr, SQLLEN Length,
MADB_DescRecord *SqlRec, MYSQL_BIND *MaBind, void **Buffer, unsigned long *LengthPtr)
{
MYSQL_TIME *tm= NULL;
SQL_INTERVAL_STRUCT *is= (SQL_INTERVAL_STRUCT *)DataPtr;
if (*Buffer == NULL)
{
tm= (MYSQL_TIME*)MADB_GetBufferForSqlValue(Stmt, CRec, sizeof(MYSQL_TIME));
if (tm == NULL)
{
/* Error is set in function responsible for allocation */
return Stmt->Error.ReturnValue;
}
*Buffer= tm;
}
else
{
tm= *Buffer;
}
tm->hour= is->intval.day_second.hour;
tm->minute= is->intval.day_second.minute;
tm->second= CRec->ConciseType == SQL_C_INTERVAL_HOUR_TO_SECOND ? is->intval.day_second.second : 0;
tm->second_part= 0;
tm->time_type= MYSQL_TIMESTAMP_TIME;
MaBind->buffer_type= MYSQL_TYPE_TIME;
*LengthPtr= sizeof(MYSQL_TIME);
return SQL_SUCCESS;
}
/* }}} */
/* {{{ MADB_Date2Sql */
SQLRETURN MADB_Date2Sql(MADB_Stmt *Stmt, MADB_DescRecord *CRec, void* DataPtr, SQLLEN Length,
MADB_DescRecord *SqlRec, MYSQL_BIND *MaBind, void **Buffer, unsigned long *LengthPtr)
{
MYSQL_TIME *tm= NULL, **BuffPtr= (MYSQL_TIME**)Buffer;
SQL_DATE_STRUCT *ts= (SQL_DATE_STRUCT *)DataPtr;
if (*BuffPtr == NULL)
{
tm= (MYSQL_TIME*)MADB_GetBufferForSqlValue(Stmt, CRec, sizeof(MYSQL_TIME));
if (tm == NULL)
{
/* Error is set in function responsible for allocation */
return Stmt->Error.ReturnValue;
}
*BuffPtr= tm;
}
else
{
tm= *BuffPtr;
}
tm->year= ts->year;
tm->month= ts->month;
tm->day= ts->day;
tm->hour= tm->minute= tm->second= tm->second_part= 0;
tm->time_type= MYSQL_TIMESTAMP_DATE;
MaBind->buffer_type= MYSQL_TYPE_DATE;
*LengthPtr= sizeof(MYSQL_TIME);
return SQL_SUCCESS;
}
/* }}} */
/* {{{ MADB_ConvertC2Sql */
SQLRETURN MADB_ConvertC2Sql(MADB_Stmt *Stmt, MADB_DescRecord *CRec, void* DataPtr, SQLLEN Length,
MADB_DescRecord *SqlRec, MYSQL_BIND *MaBind, void **Buffer, unsigned long *LengthPtr)
{
if (Buffer == NULL)
{
MaBind->buffer= NULL;
Buffer= &MaBind->buffer;
}
if (LengthPtr == NULL)
{
LengthPtr= &MaBind->buffer_length;
}
/* Switch to fill BIND structures based on C and SQL type */
switch (CRec->ConciseType)
{
case WCHAR_TYPES:
RETURN_ERROR_OR_CONTINUE(MADB_Wchar2Sql(Stmt, CRec, DataPtr, Length, SqlRec, MaBind, Buffer, LengthPtr));
break;
case CHAR_BINARY_TYPES:
RETURN_ERROR_OR_CONTINUE(MADB_Char2Sql(Stmt, CRec, DataPtr, Length, SqlRec, MaBind, Buffer, LengthPtr));
break;
case SQL_C_NUMERIC:
RETURN_ERROR_OR_CONTINUE(MADB_Numeric2Sql(Stmt, CRec, DataPtr, Length, SqlRec, MaBind, Buffer, LengthPtr));
break;
case SQL_C_TIMESTAMP:
case SQL_TYPE_TIMESTAMP:
RETURN_ERROR_OR_CONTINUE(MADB_Timestamp2Sql(Stmt, CRec, DataPtr, Length, SqlRec, MaBind, Buffer, LengthPtr));
break;
case SQL_C_TIME:
case SQL_C_TYPE_TIME:
RETURN_ERROR_OR_CONTINUE(MADB_Time2Sql(Stmt, CRec, DataPtr, Length, SqlRec, MaBind, Buffer, LengthPtr));
break;
case SQL_C_INTERVAL_HOUR_TO_MINUTE:
case SQL_C_INTERVAL_HOUR_TO_SECOND:
RETURN_ERROR_OR_CONTINUE(MADB_IntervalHtoMS2Sql(Stmt, CRec, DataPtr, Length, SqlRec, MaBind, Buffer, LengthPtr));
break;
case SQL_C_DATE:
case SQL_TYPE_DATE:
RETURN_ERROR_OR_CONTINUE(MADB_Date2Sql(Stmt, CRec, DataPtr, Length, SqlRec, MaBind, Buffer, LengthPtr));
break;
default:
/* memset(MaBind, 0, sizeof(MYSQL_BIND));
MaBind->buffer_length= 0; */
MaBind->buffer_type= 0;
MaBind->is_unsigned= 0;
*LengthPtr= (unsigned long)Length;
MaBind->buffer_type= MADB_GetMaDBTypeAndLength(CRec->ConciseType,
&MaBind->is_unsigned, &MaBind->buffer_length);
if (!CRec->OctetLength)
{
CRec->OctetLength= MaBind->buffer_length;
}
*Buffer= DataPtr;
} /* End of switch (CRec->ConsiseType) */
return SQL_SUCCESS;
}
/* }}} */
/* {{{ MADB_C2SQL */
/* Main entrance function for C type to SQL type conversion*/
SQLRETURN MADB_C2SQL(MADB_Stmt* Stmt, MADB_DescRecord *CRec, MADB_DescRecord *SqlRec, SQLULEN ParamSetIdx, MYSQL_BIND *MaBind)
{
SQLLEN *IndicatorPtr= NULL;
SQLLEN *OctetLengthPtr= NULL;
void *DataPtr= NULL;
SQLLEN Length= 0;
IndicatorPtr= (SQLLEN *)GetBindOffset(Stmt->Apd, CRec, CRec->IndicatorPtr, ParamSetIdx, sizeof(SQLLEN));
OctetLengthPtr= (SQLLEN *)GetBindOffset(Stmt->Apd, CRec, CRec->OctetLengthPtr, ParamSetIdx, sizeof(SQLLEN));
if (PARAM_IS_DAE(OctetLengthPtr))
{
if (!DAE_DONE(Stmt))
{
return SQL_NEED_DATA;
}
else
{
MaBind->buffer_type= MADB_GetMaDBTypeAndLength(CRec->ConciseType, &MaBind->is_unsigned, &MaBind->buffer_length);
/* I guess we can leave w/out this. Keeping it so far for safety */
MaBind->long_data_used= '\1';
return SQL_SUCCESS;
}
} /* -- End of DAE parameter processing -- */
if (IndicatorPtr && MADB_ProcessIndicator(Stmt, *IndicatorPtr, CRec->DefaultValue, MaBind))
{
return SQL_SUCCESS;
}
/* -- Special cases are done, i.e. not a DAE etc, general case -- */
DataPtr= GetBindOffset(Stmt->Apd, CRec, CRec->DataPtr, ParamSetIdx, CRec->OctetLength);
/* If indicator wasn't NULL_DATA, but data pointer is still NULL, we convert NULL value */
if (!DataPtr)
{
return MADB_ConvertNullValue(Stmt, MaBind);
}
Length= MADB_CalculateLength(Stmt, OctetLengthPtr, CRec, DataPtr);
RETURN_ERROR_OR_CONTINUE(MADB_ConvertC2Sql(Stmt, CRec, DataPtr, Length, SqlRec, MaBind, NULL, NULL));
return SQL_SUCCESS;
}
/* }}} */
C
1
https://gitee.com/openlookeng/hetu-odbc-driver.git
git@gitee.com:openlookeng/hetu-odbc-driver.git
openlookeng
hetu-odbc-driver
hetu-odbc-driver
master

搜索帮助