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 : }
|