LCOV - code coverage report
Current view: top level - src/ctlib - cs.c (source / functions) Hit Total Coverage
Test: FreeTDS coverage Lines: 438 721 60.7 %
Date: 2026-03-26 08:28:16 Functions: 26 42 61.9 %

          Line data    Source code
       1             : /* FreeTDS - Library of routines accessing Sybase and Microsoft databases
       2             :  * Copyright (C) 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005 Brian Bruns
       3             :  * Copyright (C) 2015  Frediano Ziglio
       4             :  *
       5             :  * This library is free software; you can redistribute it and/or
       6             :  * modify it under the terms of the GNU Library General Public
       7             :  * License as published by the Free Software Foundation; either
       8             :  * version 2 of the License, or (at your option) any later version.
       9             :  *
      10             :  * This library is distributed in the hope that it will be useful,
      11             :  * but WITHOUT ANY WARRANTY; without even the implied warranty of
      12             :  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
      13             :  * Library General Public License for more details.
      14             :  *
      15             :  * You should have received a copy of the GNU Library General Public
      16             :  * License along with this library; if not, write to the
      17             :  * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
      18             :  * Boston, MA 02111-1307, USA.
      19             :  */
      20             : 
      21             : #include <config.h>
      22             : 
      23             : #include <stdarg.h>
      24             : 
      25             : #include <freetds/time.h>
      26             : 
      27             : #include <stdio.h>
      28             : #include <assert.h>
      29             : 
      30             : #if HAVE_STDLIB_H
      31             : #include <stdlib.h>
      32             : #endif /* HAVE_STDLIB_H */
      33             : 
      34             : #if HAVE_STRING_H
      35             : #include <string.h>
      36             : #endif /* HAVE_STRING_H */
      37             : 
      38             : #include <freetds/replacements.h>
      39             : #include <freetds/utils.h>
      40             : 
      41             : #include "cspublic.h"
      42             : #include "ctlib.h"
      43             : 
      44             : #undef cs_dt_crack
      45             : 
      46             : static CS_INT cs_diag_storemsg(CS_CONTEXT *context, CS_CLIENTMSG *message);
      47             : static CS_INT cs_diag_clearmsg(CS_CONTEXT *context, CS_INT type);
      48             : static CS_INT cs_diag_getmsg(CS_CONTEXT *context, CS_INT idx, CS_CLIENTMSG *message);
      49             : static CS_INT cs_diag_countmsg(CS_CONTEXT *context, CS_INT *count);
      50             : 
      51             : const char *
      52           0 : cs_prretcode(int retcode)
      53             : {
      54             :         static char unknown[24];
      55             :         
      56           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_prretcode(%d)\n", retcode);
      57             : 
      58           0 :         switch(retcode) {
      59             :         case CS_SUCCEED:        return "CS_SUCCEED";
      60           0 :         case CS_FAIL:           return "CS_FAIL";
      61           0 :         case CS_MEM_ERROR:      return "CS_MEM_ERROR";
      62           0 :         case CS_PENDING:        return "CS_PENDING";
      63           0 :         case CS_QUIET:          return "CS_QUIET";
      64           0 :         case CS_BUSY:           return "CS_BUSY";
      65           0 :         case CS_INTERRUPT:      return "CS_INTERRUPT";
      66           0 :         case CS_BLK_HAS_TEXT:   return "CS_BLK_HAS_TEXT";
      67           0 :         case CS_CONTINUE:       return "CS_CONTINUE";
      68           0 :         case CS_FATAL:          return "CS_FATAL";
      69           0 :         case CS_RET_HAFAILOVER: return "CS_RET_HAFAILOVER";
      70           0 :         case CS_UNSUPPORTED:    return "CS_UNSUPPORTED";
      71             : 
      72           0 :         case CS_CANCELED:       return "CS_CANCELED";
      73           0 :         case CS_ROW_FAIL:       return "CS_ROW_FAIL";
      74           0 :         case CS_END_DATA:       return "CS_END_DATA";
      75           0 :         case CS_END_RESULTS:    return "CS_END_RESULTS";
      76           0 :         case CS_END_ITEM:       return "CS_END_ITEM";
      77           0 :         case CS_NOMSG:          return "CS_NOMSG";
      78           0 :         case CS_TIMED_OUT:      return "CS_TIMED_OUT";
      79             : 
      80           0 :         default:
      81           0 :                 sprintf(unknown, "oops: %u ??", retcode);
      82             :         }
      83           0 :         return unknown;
      84             : }
      85             : 
      86             : static const char *
      87         630 : _cs_get_layer(int layer)
      88             : {
      89         630 :         tdsdump_log(TDS_DBG_FUNC, "_cs_get_layer(%d)\n", layer);
      90             : 
      91         630 :         switch (layer) {
      92             :         case 2:
      93             :                 return "cslib user api layer";
      94             :                 break;
      95             :         default:
      96             :                 break;
      97             :         }
      98           0 :         return "unrecognized layer";
      99             : }
     100             : 
     101             : static const char *
     102         630 : _cs_get_origin(int origin)
     103             : {
     104         630 :         tdsdump_log(TDS_DBG_FUNC, "_cs_get_origin(%d)\n", origin);
     105             : 
     106         630 :         switch (origin) {
     107             :         case 1:
     108             :                 return "external error";
     109             :                 break;
     110           0 :         case 2:
     111           0 :                 return "internal CS-Library error";
     112             :                 break;
     113         200 :         case 4:
     114         200 :                 return "common library error";
     115             :                 break;
     116           0 :         case 5:
     117           0 :                 return "intl library error";
     118             :                 break;
     119             :         default:
     120             :                 break;
     121             :         }
     122           0 :         return "unrecognized origin";
     123             : }
     124             : 
     125             : static const char *
     126         630 : _cs_get_user_api_layer_error(int error)
     127             : {
     128         630 :         tdsdump_log(TDS_DBG_FUNC, "_cs_get_user_api_layer_error(%d)\n", error);
     129             : 
     130         630 :         switch (error) {
     131             :         case 2:
     132             :                 return "The information being retrieved will not fit in a buffer of %1! bytes.";
     133             :                 break;
     134           0 :         case 3:
     135           0 :                 return "Memory allocation failure.";
     136             :                 break;
     137          60 :         case 4:
     138          60 :                 return "The parameter %1! cannot be NULL.";
     139             :                 break;
     140         170 :         case 6:
     141         170 :                 return "An illegal value of %1! was given for parameter %2!.";
     142             :                 break;
     143          40 :         case 16:
     144          40 :                 return "Conversion between %1! and %2! datatypes is not supported.";
     145             :                 break;
     146         150 :         case 18:
     147         150 :                 return "An illegal value of %1! was placed in the %2! field of the CS_DATAFMT structure.";
     148             :                 break;
     149          60 :         case 20:
     150          60 :                 return "The conversion/operation resulted in overflow.";
     151             :                 break;
     152          50 :         case 24:
     153          50 :                 return "The conversion/operation was stopped due to a syntax error in the source field.";
     154             :                 break;
     155          90 :         case 36:
     156          90 :                 return "The result is truncated because the conversion/operation resulted in overflow.";
     157             :                 break;
     158             :         default:
     159             :                 break;
     160             :         }
     161           0 :         return "unrecognized error";
     162             : }
     163             : 
     164             : static char *
     165         630 : _cs_get_msgstr(const char *funcname, int layer, int origin, int severity, int number)
     166             : {
     167             :         char *m;
     168             : 
     169         630 :         tdsdump_log(TDS_DBG_FUNC, "_cs_get_msgstr(%s, %d, %d, %d, %d)\n", funcname, layer, origin, severity, number);
     170             : 
     171         630 :         if (asprintf(&m, "%s: %s: %s: %s", funcname, _cs_get_layer(layer), _cs_get_origin(origin), (layer == 2)
     172             :                      ? _cs_get_user_api_layer_error(number)
     173             :                      : "unrecognized error") < 0) {
     174             :                 return NULL;
     175             :         }
     176         630 :         return m;
     177             : }
     178             : 
     179             : static void
     180         640 : _csclient_msg(CS_CONTEXT * ctx, const char *funcname, int layer, int origin, int severity, int number, const char *fmt, ...)
     181             : {
     182             :         va_list ap;
     183             :         CS_CLIENTMSG cm;
     184             :         char *msgstr;
     185             : 
     186         640 :         tdsdump_log(TDS_DBG_FUNC, "_csclient_msg(%p, %s, %d, %d, %d, %d, %s)\n", ctx, funcname, layer, origin, severity, number, fmt);
     187             : 
     188         640 :         va_start(ap, fmt);
     189             : 
     190         640 :         if (ctx->cslibmsg_cb) {
     191         630 :                 cm.severity = severity;
     192        1260 :                 cm.msgnumber = (((layer << 24) & 0xFF000000)
     193         630 :                                 | ((origin << 16) & 0x00FF0000)
     194         630 :                                 | ((severity << 8) & 0x0000FF00)
     195         630 :                                 | ((number) & 0x000000FF));
     196         630 :                 msgstr = _cs_get_msgstr(funcname, layer, origin, severity, number);
     197         630 :                 tds_vstrbuild(cm.msgstring, CS_MAX_MSG, &(cm.msgstringlen), msgstr, CS_NULLTERM, fmt, CS_NULLTERM, ap);
     198         630 :                 cm.msgstring[cm.msgstringlen] = '\0';
     199         630 :                 free(msgstr);
     200         630 :                 cm.osnumber = 0;
     201         630 :                 cm.osstring[0] = '\0';
     202         630 :                 cm.osstringlen = 0;
     203         630 :                 cm.status = 0;
     204             :                 /* cm.sqlstate */
     205         630 :                 cm.sqlstatelen = 0;
     206         630 :                 ctx->cslibmsg_cb(ctx, &cm);
     207             :         }
     208             : 
     209         640 :         va_end(ap);
     210         640 : }
     211             : 
     212             : /**
     213             :  * Allocate new CS_LOCALE and initialize it
     214             :  * returns NULL on out of memory errors 
     215             :  */
     216             : static CS_LOCALE *
     217          20 : _cs_locale_alloc(void)
     218             : {
     219          20 :         tdsdump_log(TDS_DBG_FUNC, "_cs_locale_alloc()\n");
     220             : 
     221          20 :         return tds_new0(CS_LOCALE, 1);
     222             : }
     223             : 
     224             : static void
     225          20 : _cs_locale_free_contents(CS_LOCALE *locale)
     226             : {
     227          20 :         tdsdump_log(TDS_DBG_FUNC, "_cs_locale_free_contents(%p)\n", locale);
     228             : 
     229             :         /* free strings */
     230          20 :         free(locale->language);
     231          20 :         locale->language = NULL;
     232          20 :         free(locale->charset);
     233          20 :         locale->charset = NULL;
     234          20 :         free(locale->time);
     235          20 :         locale->time = NULL;
     236          20 :         free(locale->collate);
     237          20 :         locale->collate = NULL;
     238          20 : }
     239             : 
     240             : void 
     241          20 : _cs_locale_free(CS_LOCALE *locale)
     242             : {
     243          20 :         tdsdump_log(TDS_DBG_FUNC, "_cs_locale_free(%p)\n", locale);
     244             : 
     245             :         /* free contents */
     246          20 :         _cs_locale_free_contents(locale);
     247             : 
     248             :         /* free the data structure */
     249          20 :         free(locale);
     250          20 : }
     251             : 
     252             : /* returns 0 on out of memory errors, 1 for OK */
     253             : int
     254           0 : _cs_locale_copy_inplace(CS_LOCALE *new_locale, CS_LOCALE *orig)
     255             : {
     256           0 :         tdsdump_log(TDS_DBG_FUNC, "_cs_locale_copy_inplace(%p, %p)\n", new_locale, orig);
     257             : 
     258           0 :         _cs_locale_free_contents(new_locale);
     259           0 :         if (orig->language) {
     260           0 :                 new_locale->language = strdup(orig->language);
     261           0 :                 if (!new_locale->language)
     262             :                         goto Cleanup;
     263             :         }
     264             : 
     265           0 :         if (orig->charset) {
     266           0 :                 new_locale->charset = strdup(orig->charset);
     267           0 :                 if (!new_locale->charset)
     268             :                         goto Cleanup;
     269             :         }
     270             : 
     271           0 :         if (orig->time) {
     272           0 :                 new_locale->time = strdup(orig->time);
     273           0 :                 if (!new_locale->time)
     274             :                         goto Cleanup;
     275             :         }
     276             : 
     277           0 :         if (orig->collate) {
     278           0 :                 new_locale->collate = strdup(orig->collate);
     279           0 :                 if (!new_locale->collate)
     280             :                         goto Cleanup;
     281             :         }
     282             : 
     283             :         return 1;
     284             : 
     285           0 : Cleanup:
     286           0 :         _cs_locale_free_contents(new_locale);
     287           0 :         return 0;
     288             : }
     289             : 
     290             : /* returns NULL on out of memory errors */
     291             : CS_LOCALE *
     292           0 : _cs_locale_copy(CS_LOCALE *orig)
     293             : {
     294             :         CS_LOCALE *new_locale;
     295             : 
     296           0 :         tdsdump_log(TDS_DBG_FUNC, "_cs_locale_copy(%p)\n", orig);
     297             : 
     298           0 :         new_locale = _cs_locale_alloc();
     299           0 :         if (!new_locale)
     300             :                 return NULL;
     301             : 
     302           0 :         if (orig->language) {
     303           0 :                 new_locale->language = strdup(orig->language);
     304           0 :                 if (!new_locale->language)
     305             :                         goto Cleanup;
     306             :         }
     307             : 
     308           0 :         if (orig->charset) {
     309           0 :                 new_locale->charset = strdup(orig->charset);
     310           0 :                 if (!new_locale->charset)
     311             :                         goto Cleanup;
     312             :         }
     313             : 
     314           0 :         if (orig->time) {
     315           0 :                 new_locale->time = strdup(orig->time);
     316           0 :                 if (!new_locale->time)
     317             :                         goto Cleanup;
     318             :         }
     319             : 
     320           0 :         if (orig->collate) {
     321           0 :                 new_locale->collate = strdup(orig->collate);
     322           0 :                 if (!new_locale->collate)
     323             :                         goto Cleanup;
     324             :         }
     325             : 
     326             :         return new_locale;
     327             : 
     328           0 : Cleanup:
     329           0 :         _cs_locale_free(new_locale);
     330           0 :         return NULL;
     331             : }
     332             : 
     333             : CS_RETCODE
     334        1410 : cs_ctx_alloc(CS_INT version, CS_CONTEXT ** out_ctx)
     335             : {
     336             :         TDSCONTEXT *tds_ctx;
     337             :         CS_CONTEXT *ctx;
     338             : 
     339        1410 :         tdsdump_log(TDS_DBG_FUNC, "cs_ctx_alloc(%d, %p)\n", version, out_ctx);
     340             : 
     341        1410 :         ctx = tds_new0(CS_CONTEXT, 1);
     342        1410 :         if (!ctx)
     343             :                 return CS_FAIL;
     344        1410 :         ctx->use_large_identifiers = _ct_is_large_identifiers_version(version);
     345        1410 :         tds_ctx = tds_alloc_context(ctx);
     346        1410 :         if (!tds_ctx) {
     347           0 :                 free(ctx);
     348           0 :                 return CS_FAIL;
     349             :         }
     350        1410 :         ctx->tds_ctx = tds_ctx;
     351        1410 :         if (tds_ctx->locale && !tds_ctx->locale->datetime_fmt) {
     352             :                 /* set default in case there's no locale file */
     353           0 :                 tds_ctx->locale->datetime_fmt = strdup(STD_DATETIME_FMT);
     354             :         }
     355             : 
     356        1410 :         ctx->login_timeout = -1;
     357        1410 :         ctx->query_timeout = -1;
     358             : 
     359        1410 :         *out_ctx = ctx;
     360        1410 :         return CS_SUCCEED;
     361             : }
     362             : 
     363             : CS_RETCODE
     364           0 : cs_ctx_global(CS_INT version, CS_CONTEXT ** ctx)
     365             : {
     366             :         static CS_CONTEXT *global_cs_ctx = NULL;
     367             : 
     368           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_ctx_global(%d, %p)\n", version, ctx);
     369             : 
     370           0 :         if (global_cs_ctx != NULL) {
     371           0 :                 *ctx = global_cs_ctx;
     372           0 :                 return CS_SUCCEED;
     373             :         }
     374           0 :         if (cs_ctx_alloc(version, ctx) != CS_SUCCEED) {
     375             :                 return CS_FAIL;
     376             :         }
     377           0 :         global_cs_ctx = *ctx;
     378           0 :         return CS_SUCCEED;
     379             : }
     380             : 
     381             : CS_RETCODE
     382        1410 : cs_ctx_drop(CS_CONTEXT * ctx)
     383             : {
     384        1410 :         tdsdump_log(TDS_DBG_FUNC, "cs_ctx_drop(%p)\n", ctx);
     385             : 
     386        1410 :         if (ctx) {
     387        1410 :                 _ct_diag_clearmsg(ctx, CS_ALLMSG_TYPE);
     388        1410 :                 free(ctx->userdata);
     389        1410 :                 if (ctx->tds_ctx)
     390        1410 :                         tds_free_context(ctx->tds_ctx);
     391        1410 :                 free(ctx);
     392             :         }
     393        1410 :         return CS_SUCCEED;
     394             : }
     395             : 
     396             : CS_RETCODE
     397         250 : cs_config(CS_CONTEXT * ctx, CS_INT action, CS_INT property, CS_VOID * buffer, CS_INT buflen, CS_INT * outlen)
     398             : {
     399             :         CS_INT maxcp;   /* declared  for - CS_USERDATA changes - swapna*/
     400             : 
     401         250 :         tdsdump_log(TDS_DBG_FUNC, "cs_config(%p, %d, %d, %p, %d, %p)\n", ctx, action, property, buffer, buflen, outlen);
     402             : 
     403         250 :         if (action == CS_GET) {
     404          80 :                 if (buffer == NULL) {
     405             :                         return CS_SUCCEED;
     406             :                 }
     407          80 :                 switch (property) {
     408           0 :                 case CS_MESSAGE_CB:
     409           0 :                         *(CS_CSLIBMSG_FUNC*) buffer = ctx->cslibmsg_cb;
     410           0 :                         return CS_SUCCEED;
     411          80 :                 case CS_USERDATA:
     412          80 :                         if (buflen < 0) {
     413          10 :                                 _csclient_msg(ctx, "cs_config", 2, 1, 1, 6, "%d, %s", buflen, "buflen");
     414          10 :                                 return CS_FAIL;
     415             :                         }
     416             : 
     417          70 :                         maxcp = ctx->userdata_len;
     418          70 :                         if (outlen)
     419          40 :                                 *outlen = maxcp;
     420             : 
     421          70 :                         if (buflen < maxcp) {
     422          10 :                                 _csclient_msg(ctx, "cs_config", 2, 1, 1, 2, "%d", buflen);
     423          10 :                                 return CS_FAIL;
     424             :                         }
     425          60 :                         memcpy(buffer, ctx->userdata, maxcp); 
     426          60 :                         return CS_SUCCEED;
     427             : 
     428           0 :                 default:
     429             :                 case CS_EXTRA_INF:
     430             :                 case CS_LOC_PROP:
     431             :                 case CS_VERSION:
     432           0 :                         _csclient_msg(ctx, "cs_config", 2, 1, 1, 6, "%d, %s", property, "property");
     433           0 :                         return CS_FAIL;
     434             :                         break;
     435             :                 }
     436             :         }
     437         170 :         if (action == CS_SET) {
     438         160 :                 switch (property) {
     439          40 :                 case CS_MESSAGE_CB:
     440          40 :                         if ( ctx->cs_errhandletype == _CS_ERRHAND_INLINE) {
     441           0 :                                 cs_diag_clearmsg(ctx, CS_UNUSED);
     442             :                         }
     443          40 :                         ctx->cslibmsg_cb = (CS_CSLIBMSG_FUNC) buffer;
     444          40 :                         ctx->cs_errhandletype = _CS_ERRHAND_CB;
     445          40 :                         return CS_SUCCEED;
     446         110 :                 case CS_USERDATA:
     447         110 :                         if (buflen < 0 && buflen != CS_NULLTERM) {
     448          60 :                                 _csclient_msg(ctx, "cs_config", 2, 1, 1, 6, "%d, %s", buflen, "buflen");
     449          60 :                                 return CS_FAIL;
     450             :                         }
     451             : 
     452          50 :                         maxcp = _ct_get_string_length(buffer, buflen);
     453          50 :                         free(ctx->userdata);
     454          50 :                         ctx->userdata = (void *) malloc(maxcp);
     455          50 :                         if ( ctx->userdata == NULL) {
     456             :                                 return CS_FAIL;
     457             :                         }
     458          50 :                         ctx->userdata_len = maxcp;
     459             :         
     460          50 :                         if (buffer) {
     461          50 :                                 memcpy(ctx->userdata, buffer, maxcp);
     462             :                         } else {
     463             :                                 return CS_FAIL;
     464             :                         }
     465          50 :                         return CS_SUCCEED;
     466             : 
     467          10 :                 default:
     468             :                 case CS_EXTRA_INF:
     469             :                 case CS_LOC_PROP:
     470             :                 case CS_VERSION:
     471          10 :                         _csclient_msg(ctx, "cs_config", 2, 1, 1, 6, "%d, %s", property, "property");
     472          10 :                         return CS_FAIL;
     473             :                         break;
     474             :                 }
     475             :         }
     476          10 :         if (action == CS_CLEAR) {
     477           0 :                 switch (property) {
     478           0 :                 case CS_MESSAGE_CB:
     479           0 :                         if ( ctx->cs_errhandletype == _CS_ERRHAND_INLINE) {
     480           0 :                                 cs_diag_clearmsg(ctx, CS_UNUSED);
     481             :                         }
     482           0 :                         ctx->cslibmsg_cb = NULL;
     483           0 :                         ctx->cs_errhandletype = 0;
     484           0 :                         return CS_SUCCEED;
     485           0 :                 case CS_USERDATA:
     486           0 :                         free(ctx->userdata);
     487           0 :                         ctx->userdata = NULL;
     488           0 :                         ctx->userdata_len = 0;
     489             :         
     490           0 :                         return CS_SUCCEED;
     491             : 
     492           0 :                 default:
     493             :                 case CS_EXTRA_INF:
     494             :                 case CS_LOC_PROP:
     495             :                 case CS_VERSION:
     496           0 :                         _csclient_msg(ctx, "cs_config", 2, 1, 1, 6, "%d, %s", property, "property");
     497           0 :                         return CS_FAIL;
     498             :                         break;
     499             :                 }
     500             :         }
     501             : 
     502          10 :         _csclient_msg(ctx, "cs_config", 2, 1, 1, 6, "%d, %s", action, "action");
     503          10 :         return CS_FAIL;
     504             : }
     505             : 
     506             : TDS_INT
     507        9215 : _cs_cs2tds(CS_CONTEXT *ctx, const CS_DATAFMT_COMMON *srcfmt, const void *srcdata, int desttype, CONV_RESULT *cres)
     508             : {
     509             :         TDS_SERVER_TYPE src_type;
     510             :         int src_len, len;
     511             :         CS_INT datatype;
     512             : 
     513        9215 :         datatype = srcfmt->datatype;
     514        9215 :         src_type = _ct_get_server_type(NULL, datatype);
     515        9215 :         if (src_type == TDS_INVALID_TYPE)
     516             :                 return TDS_CONVERT_NOAVAIL;
     517             : 
     518        9205 :         src_len = srcfmt->maxlength;
     519        9205 :         if (datatype == CS_VARCHAR_TYPE || datatype == CS_VARBINARY_TYPE) {
     520           0 :                 const CS_VARCHAR *vc = (const CS_VARCHAR *) srcdata;
     521             : 
     522             :                 /* Prevent overflows, Sybase accept wrong values as correct */
     523           0 :                 src_len = TDS_CLAMP(vc->len, 0, sizeof(vc->str));
     524           0 :                 srcdata = vc->str;
     525             :         }
     526             : 
     527             :         /* conversion from char to binary are a bit peculiar */
     528        9205 :         if (desttype == TDS_CONVERT_BINARY && is_char_type(src_type)) {
     529         180 :                 const char *src = srcdata;
     530             :                 TDS_INT full_len;
     531             : 
     532         180 :                 src_len = tds_hex_trim(&src, src_len);
     533             : 
     534         180 :                 full_len = ((unsigned int) src_len + 1) / 2;
     535         180 :                 if (full_len > cres->cb.len)
     536          40 :                         src_len = 2 * cres->cb.len - (src_len & 1);
     537             : 
     538         180 :                 len = tds_char2hex(cres->cb.ib, cres->cb.len, src, src_len);
     539         180 :                 return len < 0 ? len : full_len;
     540             :         }
     541             : 
     542        9025 :         tdsdump_log(TDS_DBG_FUNC, "converting type %d (%d bytes) to type = %d\n", src_type, src_len, desttype);
     543             : 
     544        9025 :         len = tds_convert(ctx->tds_ctx, src_type, srcdata, src_len, desttype, cres);
     545             : 
     546        9025 :         tdsdump_log(TDS_DBG_FUNC, "_cs_cs2tds() tds_convert returned %d\n", len);
     547             : 
     548        9025 :         switch (len) {
     549             :         case TDS_CONVERT_NOAVAIL:
     550             :         case TDS_CONVERT_SYNTAX:
     551             :         case TDS_CONVERT_OVERFLOW:
     552             :                 /* for these errors messages depends on caller */
     553             :                 break;
     554           0 :         case TDS_CONVERT_NOMEM:
     555           0 :                 _csclient_msg(ctx, "cs_convert", 2, 4, 1, 3, "");
     556           0 :                 break;
     557             :         case TDS_CONVERT_FAIL:
     558             :                 /* TODO error reporting */
     559             :                 break;
     560             :         }
     561             :         return len;
     562             : }
     563             : 
     564             : static inline bool
     565             : is_power_of_two_or_zero(CS_INT value)
     566             : {
     567        6327 :         return (value & (value - 1)) == 0;
     568             : }
     569             : 
     570             : CS_RETCODE
     571        7485 : _cs_convert(CS_CONTEXT *ctx, const CS_DATAFMT_COMMON *srcfmt, CS_VOID *srcdata,
     572             :             const CS_DATAFMT_COMMON *destfmt, CS_VOID *destdata, CS_INT *resultlen)
     573             : {
     574        7485 :         CONV_RESULT cr, *p_cr = &cr;
     575             :         TDS_INT res;
     576             :         CS_INT dummy, format;
     577        7485 :         CS_VARCHAR *destvc = NULL;
     578        7485 :         CS_RETCODE rc = CS_SUCCEED;
     579        7485 :         CS_INT maxlength = -1;
     580             :         TDS_SERVER_TYPE desttype;
     581             : 
     582        7485 :         tdsdump_log(TDS_DBG_FUNC, "cs_convert(%p, %p, %p, %p, %p, %p)\n", ctx, srcfmt, srcdata, destfmt, destdata, resultlen);
     583             : 
     584             :         /* If srcfmt is NULL we have a problem */
     585        7485 :         if (srcfmt == NULL) {
     586           0 :                 _csclient_msg(ctx, "cs_convert", 2, 1, 1, 4, "srcfmt");
     587           0 :                 return CS_FAIL;
     588             :         }
     589             : 
     590             :         /* If destfmt is NULL we have a problem */
     591        7485 :         if (destfmt == NULL) {
     592          20 :                 _csclient_msg(ctx, "cs_convert", 2, 1, 1, 4, "destfmt");
     593          20 :                 return CS_FAIL;
     594             :         }
     595             : 
     596        7465 :         format = destfmt->format;
     597             : 
     598             :         /* If destination is NULL we have a problem */
     599        7465 :         if (destdata == NULL) {
     600          10 :                 _csclient_msg(ctx, "cs_convert", 2, 1, 1, 4, "destdata");
     601          10 :                 return CS_FAIL;
     602             : 
     603             :         }
     604             : 
     605        7455 :         if (resultlen == NULL)
     606          40 :                 resultlen = &dummy;
     607             : 
     608             :         /* If source is indicated to be NULL, set dest to low values */
     609        7455 :         if (srcdata == NULL) {
     610             :                 /* TODO: implement cs_setnull */
     611           0 :                 tdsdump_log(TDS_DBG_FUNC, "srcdata is null\n");
     612           0 :                 memset(destdata, '\0', destfmt->maxlength);
     613           0 :                 *resultlen = 0;
     614           0 :                 return CS_SUCCEED;
     615             :         }
     616             : 
     617             :         /* more or less from _ct_get_server_type */
     618        7455 :         switch (destfmt->datatype) {
     619             :         case CS_VARCHAR_TYPE:
     620             :                 desttype = TDS_CONVERT_CHAR;
     621             :                 goto cs_var_common;
     622          20 :         case CS_VARBINARY_TYPE:
     623          20 :                 desttype = TDS_CONVERT_BINARY;
     624          30 :               cs_var_common:
     625          30 :                 destvc = (CS_VARCHAR *) destdata;
     626          30 :                 cr.cc.c = destvc->str;
     627          30 :                 cr.cc.len = sizeof(destvc->str);
     628          30 :                 maxlength = sizeof(destvc->str);
     629          30 :                 format = CS_FMT_UNUSED;
     630          30 :                 break;
     631         280 :         case CS_BINARY_TYPE:
     632             :         case CS_IMAGE_TYPE:
     633             :         case CS_LONGBINARY_TYPE:
     634         280 :                 desttype = TDS_CONVERT_BINARY;
     635         280 :                 if ((format & (~CS_FMT_PADNULL)) != 0) {
     636          60 :                         _csclient_msg(ctx, "cs_convert", 2, 1, 1, 18, "%d, %s", format, "format");
     637          60 :                         return CS_FAIL;
     638             :                 }
     639             :                 goto cs_char_binary_common;
     640        6347 :         case CS_CHAR_TYPE:
     641             :         case CS_TEXT_TYPE:
     642             :         case CS_UNICHAR_TYPE:
     643             :         case CS_LONGCHAR_TYPE:
     644        6347 :                 desttype = TDS_CONVERT_CHAR;
     645       12674 :                 if ((format & (~(CS_FMT_NULLTERM | CS_FMT_PADNULL | CS_FMT_PADBLANK))) != 0 || !is_power_of_two_or_zero(format)) {
     646          20 :                         _csclient_msg(ctx, "cs_convert", 2, 1, 1, 18, "%d, %s", format, "format");
     647          20 :                         return CS_FAIL;
     648             :                 }
     649        6327 :               cs_char_binary_common:
     650        6547 :                 cr.cc.c = destdata;
     651        6547 :                 cr.cc.len = destfmt->maxlength;
     652        6547 :                 maxlength = destfmt->maxlength;
     653        6547 :                 if (maxlength < 0) {
     654          10 :                         _csclient_msg(ctx, "cs_convert", 2, 1, 1, 18, "%d, %s", maxlength, "maxlength");
     655          10 :                         return CS_FAIL;
     656             :                 }
     657             :                 break;
     658         130 :         case CS_NUMERIC_TYPE:
     659             :         case CS_DECIMAL_TYPE:
     660         130 :                 desttype = SYBNUMERIC;
     661         130 :                 format = CS_FMT_UNUSED;
     662             :                 {
     663             :                         /* set the output precision/scale for conversions to numeric type */
     664         130 :                         CS_INT precision = destfmt->precision;
     665         130 :                         CS_INT scale = destfmt->scale;
     666         130 :                         bool src_is_numeric = (srcfmt->datatype == CS_NUMERIC_TYPE || srcfmt->datatype == CS_DECIMAL_TYPE);
     667             : 
     668         130 :                         if (destfmt->precision == CS_SRC_VALUE && src_is_numeric)
     669          10 :                                 precision = ((TDS_NUMERIC *) srcdata)->precision;
     670         130 :                         if (destfmt->scale == CS_SRC_VALUE && src_is_numeric)
     671          20 :                                 scale = ((TDS_NUMERIC *) srcdata)->scale;
     672         130 :                         if (precision < 1 || precision > MAXPRECISION) {
     673          30 :                                 _csclient_msg(ctx, "cs_convert", 2, 1, 1, 18, "%d, %s", precision, "precision");
     674          30 :                                 return CS_FAIL;
     675             :                         }
     676         100 :                         if (scale < 0 || scale > precision) {
     677          30 :                                 _csclient_msg(ctx, "cs_convert", 2, 1, 1, 18, "%d, %s", scale, "scale");
     678          30 :                                 return CS_FAIL;
     679             :                         }
     680          70 :                         cr.n.precision = precision;
     681          70 :                         cr.n.scale = scale;
     682             :                 }
     683          70 :                 break;
     684         668 :         default:
     685         668 :                 desttype = _ct_get_server_type(NULL, destfmt->datatype);
     686         668 :                 if (desttype == TDS_INVALID_TYPE) {
     687          10 :                         _csclient_msg(ctx, "cs_convert", 2, 1, 1, 16, "%d, %d", srcfmt->datatype, destfmt->datatype);
     688          10 :                         return CS_FAIL;
     689             :                 }
     690             :                 p_cr = (CONV_RESULT *) destdata;
     691             :                 break;
     692             :         }
     693             : 
     694        7295 :         res = _cs_cs2tds(ctx, srcfmt, srcdata, desttype, p_cr);
     695        7295 :         if (res < 0) {
     696             :                 /* other errors are already reported */
     697         150 :                 switch (res) {
     698          40 :                 case TDS_CONVERT_NOAVAIL:
     699          40 :                         _csclient_msg(ctx, "cs_convert", 2, 1, 1, 16, "%d, %d", srcfmt->datatype, destfmt->datatype);
     700          40 :                         break;
     701          50 :                 case TDS_CONVERT_SYNTAX:
     702          50 :                         _csclient_msg(ctx, "cs_convert", 2, 4, 1, 24, "");
     703          50 :                         break;
     704          60 :                 case TDS_CONVERT_OVERFLOW:
     705          60 :                         _csclient_msg(ctx, "cs_convert", 2, 4, 1, 20, "");
     706          60 :                         break;
     707             :                 }
     708             :                 return CS_FAIL;
     709             :         }
     710             : 
     711        7145 :         if (maxlength >= 0) {
     712             :                 /* overflow check */
     713        6537 :                 if (res > maxlength) {
     714          90 :                         tdsdump_log(TDS_DBG_FUNC, "Data-conversion resulted in overflow\n");
     715          90 :                         _csclient_msg(ctx, "cs_convert", 2, 4, 1, 36, "");
     716          90 :                         res = maxlength;
     717          90 :                         rc = CS_FAIL;
     718             :                 }
     719             : 
     720             :                 /* padding */
     721        6537 :                 switch (format) {
     722        4533 :                 case CS_FMT_NULLTERM:
     723        4533 :                         tdsdump_log(TDS_DBG_FUNC, "cs_convert() FMT_NULLTERM\n");
     724        4533 :                         if (res < maxlength) {
     725        4533 :                                 cr.cc.c[res] = 0;
     726        4533 :                                 res++;
     727           0 :                         } else if (res == maxlength && rc == CS_SUCCEED) {
     728           0 :                                 tdsdump_log(TDS_DBG_FUNC, "not enough room for data + a null terminator - error\n");
     729           0 :                                 _csclient_msg(ctx, "cs_convert", 2, 4, 1, 36, "");
     730           0 :                                 rc = CS_FAIL;   /* not enough room for data + a null terminator - error */
     731             :                         }
     732             :                         break;
     733          10 :                 case CS_FMT_PADBLANK:
     734          10 :                         tdsdump_log(TDS_DBG_FUNC, "cs_convert() FMT_PADBLANK\n");
     735          10 :                         memset(cr.cc.c + res, ' ', maxlength - res);
     736          10 :                         res = maxlength;
     737          10 :                         break;
     738          40 :                 case CS_FMT_PADNULL:
     739          40 :                         tdsdump_log(TDS_DBG_FUNC, "cs_convert() FMT_PADNULL\n");
     740          40 :                         memset(cr.cc.c + res, '\0', maxlength - res);
     741          40 :                         res = maxlength;
     742          40 :                         break;
     743        1954 :                 case CS_FMT_UNUSED:
     744        1954 :                         tdsdump_log(TDS_DBG_FUNC, "cs_convert() FMT_UNUSED\n");
     745             :                         break;
     746             :                 default:
     747             :                         rc = CS_FAIL;
     748             :                         break;
     749             :                 }
     750         608 :         }
     751             : 
     752             :         /* fix varchar_type/varbinary_type */
     753        7145 :         if (destvc) {
     754          30 :                 destvc->len = TDS_MIN(res, sizeof(destvc->str));
     755          30 :                 res = sizeof(*destvc);
     756             :         }
     757             : 
     758             :         /* numeric */
     759        7145 :         if (desttype == SYBNUMERIC) {
     760          70 :                 res = tds_numeric_bytes_per_prec[cr.n.precision] + 2;
     761          70 :                 memcpy(destdata, &(cr.n), res);
     762             :         }
     763             : 
     764        7145 :         *resultlen = res;
     765        7145 :         tdsdump_log(TDS_DBG_FUNC, "cs_convert() returning  %s\n", cs_prretcode(rc));
     766             :         return rc;
     767             : }
     768             : 
     769             : CS_RETCODE
     770        2730 : cs_convert(CS_CONTEXT *ctx, CS_DATAFMT *srcfmt, CS_VOID *srcdata, CS_DATAFMT *destfmt, CS_VOID *destdata, CS_INT *resultlen)
     771             : {
     772        2730 :         return _cs_convert(ctx, _ct_datafmt_common(ctx, srcfmt), srcdata, _ct_datafmt_common(ctx, destfmt), destdata, resultlen);
     773             : }
     774             : 
     775             : CS_RETCODE
     776           0 : cs_dt_crack_v2(CS_CONTEXT * ctx, CS_INT datetype, CS_VOID * dateval, CS_DATEREC * daterec)
     777             : {
     778             :         TDSDATEREC dr;
     779             :         TDS_INT tds_type;
     780           0 :         bool extended = false;
     781             : 
     782           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_dt_crack_v2(%p, %d, %p, %p)\n", ctx, datetype, dateval, daterec);
     783             : 
     784           0 :         switch (datetype) {
     785             :         case CS_DATETIME_TYPE:
     786             :                 tds_type = SYBDATETIME;
     787             :                 break;
     788           0 :         case CS_DATETIME4_TYPE:
     789           0 :                 tds_type = SYBDATETIME4;
     790           0 :                 break;
     791           0 :         case CS_DATE_TYPE:
     792           0 :                 tds_type = SYBDATE;
     793           0 :                 break;
     794           0 :         case CS_TIME_TYPE:
     795           0 :                 tds_type = SYBTIME;
     796           0 :                 break;
     797           0 :         case CS_BIGDATETIME_TYPE:
     798           0 :                 tds_type = SYB5BIGDATETIME;
     799           0 :                 extended = true;
     800           0 :                 break;
     801           0 :         case CS_BIGTIME_TYPE:
     802           0 :                 tds_type = SYB5BIGTIME;
     803           0 :                 extended = true;
     804           0 :                 break;
     805             :         default:
     806             :                 return CS_FAIL;
     807             :         }
     808           0 :         tds_datecrack(tds_type, dateval, &dr);
     809             : 
     810             :         /* Sybase CT-Library does not set these fields for CS_BIGTIME_TYPE */
     811           0 :         if (tds_type != SYB5BIGTIME) {
     812           0 :                 daterec->dateyear = dr.year;
     813           0 :                 daterec->datemonth = dr.month;
     814           0 :                 daterec->datedmonth = dr.day;
     815           0 :                 daterec->datedyear = dr.dayofyear;
     816           0 :                 daterec->datedweek = dr.weekday;
     817             :         }
     818           0 :         daterec->datehour = dr.hour;
     819           0 :         daterec->dateminute = dr.minute;
     820           0 :         daterec->datesecond = dr.second;
     821           0 :         daterec->datemsecond = dr.decimicrosecond / 10000u;
     822           0 :         daterec->datetzone = dr.timezone;
     823           0 :         if (extended) {
     824           0 :                 daterec->datesecfrac = dr.decimicrosecond / 10u;
     825           0 :                 daterec->datesecprec = 1000000;
     826             :         }
     827             : 
     828             :         return CS_SUCCEED;
     829             : }
     830             : 
     831             : CS_RETCODE
     832           0 : cs_dt_crack(CS_CONTEXT * ctx, CS_INT datetype, CS_VOID * dateval, CS_DATEREC * daterec)
     833             : {
     834           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_dt_crack(%p, %d, %p, %p)\n", ctx, datetype, dateval, daterec);
     835             : 
     836             :         /* Avoid handling CS_BIGDATETIME_TYPE and CS_BIGTIME_TYPE as these
     837             :          * types requires a larger structure that would cause a buffer overflow. */
     838           0 :         if (datetype != CS_BIGDATETIME_TYPE && datetype != CS_BIGTIME_TYPE)
     839           0 :                 return cs_dt_crack_v2(ctx, datetype, dateval, daterec);
     840             :         return CS_FAIL;
     841             : }
     842             : 
     843             : CS_RETCODE
     844          40 : cs_loc_alloc(CS_CONTEXT * ctx, CS_LOCALE ** loc_pointer)
     845             : {
     846             :         CS_LOCALE *tds_csloc;
     847             : 
     848          40 :         tdsdump_log(TDS_DBG_FUNC, "cs_loc_alloc(%p, %p)\n", ctx, loc_pointer);
     849             : 
     850          40 :         if (!ctx)
     851             :                 return CS_FAIL;
     852             : 
     853          30 :         if (!loc_pointer) {
     854          10 :                 _csclient_msg(ctx, "cs_loc_alloc", 2, 1, 1, 4, "loc_pointer");
     855          10 :                 return CS_FAIL;
     856             :         }
     857             : 
     858          20 :         tds_csloc = _cs_locale_alloc();
     859          20 :         if (!tds_csloc)
     860             :                 return CS_FAIL;
     861             : 
     862          20 :         *loc_pointer = tds_csloc;
     863          20 :         return CS_SUCCEED;
     864             : }
     865             : 
     866             : CS_RETCODE
     867          40 : cs_loc_drop(CS_CONTEXT * ctx, CS_LOCALE * locale)
     868             : {
     869          40 :         tdsdump_log(TDS_DBG_FUNC, "cs_loc_drop(%p, %p)\n", ctx, locale);
     870             : 
     871          40 :         if (!ctx)
     872             :                 return CS_FAIL;
     873             : 
     874          30 :         if (!locale) {
     875          10 :                 _csclient_msg(ctx, "cs_loc_drop", 2, 1, 1, 4, "locale");
     876          10 :                 return CS_FAIL;
     877             :         }
     878             : 
     879          20 :         _cs_locale_free(locale);
     880          20 :         return CS_SUCCEED;
     881             : }
     882             : 
     883             : CS_RETCODE
     884          60 : cs_locale(CS_CONTEXT * ctx, CS_INT action, CS_LOCALE * locale, CS_INT type, CS_VOID * buffer, CS_INT buflen, CS_INT * outlen)
     885             : {
     886             :         static const char func[] = "cs_locale";
     887          60 :         CS_RETCODE code = CS_FAIL;
     888             : 
     889          60 :         tdsdump_log(TDS_DBG_FUNC, "cs_locale(%p, %d, %p, %d, %p, %d, %p)\n", ctx, action, locale, type, buffer, buflen, outlen);
     890             : 
     891          60 :         if (!ctx)
     892             :                 return CS_FAIL;
     893             : 
     894          50 :         if (!locale) {
     895          10 :                 _csclient_msg(ctx, func, 2, 1, 1, 4, "locale");
     896          10 :                 return CS_FAIL;
     897             :         }
     898             : 
     899          40 :         if (action == CS_SET) {
     900          30 :                 switch (type) {
     901           0 :                 case CS_LC_ALL:
     902             :                         /* what to do here if there is locale data? */
     903           0 :                         if (!buffer) {
     904           0 :                                 code = CS_SUCCEED;
     905             :                         }
     906             :                         break;
     907             : 
     908          20 :                 case CS_SYB_CHARSET:
     909          20 :                         buflen = _ct_get_string_length(buffer, buflen);
     910          20 :                         if (buflen < 0) {
     911          10 :                                 _csclient_msg(ctx, func, 2, 1, 1, 6, "%d, buflen", buflen);
     912          10 :                                 return CS_FAIL;
     913             :                         }
     914             :                         
     915          10 :                         free(locale->charset);
     916          10 :                         locale->charset = tds_strndup(buffer, buflen);
     917          10 :                         if (!locale->charset)
     918             :                                 break;
     919             : 
     920          10 :                         code = CS_SUCCEED;
     921          10 :                         break;
     922             : 
     923           0 :                 case CS_SYB_LANG:
     924           0 :                         buflen = _ct_get_string_length(buffer, buflen);
     925           0 :                         if (buflen < 0) {
     926           0 :                                 _csclient_msg(ctx, func, 2, 1, 1, 6, "%d, buflen", buflen);
     927           0 :                                 return CS_FAIL;
     928             :                         }
     929             :                         
     930           0 :                         free(locale->language);
     931           0 :                         locale->language = tds_strndup(buffer, buflen);
     932           0 :                         if (!locale->language)
     933             :                                 break;
     934             : 
     935           0 :                         code = CS_SUCCEED;
     936           0 :                         break;
     937             : 
     938           0 :                 case CS_SYB_LANG_CHARSET:
     939             :                 {
     940             :                         int i;
     941           0 :                         char *b = (char *)buffer;
     942             : 
     943           0 :                         buflen = _ct_get_string_length(buffer, buflen);
     944           0 :                         if (buflen < 0) {
     945           0 :                                 _csclient_msg(ctx, func, 2, 1, 1, 6, "%d, buflen", buflen);
     946           0 :                                 return CS_FAIL;
     947             :                         }
     948             : 
     949             :                         /* find '.' character */
     950           0 :                         for (i = 0; i < buflen; ++i) {
     951           0 :                                 if (b[i] == '.') {
     952             :                                         break;
     953             :                                 }                                       
     954             :                         }
     955             :                         /* not found */
     956           0 :                         if (i == buflen) {
     957             :                                 break;
     958             :                         }
     959           0 :                         if (i) {
     960           0 :                                 free(locale->language);
     961           0 :                                 locale->language = tds_strndup(b, i);
     962           0 :                                 if (!locale->language)
     963             :                                         break;
     964             :                         }
     965           0 :                         if (i != (buflen - 1)) {
     966           0 :                                 free(locale->charset);
     967           0 :                                 locale->charset = tds_strndup(b + i + 1, buflen - i - 1);
     968           0 :                                 if (!locale->charset)
     969             :                                         break;
     970             :                         }
     971             :                         code = CS_SUCCEED;
     972             :                         break;
     973             :                 }
     974             : 
     975             :                 /* TODO commented out until the code works end-to-end
     976             :                 case CS_SYB_SORTORDER:
     977             :                         buflen = _ct_get_string_length(buffer, buflen);
     978             :                         if (buflen < 0) {
     979             :                                 _csclient_msg(ctx, func, 2, 1, 1, 6, "%d, buflen", buflen);
     980             :                                 return CS_FAIL;
     981             :                         }
     982             :                         
     983             :                         free(locale->collate);
     984             :                         locale->collate = tds_strndup(buffer, buflen);
     985             :                         if (!locale->collate)
     986             :                                 break;
     987             : 
     988             :                         code = CS_SUCCEED;
     989             :                         break;
     990             :                 */
     991             : 
     992          10 :                 default:
     993          10 :                         _csclient_msg(ctx, func, 2, 1, 1, 6, "%d, type", type);
     994          10 :                         code = CS_FAIL;
     995          10 :                         break;
     996             :                 }
     997           0 :         }
     998          10 :         else if (action == CS_GET)
     999             :         {
    1000             :                 CS_INT tlen;
    1001             : 
    1002           0 :                 switch (type) {
    1003           0 :                 case CS_SYB_CHARSET:
    1004           0 :                         tlen = (locale->charset ? (CS_INT) strlen(locale->charset) : 0) + 1;
    1005           0 :                         if (buflen < tlen)
    1006             :                         {
    1007           0 :                                 if (outlen)
    1008           0 :                                         *outlen = tlen;
    1009             :                                 break;
    1010             :                         }
    1011           0 :                         if (locale->charset)
    1012           0 :                                 strcpy((char *)buffer, locale->charset);
    1013             :                         else
    1014           0 :                                 ((char *)buffer)[0] = '\0';
    1015             :                         code = CS_SUCCEED;
    1016             :                         break;
    1017             : 
    1018           0 :                 case CS_SYB_LANG:
    1019           0 :                         tlen = (locale->language ? (CS_INT) strlen(locale->language) : 0) + 1;
    1020           0 :                         if (buflen < tlen)
    1021             :                         {
    1022           0 :                                 if (outlen)
    1023           0 :                                         *outlen = tlen;
    1024             :                                 break;
    1025             :                         }
    1026           0 :                         if (locale->language)
    1027           0 :                                 strcpy((char *)buffer, locale->language);
    1028             :                         else
    1029           0 :                                 ((char *)buffer)[0] = '\0';
    1030             :                         code = CS_SUCCEED;
    1031             :                         break;
    1032             : 
    1033           0 :                 case CS_SYB_LANG_CHARSET:
    1034             :                 {
    1035             :                         CS_INT clen;
    1036             : 
    1037           0 :                         tlen = (locale->language ? (CS_INT) strlen(locale->language) : 0) + 1;
    1038           0 :                         clen = (locale->charset ? (CS_INT) strlen(locale->charset) : 0) + 1;
    1039             :                         
    1040           0 :                         if (buflen < (tlen + clen))
    1041             :                         {
    1042           0 :                                 if (outlen)
    1043           0 :                                         *outlen = tlen + clen;
    1044             :                                 break;
    1045             :                         }
    1046           0 :                         if (locale->language)
    1047           0 :                                 strcpy((char *)buffer, locale->language);
    1048             :                         else
    1049           0 :                                 ((char *)buffer)[0] = '\0';
    1050           0 :                         strcat((char *)buffer, ".");
    1051           0 :                         if (locale->charset) {
    1052           0 :                                 tlen = (CS_INT) strlen((char *)buffer);
    1053           0 :                                 strcpy((char *)buffer + tlen, locale->charset);
    1054             :                         }
    1055             :                         code = CS_SUCCEED;
    1056             :                         break;
    1057             :                 }
    1058             : 
    1059           0 :                 case CS_SYB_SORTORDER:
    1060           0 :                         tlen = (locale->collate ? (CS_INT) strlen(locale->collate) : 0) + 1;
    1061           0 :                         if (buflen < tlen)
    1062             :                         {
    1063           0 :                                 if (outlen)
    1064           0 :                                         *outlen = tlen;
    1065             :                                 break;
    1066             :                         }
    1067           0 :                         if (locale->collate)
    1068           0 :                                 strcpy((char *)buffer, locale->collate);
    1069             :                         else
    1070           0 :                                 ((char *)buffer)[0] = '\0';
    1071             :                         code = CS_SUCCEED;
    1072             :                         break;
    1073             : 
    1074           0 :                 default:
    1075           0 :                         _csclient_msg(ctx, func, 2, 1, 1, 6, "%d, type", type);
    1076           0 :                         code = CS_FAIL;
    1077           0 :                         break;
    1078             :                 }
    1079             :         } else {
    1080          10 :                 _csclient_msg(ctx, func, 2, 1, 1, 6, "%d, action", action);
    1081          10 :                 code = CS_FAIL;
    1082             :         }
    1083             :         return code;
    1084             : }
    1085             : 
    1086             : typedef struct
    1087             : {
    1088             :         CS_INT convfmt;
    1089             :         const char datetime[26];
    1090             :         uint8_t date_len;
    1091             :         uint8_t time_start;
    1092             : } CONV_FORMAT;
    1093             : 
    1094             : #define ONLY_DATE(convfmt, datetime) \
    1095             :         { convfmt, datetime, sizeof(datetime)-1, sizeof(datetime)-1 }
    1096             : #define DATE_TIME(convfmt, date, time) \
    1097             :         { convfmt, date " " time, sizeof(date)-1, sizeof(date) }
    1098             : #define ALL__SAME(convfmt, datetime) \
    1099             :         { convfmt, datetime, sizeof(datetime)-1, 0 }
    1100             : static const CONV_FORMAT formats[] = {
    1101             :         DATE_TIME(CS_DATES_SHORT, "%b %e %Y", "%l:%M%p"),
    1102             :         ONLY_DATE(CS_DATES_MDY1, "%m/%d/%y"),
    1103             :         ONLY_DATE(CS_DATES_YMD1, "%y.%m.%d"),
    1104             :         ONLY_DATE(CS_DATES_DMY1, "%d/%m/%y"),
    1105             :         ONLY_DATE(CS_DATES_DMY2, "%d.%m.%y"),
    1106             :         ONLY_DATE(CS_DATES_DMY3, "%d-%m-%y"),
    1107             :         ONLY_DATE(CS_DATES_DMY4, "%d %b %y"),
    1108             :         ONLY_DATE(CS_DATES_MDY2, "%b %d, %y"),
    1109             :         ALL__SAME(CS_DATES_HMS, "%H:%M:%S"),
    1110             :         DATE_TIME(CS_DATES_LONG, "%b %e %Y", "%l:%M:%S:%z%p"),
    1111             :         ONLY_DATE(CS_DATES_MDY3, "%m-%d-%y"),
    1112             :         ONLY_DATE(CS_DATES_YMD2, "%y/%m/%d"),
    1113             :         ONLY_DATE(CS_DATES_YMD3, "%y%m%d"),
    1114             :         ONLY_DATE(CS_DATES_YDM1, "%y/%d/%m"),
    1115             :         ONLY_DATE(CS_DATES_MYD1, "%m/%y/%d"),
    1116             :         ONLY_DATE(CS_DATES_DYM1, "%d/%y/%m"),
    1117             :         ALL__SAME(CS_DATES_MDYHMS, "%b %e %Y %H:%M:%S"),
    1118             :         ALL__SAME(CS_DATES_HMA, "%l:%M%p"),
    1119             :         ALL__SAME(CS_DATES_HM, "%H:%M"),
    1120             :         ALL__SAME(CS_DATES_HMSZA, "%l:%M:%S:%3z%p"),
    1121             :         ALL__SAME(CS_DATES_HMSZ, "%H:%M:%S:%3z"),
    1122             :         ALL__SAME(CS_DATES_YMDHMS, "%y/%m/%d %H:%M:%S"),
    1123             :         ALL__SAME(CS_DATES_YMDHMA, "%y/%m/%d %l:%M%p"),
    1124             :         ALL__SAME(CS_DATES_YMDTHMS, "%Y-%m-%dT%H:%M:%S"),
    1125             :         ALL__SAME(CS_DATES_HMSUSA, "%l:%M:%S.%6z%p"),
    1126             :         ALL__SAME(CS_DATES_HMSUS, "%H:%M:%S.%6z"),
    1127             :         ALL__SAME(CS_DATES_LONGUSA, "%b %e %y %l:%M:%S.%6z%p"),
    1128             :         ALL__SAME(CS_DATES_LONGUS, "%b %e %y %H:%M:%S.%6z"),
    1129             :         ALL__SAME(CS_DATES_YMDHMSUS, "%y-%m-%d %H:%M:%S.%6z"),
    1130             :         ALL__SAME(CS_DATES_SHORT_ALT, "%b %e %Y %l:%M%p"),
    1131             :         ONLY_DATE(CS_DATES_MDY1_YYYY, "%m/%d/%Y"),
    1132             :         ONLY_DATE(CS_DATES_YMD1_YYYY, "%Y.%m.%d"),
    1133             :         ONLY_DATE(CS_DATES_DMY1_YYYY, "%d/%m/%Y"),
    1134             :         ONLY_DATE(CS_DATES_DMY2_YYYY, "%d.%m.%Y"),
    1135             :         ONLY_DATE(CS_DATES_DMY3_YYYY, "%d-%m-%Y"),
    1136             :         ONLY_DATE(CS_DATES_DMY4_YYYY, "%d %b %Y"),
    1137             :         ONLY_DATE(CS_DATES_MDY2_YYYY, "%b %d, %Y"),
    1138             :         ALL__SAME(CS_DATES_HMS_ALT, "%H:%M:%S"),
    1139             :         ALL__SAME(CS_DATES_LONG_ALT, "%b %e %Y %l:%M:%S:%3z%p"),
    1140             :         ONLY_DATE(CS_DATES_MDY3_YYYY, "%m-%d-%Y"),
    1141             :         ONLY_DATE(CS_DATES_YMD2_YYYY, "%Y/%m/%d"),
    1142             :         ONLY_DATE(CS_DATES_YMD3_YYYY, "%Y%m%d"),
    1143             :         ONLY_DATE(CS_DATES_YDM1_YYYY, "%Y/%d/%m"),
    1144             :         ONLY_DATE(CS_DATES_MYD1_YYYY, "%m/%Y/%d"),
    1145             :         ONLY_DATE(CS_DATES_DYM1_YYYY, "%d/%Y/%m"),
    1146             :         ALL__SAME(CS_DATES_MDYHMS_ALT, "%b %e %Y %H:%M:%S"),
    1147             :         ALL__SAME(CS_DATES_HMA_ALT, "%l:%M%p"),
    1148             :         ALL__SAME(CS_DATES_HM_ALT, "%H:%M"),
    1149             :         ALL__SAME(CS_DATES_YMDHMS_YYYY, "%Y/%m/%d %H:%M:%S"),
    1150             :         ALL__SAME(CS_DATES_YMDHMA_YYYY, "%Y/%m/%d %l:%M%p"),
    1151             :         ALL__SAME(CS_DATES_HMSUSA_YYYY, "%l:%M:%S.%6z%p"),
    1152             :         ALL__SAME(CS_DATES_HMSUS_YYYY, "%H:%M:%S.%6z"),
    1153             :         ALL__SAME(CS_DATES_LONGUSA_YYYY, "%b %e %Y %l:%M:%S.%6z%p"),
    1154             :         ALL__SAME(CS_DATES_LONGUS_YYYY, "%b %e %Y %H:%M:%S.%6z"),
    1155             :         ALL__SAME(CS_DATES_YMDHMSUS_YYYY, "%Y-%m-%d %H:%M:%S.%6z"),
    1156             :         {-1, "", 0, 0}
    1157             : };
    1158             : 
    1159             : static CS_RETCODE
    1160        1880 : _cs_set_locale_convfmt(CS_CONTEXT *ctx, TDSLOCALE *loc, CS_INT convfmt)
    1161             : {
    1162             :         const CONV_FORMAT *fmt;
    1163             : 
    1164       33460 :         for (fmt = formats; fmt->convfmt >= 0; ++fmt) {
    1165             :                 char *datetime, *date, *time;
    1166             : 
    1167       16730 :                 if (fmt->convfmt != convfmt)
    1168       14850 :                         continue;
    1169        1880 :                 datetime = strdup(fmt->datetime);
    1170        1880 :                 date = tds_strndup(fmt->datetime, fmt->date_len);
    1171        1880 :                 time = strdup(fmt->datetime + fmt->time_start);
    1172        1880 :                 if (!datetime || !date || !time) {
    1173           0 :                         free(datetime);
    1174           0 :                         free(date);
    1175           0 :                         free(time);
    1176           0 :                         _csclient_msg(ctx, "cs_dt_info", 2, 4, 1, 3, "");
    1177           0 :                         return CS_FAIL;
    1178             :                 }
    1179        1880 :                 free(loc->datetime_fmt);
    1180        1880 :                 loc->datetime_fmt = datetime;
    1181        1880 :                 free(loc->date_fmt);
    1182        1880 :                 loc->date_fmt = date;
    1183        1880 :                 free(loc->time_fmt);
    1184        1880 :                 loc->time_fmt = time;
    1185        1880 :                 break;
    1186             :         }
    1187             :         return CS_SUCCEED;
    1188             : }
    1189             : 
    1190             : CS_RETCODE
    1191        1990 : cs_dt_info(CS_CONTEXT *ctx, CS_INT action, CS_LOCALE *locale, CS_INT type, CS_INT item, CS_VOID *buffer, CS_INT buflen,
    1192             :            CS_INT *outlen)
    1193             : {
    1194             :         static const char func[] = "cs_dt_info";
    1195             : 
    1196        1990 :         tdsdump_log(TDS_DBG_FUNC, "cs_dt_info(%p, %d, %p, %d, %d, %p, %d, %p)\n",
    1197             :                     ctx, action, locale, type, item, buffer, buflen, outlen);
    1198             : 
    1199        1990 :         if (!ctx)
    1200             :                 return CS_FAIL;
    1201             : 
    1202        1980 :         if (action == CS_SET) {
    1203        1960 :                 switch (type) {
    1204        1950 :                 case CS_DT_CONVFMT:
    1205        1950 :                         if (buflen < 0) {
    1206          20 :                                 _csclient_msg(ctx, func, 2, 1, 1, 6, "%d, buflen", buflen);
    1207          20 :                                 return CS_FAIL;
    1208             :                         }
    1209        1930 :                         if (!locale && buffer) {
    1210        1880 :                                 CS_INT convfmt = *(CS_INT *) buffer;
    1211        1880 :                                 TDSLOCALE *loc = ctx->tds_ctx->locale;
    1212             : 
    1213        1880 :                                 return _cs_set_locale_convfmt(ctx, loc, convfmt);
    1214             :                         }
    1215             :                         break;
    1216          10 :                 default:
    1217          10 :                         _csclient_msg(ctx, func, 2, 1, 1, 6, "%d, type", type);
    1218          10 :                         return CS_FAIL;
    1219             :                         break;
    1220             :                 }
    1221          20 :         } else if (action == CS_GET) {
    1222             :                 switch (type) {
    1223          10 :                 default:
    1224          10 :                         _csclient_msg(ctx, func, 2, 1, 1, 6, "%d, type", type);
    1225          10 :                         return CS_FAIL;
    1226             :                         break;
    1227             :                 }
    1228             :         } else {
    1229          10 :                 _csclient_msg(ctx, func, 2, 1, 1, 6, "%d, action", action);
    1230          10 :                 return CS_FAIL;
    1231             :         }
    1232             :         return CS_SUCCEED;
    1233             : }
    1234             : 
    1235             : CS_RETCODE
    1236           0 : cs_strbuild(CS_CONTEXT * ctx, CS_CHAR * buffer, CS_INT buflen, CS_INT * resultlen, CS_CHAR * text, CS_INT textlen,
    1237             :             CS_CHAR * formats, CS_INT formatlen, ...)
    1238             : {
    1239             :         va_list ap;
    1240             :         TDSRET rc;
    1241             : 
    1242           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_strbuild(%p, %p, %d, %p, %p, %d, %p, %d)\n", 
    1243             :                                 ctx, buffer, buflen, resultlen, text, textlen, formats, formatlen);
    1244             : 
    1245           0 :         va_start(ap, formatlen);
    1246           0 :         rc = tds_vstrbuild(buffer, buflen, resultlen, text, textlen, formats, formatlen, ap);
    1247           0 :         va_end(ap);
    1248             : 
    1249           0 :         return TDS_SUCCEED(rc) ? CS_SUCCEED : CS_FAIL;
    1250             : }
    1251             : 
    1252             : CS_RETCODE
    1253           0 : cs_calc(CS_CONTEXT * ctx, CS_INT op, CS_INT datatype, CS_VOID * var1, CS_VOID * var2, CS_VOID * dest)
    1254             : {
    1255             : 
    1256           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_calc(%p, %d, %d, %p, %p, %p)\n", ctx, op, datatype, var1, var2, dest);
    1257             : 
    1258           0 :         tdsdump_log(TDS_DBG_FUNC, "UNIMPLEMENTED cs_calc()\n");
    1259           0 :         return CS_FAIL;
    1260             : }
    1261             : 
    1262             : CS_RETCODE
    1263           0 : cs_cmp(CS_CONTEXT * ctx, CS_INT datatype, CS_VOID * var1, CS_VOID * var2, CS_INT * result)
    1264             : {
    1265             : 
    1266           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_cmp(%p, %d, %p, %p, %p)\n", ctx, datatype, var1, var2, result);
    1267             : 
    1268           0 :         tdsdump_log(TDS_DBG_FUNC, "UNIMPLEMENTED cs_cmp()\n");
    1269           0 :         return CS_FAIL;
    1270             : }
    1271             : 
    1272             : CS_RETCODE
    1273           0 : cs_conv_mult(CS_CONTEXT * ctx, CS_LOCALE * srcloc, CS_LOCALE * destloc, CS_INT * conv_multiplier)
    1274             : {
    1275             : 
    1276           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_conv_mult(%p, %p, %p, %p)\n", ctx, srcloc, destloc, conv_multiplier);
    1277             : 
    1278           0 :         tdsdump_log(TDS_DBG_FUNC, "UNIMPLEMENTED cs_conv_mult()\n");
    1279           0 :         return CS_FAIL;
    1280             : }
    1281             : 
    1282             : CS_RETCODE
    1283          70 : cs_diag(CS_CONTEXT * ctx, CS_INT operation, CS_INT type, CS_INT idx, CS_VOID * buffer)
    1284             : {
    1285          70 :         tdsdump_log(TDS_DBG_FUNC, "cs_diag(%p, %d, %d, %d, %p)\n", ctx, operation, type, idx, buffer);
    1286             : 
    1287          70 :         switch (operation) {
    1288             : 
    1289          10 :                 case CS_INIT:
    1290          10 :                         if ( ctx->cs_errhandletype == _CS_ERRHAND_CB) {
    1291             :                                 /* contrary to the manual page you don't seem to */
    1292             :                                 /* be able to turn on inline message handling    */
    1293             :                                 /* using cs_diag, once a callback is installed!  */
    1294             :                                 return CS_FAIL;
    1295             :                         }
    1296          10 :                         ctx->cs_errhandletype = _CS_ERRHAND_INLINE;
    1297          10 :                         ctx->cs_diag_msglimit = CS_NO_LIMIT;
    1298          10 :                         ctx->cslibmsg_cb = (CS_CSLIBMSG_FUNC) cs_diag_storemsg;
    1299          10 :                         break;
    1300           0 :                 case CS_MSGLIMIT:
    1301           0 :                         if ( ctx->cs_errhandletype != _CS_ERRHAND_INLINE) {
    1302             :                                 return CS_FAIL;
    1303             :                         }
    1304           0 :                         ctx->cs_diag_msglimit = *(CS_INT *)buffer;
    1305           0 :                         break;
    1306          10 :                 case CS_CLEAR:
    1307          10 :                         if ( ctx->cs_errhandletype != _CS_ERRHAND_INLINE) {
    1308             :                                 return CS_FAIL;
    1309             :                         }
    1310          10 :                         return (cs_diag_clearmsg(ctx, type));
    1311             : 
    1312             :                         break;
    1313          30 :                 case CS_GET:
    1314          30 :                         if ( ctx->cs_errhandletype != _CS_ERRHAND_INLINE) {
    1315             :                                 return CS_FAIL;
    1316             :                         }
    1317          30 :                         if (buffer == NULL)
    1318             :                                 return CS_FAIL;
    1319             : 
    1320          30 :                         if (idx == 0 || (ctx->cs_diag_msglimit != CS_NO_LIMIT && idx > ctx->cs_diag_msglimit) )
    1321             :                                 return CS_FAIL;
    1322             : 
    1323          30 :                         return (cs_diag_getmsg(ctx, idx, (CS_CLIENTMSG *)buffer)); 
    1324             :                         
    1325             :                         break;
    1326          20 :                 case CS_STATUS:
    1327          20 :                         if ( ctx->cs_errhandletype != _CS_ERRHAND_INLINE) {
    1328             :                                 return CS_FAIL;
    1329             :                         }
    1330          20 :                         if (buffer == NULL) 
    1331             :                                 return CS_FAIL;
    1332             : 
    1333          20 :                         return (cs_diag_countmsg(ctx, (CS_INT *)buffer));
    1334             :                         break;
    1335             :         }
    1336             :         return CS_SUCCEED;
    1337             :                 
    1338             : }
    1339             : 
    1340             : CS_RETCODE
    1341           0 : cs_manage_convert(CS_CONTEXT * ctx, CS_INT action, CS_INT srctype, CS_CHAR * srcname, CS_INT srcnamelen, CS_INT desttype,
    1342             :                   CS_CHAR * destname, CS_INT destnamelen, CS_INT * conv_multiplier, CS_CONV_FUNC * func)
    1343             : {
    1344             : 
    1345           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_manage_convert(%p, %d, %d, %p, %d, %d, %p, %d, %p, %p)\n", 
    1346             :                                 ctx, action, srctype, srcname, srcnamelen, desttype, destname, destnamelen, conv_multiplier, func);
    1347             : 
    1348           0 :         tdsdump_log(TDS_DBG_FUNC, "UNIMPLEMENTED cs_manage_convert()\n");
    1349           0 :         return CS_FAIL;
    1350             : }
    1351             : 
    1352             : CS_RETCODE
    1353           0 : cs_objects(CS_CONTEXT * ctx, CS_INT action, CS_OBJNAME * objname, CS_OBJDATA * objdata)
    1354             : {
    1355             : 
    1356           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_objects(%p, %d, %p, %p)\n", ctx, action, objname, objdata);
    1357             : 
    1358           0 :         tdsdump_log(TDS_DBG_FUNC, "UNIMPLEMENTED cs_objects()\n");
    1359           0 :         return CS_FAIL;
    1360             : }
    1361             : 
    1362             : CS_RETCODE
    1363           0 : cs_set_convert(CS_CONTEXT * ctx, CS_INT action, CS_INT srctype, CS_INT desttype, CS_CONV_FUNC * func)
    1364             : {
    1365             : 
    1366           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_set_convert(%p, %d, %d, %d, %p)\n", ctx, action, srctype, desttype, func);
    1367             : 
    1368           0 :         tdsdump_log(TDS_DBG_FUNC, "UNIMPLEMENTED cs_set_convert()\n");
    1369           0 :         return CS_FAIL;
    1370             : }
    1371             : 
    1372             : CS_RETCODE
    1373           0 : cs_setnull(CS_CONTEXT * ctx, CS_DATAFMT * datafmt, CS_VOID * buffer, CS_INT buflen)
    1374             : {
    1375             : 
    1376           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_setnull(%p, %p, %p, %d)\n", ctx, datafmt, buffer, buflen);
    1377             : 
    1378           0 :         tdsdump_log(TDS_DBG_FUNC, "UNIMPLEMENTED cs_setnull()\n");
    1379           0 :         return CS_FAIL;
    1380             : }
    1381             : 
    1382             : CS_RETCODE
    1383           0 : cs_strcmp(CS_CONTEXT * ctx, CS_LOCALE * locale, CS_INT type, CS_CHAR * str1, CS_INT len1, CS_CHAR * str2, CS_INT len2,
    1384             :           CS_INT * result)
    1385             : {
    1386             : 
    1387           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_strcmp(%p, %p, %d, %p, %d, %p, %d, %p)\n", 
    1388             :                                         ctx, locale, type, str1, len1, str2, len2, result);
    1389             : 
    1390           0 :         tdsdump_log(TDS_DBG_FUNC, "UNIMPLEMENTED cs_strcmp()\n");
    1391           0 :         return CS_FAIL;
    1392             : }
    1393             : 
    1394             : CS_RETCODE
    1395           0 : cs_time(CS_CONTEXT * ctx, CS_LOCALE * locale, CS_VOID * buffer, CS_INT buflen, CS_INT * outlen, CS_DATEREC * daterec)
    1396             : {
    1397             : 
    1398           0 :         tdsdump_log(TDS_DBG_FUNC, "cs_time(%p, %p, %p, %d, %p, %p)\n", ctx, locale, buffer, buflen, outlen, daterec);
    1399             : 
    1400           0 :         tdsdump_log(TDS_DBG_FUNC, "UNIMPLEMENTED cs_time()\n");
    1401           0 :         return CS_FAIL;
    1402             : }
    1403             : 
    1404             : CS_RETCODE
    1405      935300 : cs_will_convert(CS_CONTEXT * ctx, CS_INT srctype, CS_INT desttype, CS_BOOL * result)
    1406             : {
    1407             : 
    1408      935300 :         tdsdump_log(TDS_DBG_FUNC, "cs_will_convert(%p, %d, %d, %p)\n", ctx, srctype, desttype, result);
    1409             : 
    1410      935300 :         srctype = _ct_get_server_type(NULL, srctype);
    1411      935300 :         desttype = _ct_get_server_type(NULL, desttype);
    1412      935300 :         *result = (tds_willconvert(srctype, desttype) ? CS_TRUE : CS_FALSE);
    1413      935300 :         return CS_SUCCEED;
    1414             : }
    1415             : 
    1416             : static CS_INT 
    1417          20 : cs_diag_storemsg(CS_CONTEXT *context, CS_CLIENTMSG *message)
    1418             : {
    1419             :         struct cs_diag_msg **curptr;
    1420          20 :         CS_INT msg_count = 0;
    1421             : 
    1422          20 :         tdsdump_log(TDS_DBG_FUNC, "cs_diag_storemsg(%p, %p)\n", context, message);
    1423             : 
    1424          20 :         curptr = &(context->msgstore);
    1425             : 
    1426             :         /* if we already have a list of messages, */
    1427             :         /* go to the end of the list...           */
    1428             : 
    1429          50 :         while (*curptr != NULL) {
    1430          10 :                 msg_count++;
    1431          10 :                 curptr = &((*curptr)->next);
    1432             :         }
    1433             : 
    1434             :         /* messages over and above the agreed limit */
    1435             :         /* are simply discarded...                  */
    1436             : 
    1437          20 :         if (context->cs_diag_msglimit != CS_NO_LIMIT &&
    1438             :                 msg_count >= context->cs_diag_msglimit) {
    1439             :                 return CS_FAIL;
    1440             :         }
    1441             : 
    1442          20 :         *curptr = tds_new(struct cs_diag_msg, 1);
    1443          20 :         if (*curptr == NULL) { 
    1444             :                 return CS_FAIL;
    1445             :         } else {
    1446          20 :                 (*curptr)->next = NULL;
    1447          20 :                 (*curptr)->msg  = tds_new(CS_CLIENTMSG, 1);
    1448          20 :                 if ((*curptr)->msg == NULL) {
    1449             :                         return CS_FAIL;
    1450             :                 } else {
    1451          20 :                         memcpy((*curptr)->msg, message, sizeof(CS_CLIENTMSG));
    1452             :                 }
    1453             :         }
    1454             : 
    1455          20 :         return CS_SUCCEED;
    1456             : }
    1457             : 
    1458             : static CS_INT 
    1459          30 : cs_diag_getmsg(CS_CONTEXT *context, CS_INT idx, CS_CLIENTMSG *message)
    1460             : {
    1461             :         struct cs_diag_msg *curptr;
    1462          30 :         CS_INT msg_count = 0, msg_found = 0;
    1463             : 
    1464          30 :         tdsdump_log(TDS_DBG_FUNC, "cs_diag_getmsg(%p, %d, %p)\n", context, idx, message);
    1465             : 
    1466          30 :         curptr = context->msgstore;
    1467             : 
    1468             :         /* if we already have a list of messages, */
    1469             :         /* go to the end of the list...           */
    1470             : 
    1471          90 :         while (curptr != NULL) {
    1472          50 :                 msg_count++;
    1473          50 :                 if (msg_count == idx) {
    1474             :                         msg_found++;
    1475             :                         break;
    1476             :                 }
    1477          30 :                 curptr = curptr->next;
    1478             :         }
    1479          30 :         if (msg_found) {
    1480          20 :                 memcpy(message, curptr->msg, sizeof(CS_CLIENTMSG));
    1481          20 :                 return CS_SUCCEED;
    1482             :         } else {
    1483             :                 return CS_NOMSG;
    1484             :         }
    1485             : }
    1486             : 
    1487             : static CS_INT 
    1488          10 : cs_diag_clearmsg(CS_CONTEXT *context, CS_INT type)
    1489             : {
    1490             :         struct cs_diag_msg *curptr, *freeptr;
    1491             : 
    1492          10 :         tdsdump_log(TDS_DBG_FUNC, "cs_diag_clearmsg(%p, %d)\n", context, type);
    1493             : 
    1494          10 :         curptr = context->msgstore;
    1495          10 :         context->msgstore = NULL;
    1496             : 
    1497          40 :         while (curptr != NULL ) {
    1498          20 :                 freeptr = curptr;
    1499          20 :                 curptr = freeptr->next;
    1500          20 :                 free(freeptr->msg);
    1501          20 :                 free(freeptr);
    1502             :         }
    1503          10 :         return CS_SUCCEED;
    1504             : }
    1505             : 
    1506             : static CS_INT 
    1507          20 : cs_diag_countmsg(CS_CONTEXT *context, CS_INT *count)
    1508             : {
    1509             :         struct cs_diag_msg *curptr;
    1510          20 :         CS_INT msg_count = 0;
    1511             : 
    1512          20 :         tdsdump_log(TDS_DBG_FUNC, "cs_diag_countmsg(%p, %p)\n", context, count);
    1513             : 
    1514          20 :         curptr = context->msgstore;
    1515             : 
    1516          60 :         while (curptr != NULL) {
    1517          20 :                 msg_count++;
    1518          20 :                 curptr = curptr->next;
    1519             :         }
    1520          20 :         *count = msg_count;
    1521          20 :         return CS_SUCCEED;
    1522             : }
    1523             : 
    1524             : /**
    1525             :  * Try to convert to a type we can handle.
    1526             :  * Used before calling _cs_convert to handle TDS types that do not have a
    1527             :  * corresponding CS_xxx_TYPE.
    1528             :  * @param ctx             Library context, used for conversion, can be NULL.
    1529             :  * @param curcol          Column with data to convert.
    1530             :  * @param convert_buffer  Buffer to store conversion, can be NULL.
    1531             :  * @param p_src           Pointer to pointer to source data, can be NULL.
    1532             :  * @return Converted type or CS_ILLEGAL_TYPE.
    1533             :  */
    1534             : int
    1535        4835 : _cs_convert_not_client(CS_CONTEXT *ctx, const TDSCOLUMN *curcol, CONV_RESULT *convert_buffer, unsigned char **p_src)
    1536             : {
    1537             :         int ct_type;
    1538        4835 :         TDS_SERVER_TYPE desttype, srctype = curcol->column_type;
    1539             : 
    1540        4835 :         if (srctype == SYBVARIANT)
    1541         554 :                 srctype = ((const TDSVARIANT *) curcol->column_data)->type;
    1542             : 
    1543        4835 :         switch (srctype) {
    1544             :         case SYBMSDATE:
    1545             :                 desttype = SYBDATE;
    1546             :                 ct_type = CS_DATE_TYPE;
    1547             :                 break;
    1548          40 :         case SYBMSTIME:
    1549          40 :                 desttype = SYB5BIGTIME;
    1550          40 :                 ct_type = CS_BIGTIME_TYPE;
    1551          40 :                 break;
    1552          80 :         case SYBMSDATETIME2:
    1553             :         case SYBMSDATETIMEOFFSET:
    1554          80 :                 desttype = SYB5BIGDATETIME;
    1555          80 :                 ct_type = CS_BIGDATETIME_TYPE;
    1556          80 :                 break;
    1557             :         default:
    1558             :                 return CS_ILLEGAL_TYPE;
    1559             :         }
    1560             : 
    1561         160 :         if (convert_buffer) {
    1562         160 :                 int len = tds_convert(ctx->tds_ctx, srctype, *p_src,
    1563          80 :                                       curcol->column_cur_size, desttype, convert_buffer);
    1564          80 :                 if (len < 0)
    1565             :                         return CS_ILLEGAL_TYPE; /* TODO _csclient_msg ?? */
    1566          80 :                 *p_src = (unsigned char *) convert_buffer;
    1567             :         }
    1568             :         return ct_type;
    1569             : }

Generated by: LCOV version 1.13