mirror of
https://github.com/php/php-src.git
synced 2025-01-06 10:53:39 +08:00
295 lines
6.9 KiB
C
295 lines
6.9 KiB
C
/*
|
|
+----------------------------------------------------------------------+
|
|
| PHP Version 5 |
|
|
+----------------------------------------------------------------------+
|
|
| Copyright (c) 1997-2010 The PHP Group |
|
|
+----------------------------------------------------------------------+
|
|
| This source file is subject to version 3.01 of the PHP license, |
|
|
| that is bundled with this package in the file LICENSE, and is |
|
|
| available through the world-wide-web at the following url: |
|
|
| http://www.php.net/license/3_01.txt |
|
|
| If you did not receive a copy of the PHP license and are unable to |
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
| license@php.net so we can mail you a copy immediately. |
|
|
+----------------------------------------------------------------------+
|
|
| Author: Wez Furlong <wez@php.net> |
|
|
| Frank M. Kromann <frank@kromann.info> |
|
|
+----------------------------------------------------------------------+
|
|
*/
|
|
|
|
/* $Id$ */
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include "php.h"
|
|
#include "php_ini.h"
|
|
#include "ext/standard/info.h"
|
|
#include "pdo/php_pdo.h"
|
|
#include "pdo/php_pdo_driver.h"
|
|
#include "php_pdo_dblib.h"
|
|
#include "php_pdo_dblib_int.h"
|
|
#include "zend_exceptions.h"
|
|
|
|
static void free_rows(pdo_dblib_stmt *S TSRMLS_DC)
|
|
{
|
|
int i, j;
|
|
|
|
for (i = 0; i < S->nrows; i++) {
|
|
for (j = 0; j < S->ncols; j++) {
|
|
pdo_dblib_colval *val = &S->rows[i] + j;
|
|
if (val->data) {
|
|
efree(val->data);
|
|
val->data = NULL;
|
|
}
|
|
}
|
|
}
|
|
efree(S->rows);
|
|
S->rows = NULL;
|
|
S->nrows = 0;
|
|
}
|
|
|
|
static int pdo_dblib_stmt_dtor(pdo_stmt_t *stmt TSRMLS_DC)
|
|
{
|
|
pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
|
|
|
|
if (S->rows) {
|
|
free_rows(S TSRMLS_CC);
|
|
}
|
|
if (S->cols) {
|
|
efree(S->cols);
|
|
}
|
|
efree(S);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int pdo_dblib_stmt_execute(pdo_stmt_t *stmt TSRMLS_DC)
|
|
{
|
|
pdo_dbh_t *dbh = stmt->dbh;
|
|
pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
|
|
pdo_dblib_db_handle *H = S->H;
|
|
RETCODE resret, ret;
|
|
int i, j;
|
|
int arows;
|
|
unsigned int size;
|
|
|
|
dbsetuserdata(H->link, &S->err);
|
|
|
|
if (S->rows) {
|
|
/* clean them up */
|
|
free_rows(S TSRMLS_CC);
|
|
}
|
|
|
|
if (FAIL == dbcmd(H->link, stmt->active_query_string)) {
|
|
return 0;
|
|
}
|
|
if (FAIL == dbsqlexec(H->link)) {
|
|
return 0;
|
|
}
|
|
|
|
resret = dbresults(H->link);
|
|
if (resret == FAIL) {
|
|
return 0;
|
|
}
|
|
|
|
ret = dbnextrow(H->link);
|
|
|
|
stmt->row_count = DBCOUNT(H->link);
|
|
|
|
if (ret == NO_MORE_ROWS) {
|
|
return 1;
|
|
}
|
|
|
|
if (!S->cols) {
|
|
S->ncols = dbnumcols(H->link);
|
|
|
|
if (S->ncols <= 0) {
|
|
return 1;
|
|
}
|
|
|
|
S->cols = ecalloc(S->ncols, sizeof(pdo_dblib_col));
|
|
stmt->column_count = S->ncols;
|
|
|
|
for (i = 0, j = 0; i < S->ncols; i++) {
|
|
char *tmp = NULL;
|
|
|
|
S->cols[i].coltype = dbcoltype(H->link, i+1);
|
|
S->cols[i].name = (char*)dbcolname(H->link, i+1);
|
|
|
|
if (!strlen(S->cols[i].name)) {
|
|
if (j) {
|
|
spprintf(&tmp, 0, "computed%d", j++);
|
|
strlcpy(S->cols[i].name, tmp, strlen(tmp)+1);
|
|
efree(tmp);
|
|
} else {
|
|
S->cols[i].name = "computed";
|
|
j++;
|
|
}
|
|
}
|
|
|
|
S->cols[i].source = (char*)dbcolsource(H->link, i+1);
|
|
tmp = estrdup(S->cols[i].source ? S->cols[i].source : "");
|
|
S->cols[i].source = tmp;
|
|
efree(tmp);
|
|
|
|
S->cols[i].maxlen = dbcollen(H->link, i+1);
|
|
}
|
|
}
|
|
|
|
arows = 100;
|
|
size = S->ncols * sizeof(pdo_dblib_colval);
|
|
S->rows = safe_emalloc(arows, size, 0);
|
|
|
|
/* let's fetch all the data */
|
|
do {
|
|
if (S->nrows >= arows) {
|
|
arows *= 2;
|
|
S->rows = erealloc(S->rows, arows * size);
|
|
}
|
|
for (i = 0; i < S->ncols; i++) {
|
|
pdo_dblib_colval *val = &S->rows[S->nrows * S->ncols + i];
|
|
|
|
if (dbdatlen(H->link, i+1) == 0 && dbdata(H->link, i+1) == NULL) {
|
|
val->len = 0;
|
|
val->data = NULL;
|
|
} else {
|
|
switch (S->cols[i].coltype) {
|
|
case SQLCHAR:
|
|
case SQLTEXT:
|
|
case SQLVARBINARY:
|
|
case SQLBINARY:
|
|
case SQLIMAGE:
|
|
val->len = dbdatlen(H->link, i+1);
|
|
val->data = emalloc(val->len + 1);
|
|
memcpy(val->data, dbdata(H->link, i+1), val->len);
|
|
val->data[val->len] = '\0';
|
|
break;
|
|
case SQLMONEY:
|
|
case SQLMONEY4:
|
|
case SQLMONEYN: {
|
|
DBFLT8 money_value;
|
|
dbconvert(NULL, S->cols[i].coltype, dbdata(H->link, i+1), dbdatlen(H->link, i+1), SQLFLT8, (LPBYTE)&money_value, val->len);
|
|
val->len = spprintf(val->data, 0, "%.4f", money_value);
|
|
}
|
|
break;
|
|
default:
|
|
if (dbwillconvert(S->cols[i].coltype, SQLCHAR)) {
|
|
val->len = 32 + (2 * dbdatlen(H->link, i+1));
|
|
val->data = emalloc(val->len);
|
|
|
|
val->len = dbconvert(NULL, S->cols[i].coltype, dbdata(H->link, i+1),
|
|
dbdatlen(H->link, i+1), SQLCHAR, val->data, val->len);
|
|
|
|
if (val->len >= 0) {
|
|
val->data[val->len] = '\0';
|
|
}
|
|
} else {
|
|
val->len = 0;
|
|
val->data = NULL;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
S->nrows++;
|
|
|
|
ret = dbnextrow(H->link);
|
|
|
|
if (ret == BUF_FULL) {
|
|
dbclrbuf(H->link, DBLASTROW(H->link)-1);
|
|
}
|
|
} while (ret != FAIL && ret != NO_MORE_ROWS);
|
|
|
|
if (resret != NO_MORE_RESULTS) {
|
|
/* there are additional result sets available */
|
|
dbresults(H->link);
|
|
/* cancel pending rows */
|
|
dbcanquery(H->link);
|
|
|
|
/* TODO: figure out a sane solution */
|
|
}
|
|
|
|
S->current = -1;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int pdo_dblib_stmt_fetch(pdo_stmt_t *stmt,
|
|
enum pdo_fetch_orientation ori, long offset TSRMLS_DC)
|
|
{
|
|
pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
|
|
|
|
if (!S->rows) {
|
|
return 0;
|
|
}
|
|
|
|
if (++S->current < S->nrows) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int pdo_dblib_stmt_describe(pdo_stmt_t *stmt, int colno TSRMLS_DC)
|
|
{
|
|
pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
|
|
struct pdo_column_data *col = &stmt->columns[colno];
|
|
|
|
if (!S->rows) {
|
|
return 0;
|
|
}
|
|
|
|
col->maxlen = S->cols[colno].maxlen;
|
|
col->namelen = strlen(S->cols[colno].name);
|
|
col->name = estrdup(S->cols[colno].name);
|
|
col->param_type = PDO_PARAM_STR;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int pdo_dblib_stmt_get_col(pdo_stmt_t *stmt, int colno, char **ptr,
|
|
unsigned long *len, int *caller_frees TSRMLS_DC)
|
|
{
|
|
pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
|
|
pdo_dblib_colval *val = &S->rows[S->current * S->ncols + colno];
|
|
|
|
*ptr = val->data;
|
|
*len = val->len;
|
|
return 1;
|
|
}
|
|
|
|
static int pdo_dblib_stmt_param_hook(pdo_stmt_t *stmt, struct pdo_bound_param_data *param,
|
|
enum pdo_param_event event_type TSRMLS_DC)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static int dblib_dblib_stmt_cursor_closer(pdo_stmt_t *stmt TSRMLS_DC)
|
|
{
|
|
pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
|
|
|
|
if (S->rows) {
|
|
free_rows(S TSRMLS_CC);
|
|
S->rows = NULL;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
struct pdo_stmt_methods dblib_stmt_methods = {
|
|
pdo_dblib_stmt_dtor,
|
|
pdo_dblib_stmt_execute,
|
|
pdo_dblib_stmt_fetch,
|
|
pdo_dblib_stmt_describe,
|
|
pdo_dblib_stmt_get_col,
|
|
pdo_dblib_stmt_param_hook,
|
|
NULL, /* set attr */
|
|
NULL, /* get attr */
|
|
NULL, /* meta */
|
|
NULL, /* nextrow */
|
|
dblib_dblib_stmt_cursor_closer
|
|
};
|
|
|