⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 cs.c

📁 在Linux/Unix下面访问WINDOWS SQLSERVER 的ODBC驱动程序
💻 C
📖 第 1 页 / 共 2 页
字号:
/* FreeTDS - Library of routines accessing Sybase and Microsoft databases * Copyright (C) 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005 Brian Bruns * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */#if HAVE_CONFIG_H#include <config.h>#endif#include <stdarg.h>#if TIME_WITH_SYS_TIME# include <sys/time.h># include <time.h>#else# if HAVE_SYS_TIME_H#  include <sys/time.h># else#  include <time.h># endif#endif#include <stdio.h>#if HAVE_STDLIB_H#include <stdlib.h>#endif /* HAVE_STDLIB_H */#if HAVE_STRING_H#include <string.h>#endif /* HAVE_STRING_H */#include "cspublic.h"#include "ctlib.h"#include "tdsconvert.h"#include "replacements.h"TDS_RCSID(var, "$Id: cs.c,v 1.65 2007/12/23 21:12:02 jklowden Exp $");static int _cs_datatype_length(int dtype);static CS_INT cs_diag_storemsg(CS_CONTEXT *context, CS_CLIENTMSG *message);static CS_INT cs_diag_clearmsg(CS_CONTEXT *context, CS_INT type);static CS_INT cs_diag_getmsg(CS_CONTEXT *context, CS_INT idx, CS_CLIENTMSG *message);static CS_INT cs_diag_countmsg(CS_CONTEXT *context, CS_INT *count);/** * returns the fixed length of the specified data type, or 0 if not a  * fixed length data type */static int _cs_datatype_length(int dtype){	switch (dtype) {		case SYBINT1:			return 1;		case SYBINT2:			return 2;		case SYBINT4:			return 4;		case SYBINT8:			return 8;		case SYBFLT8:			return 8;		case SYBREAL:			return 4;		case SYBBIT:			return 1;		case SYBMONEY:			return 8;		case SYBMONEY4:			return 4;		case SYBDATETIME:			return 8;		case SYBDATETIME4:			return 4;		default:			return 0;	}}static const char *_cs_get_layer(int layer){	switch (layer) {	case 2:		return "cslib user api layer";		break;	default:		break;	}	return "unrecognized layer";}static const char *_cs_get_origin(int origin){	switch (origin) {	case 1:		return "external error";		break;	case 2:		return "internal CS-Library error";		break;	case 4:		return "common library error";		break;	case 5:		return "intl library error";		break;	default:		break;	}	return "unrecognized origin";}static const char *_cs_get_user_api_layer_error(int error){	switch (error) {	case 3:		return "Memory allocation failure.";		break;	case 16:		return "Conversion between %1! and %2! datatypes is not supported.";		break;	case 20:		return "The conversion/operation resulted in overflow.";		break;	case 24:		return "The conversion/operation was stopped due to a syntax error in the source field.";		break;	default:		break;	}	return "unrecognized error";}static char *_cs_get_msgstr(const char *funcname, int layer, int origin, int severity, int number){	char *m;	if (asprintf(&m, "%s: %s: %s: %s", funcname, _cs_get_layer(layer), _cs_get_origin(origin), (layer == 2)		     ? _cs_get_user_api_layer_error(number)		     : "unrecognized error") < 0) {		return NULL;	}	return m;}static void_csclient_msg(CS_CONTEXT * ctx, const char *funcname, int layer, int origin, int severity, int number, const char *fmt, ...){	va_list ap;	CS_CLIENTMSG cm;	char *msgstr;	va_start(ap, fmt);	if (ctx->_cslibmsg_cb) {		cm.severity = severity;		cm.msgnumber = (((layer << 24) & 0xFF000000)				| ((origin << 16) & 0x00FF0000)				| ((severity << 8) & 0x0000FF00)				| ((number) & 0x000000FF));		msgstr = _cs_get_msgstr(funcname, layer, origin, severity, number);		tds_vstrbuild(cm.msgstring, CS_MAX_MSG, &(cm.msgstringlen), msgstr, CS_NULLTERM, fmt, CS_NULLTERM, ap);		cm.msgstring[cm.msgstringlen] = '\0';		free(msgstr);		cm.osnumber = 0;		cm.osstring[0] = '\0';		cm.osstringlen = 0;		cm.status = 0;		/* cm.sqlstate */		cm.sqlstatelen = 0;		ctx->_cslibmsg_cb(ctx, &cm);	}	va_end(ap);}/** * Allocate new CS_LOCALE and initialize it * returns NULL on out of memory errors  */static CS_LOCALE *_cs_locale_alloc(void){	return (CS_LOCALE *) calloc(1, sizeof(CS_LOCALE));}static void_cs_locale_free_contents(CS_LOCALE *locale){	/* free strings */	free(locale->language);	locale->language = NULL;	free(locale->charset);	locale->charset = NULL;	free(locale->time);	locale->time = NULL;	free(locale->collate);	locale->collate = NULL;}void _cs_locale_free(CS_LOCALE *locale){	/* free contents */	_cs_locale_free_contents(locale);	/* free the data structure */	free(locale);}/* returns 0 on out of memory errors, 1 for OK */int_cs_locale_copy_inplace(CS_LOCALE *new_locale, CS_LOCALE *orig){	_cs_locale_free_contents(new_locale);	if (orig->language) {		new_locale->language = strdup(orig->language);		if (!new_locale->language)			goto Cleanup;	}	if (orig->charset) {		new_locale->charset = strdup(orig->charset);		if (!new_locale->charset)			goto Cleanup;	}	if (orig->time) {		new_locale->time = strdup(orig->time);		if (!new_locale->time)			goto Cleanup;	}	if (orig->collate) {		new_locale->collate = strdup(orig->collate);		if (!new_locale->collate)			goto Cleanup;	}	return 1;Cleanup:	_cs_locale_free_contents(new_locale);	return 0;}/* returns NULL on out of memory errors */CS_LOCALE *_cs_locale_copy(CS_LOCALE *orig){	CS_LOCALE *new_locale;	new_locale = _cs_locale_alloc();	if (!new_locale)		return NULL;	if (orig->language) {		new_locale->language = strdup(orig->language);		if (!new_locale->language)			goto Cleanup;	}	if (orig->charset) {		new_locale->charset = strdup(orig->charset);		if (!new_locale->charset)			goto Cleanup;	}	if (orig->time) {		new_locale->time = strdup(orig->time);		if (!new_locale->time)			goto Cleanup;	}	if (orig->collate) {		new_locale->collate = strdup(orig->collate);		if (!new_locale->collate)			goto Cleanup;	}	return new_locale;Cleanup:	_cs_locale_free(new_locale);	return NULL;}extern const char STD_DATETIME_FMT[];CS_RETCODEcs_ctx_alloc(CS_INT version, CS_CONTEXT ** ctx){TDSCONTEXT *tds_ctx;	*ctx = (CS_CONTEXT *) malloc(sizeof(CS_CONTEXT));	memset(*ctx, '\0', sizeof(CS_CONTEXT));	tds_ctx = tds_alloc_context(*ctx);	if (!tds_ctx) {		free(*ctx);		return CS_FAIL;	}	(*ctx)->tds_ctx = tds_ctx;	if (tds_ctx->locale && !tds_ctx->locale->date_fmt) {		/* set default in case there's no locale file */		tds_ctx->locale->date_fmt = strdup(STD_DATETIME_FMT);	}	return CS_SUCCEED;}CS_RETCODEcs_ctx_global(CS_INT version, CS_CONTEXT ** ctx){static CS_CONTEXT *global_cs_ctx = NULL;	if (global_cs_ctx != NULL) {		*ctx = global_cs_ctx;		return CS_SUCCEED;	}	if (cs_ctx_alloc(version, ctx) != CS_SUCCEED) {		return CS_FAIL;	}	global_cs_ctx = *ctx;	return CS_SUCCEED;}CS_RETCODEcs_ctx_drop(CS_CONTEXT * ctx){	if (ctx) {		_ct_diag_clearmsg(ctx, CS_ALLMSG_TYPE);		free(ctx->userdata);		if (ctx->tds_ctx)			tds_free_context(ctx->tds_ctx);		free(ctx);	}	return CS_SUCCEED;}CS_RETCODEcs_config(CS_CONTEXT * ctx, CS_INT action, CS_INT property, CS_VOID * buffer, CS_INT buflen, CS_INT * outlen){/* declared  for - CS_USERDATA changes - swapna*/CS_INT maxcp;	if (action == CS_GET) {		if (buffer == NULL) {			return CS_SUCCEED;		}		switch (property) {		case CS_MESSAGE_CB:			*(void **) buffer = (void*) ctx->_cslibmsg_cb;			return CS_SUCCEED;		case CS_USERDATA:			maxcp = ctx->userdata_len;			if (outlen)				*outlen = maxcp;			if (maxcp > buflen)				maxcp = buflen;			memcpy(buffer, ctx->userdata, maxcp); 						return CS_SUCCEED;		case CS_EXTRA_INF:		case CS_LOC_PROP:		case CS_VERSION:			return CS_FAIL;			break;		}	}	if (action == CS_SET) {		switch (property) {		case CS_MESSAGE_CB:			if ( ctx->cs_errhandletype == _CS_ERRHAND_INLINE) {				cs_diag_clearmsg(ctx, CS_UNUSED);			}			ctx->_cslibmsg_cb = (CS_CSLIBMSG_FUNC) buffer;			ctx->cs_errhandletype = _CS_ERRHAND_CB;			return CS_SUCCEED;		case CS_USERDATA:			free(ctx->userdata);				if (buflen == CS_NULLTERM) {				maxcp = strlen(buffer) + 1;			} else {				maxcp = buflen;			}			ctx->userdata = (void *) malloc(maxcp);			if ( ctx->userdata == NULL) {				return CS_FAIL;			}			ctx->userdata_len = maxcp;				if (buffer) {				memcpy(ctx->userdata, buffer, maxcp);			} else {				return CS_FAIL;			}			return CS_SUCCEED;			case CS_EXTRA_INF:		case CS_LOC_PROP:		case CS_VERSION:			return CS_FAIL;			break;		}	}	if (action == CS_CLEAR) {		switch (property) {		case CS_MESSAGE_CB:			if ( ctx->cs_errhandletype == _CS_ERRHAND_INLINE) {				cs_diag_clearmsg(ctx, CS_UNUSED);			}			ctx->_cslibmsg_cb = NULL;			ctx->cs_errhandletype = 0;			return CS_SUCCEED;		case CS_USERDATA:			free(ctx->userdata);			ctx->userdata = NULL;				return CS_SUCCEED;			case CS_EXTRA_INF:		case CS_LOC_PROP:		case CS_VERSION:			return CS_FAIL;			break;		}	}	return CS_FAIL;}CS_RETCODEcs_convert(CS_CONTEXT * ctx, CS_DATAFMT * srcfmt, CS_VOID * srcdata, CS_DATAFMT * destfmt, CS_VOID * destdata, CS_INT * resultlen){int src_type, src_len, desttype, destlen, len, i = 0;CONV_RESULT cres;unsigned char *dest;CS_RETCODE ret;	tdsdump_log(TDS_DBG_FUNC, "cs_convert()\n");	src_type = _ct_get_server_type(srcfmt->datatype);	src_len = srcfmt ? srcfmt->maxlength : 0;	desttype = _ct_get_server_type(destfmt->datatype);	destlen = destfmt ? destfmt->maxlength : 0;	tdsdump_log(TDS_DBG_FUNC, "cs_convert() srctype = %d (%d) desttype = %d (%d)\n",		    src_type, src_len, desttype, destlen);	if (!is_fixed_type(desttype) && (destlen <= 0)) {		return CS_FAIL;	}	dest = (unsigned char *) destdata;	/* If source is indicated to be NULL, set dest to low values */	if (srcdata == NULL) {		tdsdump_log(TDS_DBG_FUNC, "cs_convert() srcdata is null\n");		memset(dest, '\0', destlen);		if (resultlen != NULL)			*resultlen = 0;		return CS_SUCCEED;	}	/* many times we are asked to convert a data type to itself */	if (src_type == desttype) {		tdsdump_log(TDS_DBG_FUNC, "cs_convert() srctype = desttype\n");		switch (desttype) {		case SYBLONGBINARY:		case SYBBINARY:		case SYBVARBINARY:		case SYBIMAGE:			if (src_len > destlen) {				ret = CS_FAIL;			} else {				switch (destfmt->format) {				case CS_FMT_PADNULL:					memcpy(dest, srcdata, src_len);					for (i = src_len; i < destlen; i++)						dest[i] = '\0';					if (resultlen != NULL)						*resultlen = destlen;					ret = CS_SUCCEED;					break;				case CS_FMT_UNUSED:					memcpy(dest, srcdata, src_len);					if (resultlen != NULL)						*resultlen = src_len;					ret = CS_SUCCEED;					break;				default:					ret = CS_FAIL;					break;				}			}			break;		case SYBCHAR:		case SYBVARCHAR:		case SYBTEXT:			tdsdump_log(TDS_DBG_FUNC, "cs_convert() desttype = character\n");			if (src_len > destlen) {				tdsdump_log(TDS_DBG_FUNC, "src_len > destlen\n");				ret = CS_FAIL;			} else {				switch (destfmt->format) {				case CS_FMT_NULLTERM:					if (src_len == destlen) {						ret = CS_FAIL;	/* not enough room for data + a null terminator - error */					} else {						memcpy(dest, srcdata, src_len);						dest[src_len] = '\0';						if (resultlen != NULL)							*resultlen = src_len + 1;						ret = CS_SUCCEED;					}					break;				case CS_FMT_PADBLANK:					memcpy(dest, srcdata, src_len);					for (i = src_len; i < destlen; i++)						dest[i] = ' ';					if (resultlen != NULL)						*resultlen = destlen;					ret = CS_SUCCEED;					break;				case CS_FMT_PADNULL:					memcpy(dest, srcdata, src_len);					for (i = src_len; i < destlen; i++)						dest[i] = '\0';					if (resultlen != NULL)						*resultlen = destlen;					ret = CS_SUCCEED;					break;				case CS_FMT_UNUSED:					memcpy(dest, srcdata, src_len);					if (resultlen != NULL)						*resultlen = src_len;					ret = CS_SUCCEED;					break;				default:					tdsdump_log(TDS_DBG_FUNC, "no destination format specified!\n");					ret = CS_FAIL;					break;				}			}			break;		case SYBINT1:		case SYBINT2:		case SYBINT4:		case SYBINT8:		case SYBFLT8:		case SYBREAL:		case SYBBIT:		case SYBMONEY:		case SYBMONEY4:		case SYBDATETIME:		case SYBDATETIME4:			memcpy(dest, srcdata, _cs_datatype_length(src_type));			if (resultlen != NULL)				*resultlen = _cs_datatype_length(src_type);			ret = CS_SUCCEED;			break;		case SYBNUMERIC:		case SYBDECIMAL:			src_len = tds_numeric_bytes_per_prec[((TDS_NUMERIC *) srcdata)->precision] + 2;		case SYBBITN:			if (src_len > destlen) {				ret = CS_FAIL;			} else {				memcpy(dest, srcdata, src_len);				if (resultlen != NULL)					*resultlen = src_len;				ret = CS_SUCCEED;			}			break;		default:			ret = CS_FAIL;			break;		}		return ret;	}	/* src type != dest type */	/* set the output precision/scale for conversions to numeric type */	if (is_numeric_type(desttype)) {		cres.n.precision = destfmt->precision;		cres.n.scale = destfmt->scale;		if (destfmt->precision == CS_SRC_VALUE)			cres.n.precision = srcfmt->precision;		if (destfmt->scale == CS_SRC_VALUE)			cres.n.scale = srcfmt->scale;	}	tdsdump_log(TDS_DBG_FUNC, "cs_convert() calling tds_convert\n");	len = tds_convert(ctx->tds_ctx, src_type, (TDS_CHAR*) srcdata, src_len, desttype, &cres);	tdsdump_log(TDS_DBG_FUNC, "cs_convert() tds_convert returned %d\n", len);	switch (len) {	case TDS_CONVERT_NOAVAIL:		_csclient_msg(ctx, "cs_convert", 2, 1, 1, 16, "%d, %d", src_type, desttype);		return CS_FAIL;		break;	case TDS_CONVERT_SYNTAX:		_csclient_msg(ctx, "cs_convert", 2, 4, 1, 24, "");		return CS_FAIL;		break;	case TDS_CONVERT_NOMEM:		_csclient_msg(ctx, "cs_convert", 2, 4, 1, 3, "");		return CS_FAIL;		break;	case TDS_CONVERT_OVERFLOW:		_csclient_msg(ctx, "cs_convert", 2, 4, 1, 20, "");		return CS_FAIL;		break;	case TDS_CONVERT_FAIL:		return CS_FAIL;		break;	default:		if (len < 0) {			return CS_FAIL;		}		break;	}	switch (desttype) {

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -