diff options
author | Mark A. Hershberger <mah@debian.(none)> | 2009-03-25 00:34:21 -0400 |
---|---|---|
committer | Mark A. Hershberger <mah@debian.(none)> | 2009-03-25 00:34:21 -0400 |
commit | 0e920280a2e04b110827bb766b9f29e3d581c4ee (patch) | |
tree | 8f2125f3d00fe3089e3b94adb06f04479ee15f2a /ext/oci8 | |
download | php-0e920280a2e04b110827bb766b9f29e3d581c4ee.tar.gz |
Imported Upstream version 5.0.4upstream/5.0.4
Diffstat (limited to 'ext/oci8')
-rw-r--r-- | ext/oci8/CREDITS | 2 | ||||
-rw-r--r-- | ext/oci8/config.m4 | 271 | ||||
-rw-r--r-- | ext/oci8/config.w32 | 23 | ||||
-rw-r--r-- | ext/oci8/oci8.c | 7154 | ||||
-rw-r--r-- | ext/oci8/oci8.dsp | 113 | ||||
-rw-r--r-- | ext/oci8/package.xml | 85 | ||||
-rw-r--r-- | ext/oci8/php_oci8.h | 225 | ||||
-rw-r--r-- | ext/oci8/tests/bug26133.phpt | 36 | ||||
-rw-r--r-- | ext/oci8/tests/connect.inc | 39 | ||||
-rw-r--r-- | ext/oci8/tests/create_table.inc | 12 | ||||
-rw-r--r-- | ext/oci8/tests/drop_table.inc | 12 | ||||
-rw-r--r-- | ext/oci8/tests/skipif.inc | 10 |
12 files changed, 7982 insertions, 0 deletions
diff --git a/ext/oci8/CREDITS b/ext/oci8/CREDITS new file mode 100644 index 000000000..884efcaa9 --- /dev/null +++ b/ext/oci8/CREDITS @@ -0,0 +1,2 @@ +OCI8 +Stig Bakken, Thies C. Arntzen, Andy Sautins, David Benson, Maxim Maletsky, Harald Radi, Antony Dovgal diff --git a/ext/oci8/config.m4 b/ext/oci8/config.m4 new file mode 100644 index 000000000..48ac58746 --- /dev/null +++ b/ext/oci8/config.m4 @@ -0,0 +1,271 @@ +dnl +dnl $Id: config.m4,v 1.50.2.3 2005/02/25 11:32:01 tony2001 Exp $ +dnl + +AC_DEFUN([PHP_OCI_IF_DEFINED],[ + old_CPPFLAGS=$CPPFLAGS + CPPFLAGS=$3 + AC_EGREP_CPP(yes,[ +#include <oci.h> +#if defined($1) + yes +#endif + ],[ + CPPFLAGS=$old_CPPFLAGS + $2 + ],[ + CPPFLAGS=$old_CPPFLAGS + ]) +]) + +AC_DEFUN([AC_OCI8_VERSION],[ + AC_MSG_CHECKING([Oracle version]) + if test -s "$OCI8_DIR/orainst/unix.rgs"; then + OCI8_VERSION=`grep '"ocommon"' $OCI8_DIR/orainst/unix.rgs | sed 's/[ ][ ]*/:/g' | cut -d: -f 6 | cut -c 2-4` + test -z "$OCI8_VERSION" && OCI8_VERSION=7.3 + elif test -f $OCI8_DIR/lib/libclntsh.$SHLIB_SUFFIX_NAME.10.1; then + OCI8_VERSION=10.1 + elif test -f $OCI8_DIR/lib/libclntsh.$SHLIB_SUFFIX_NAME.9.0; then + OCI8_VERSION=9.0 + elif test -f $OCI8_DIR/lib/libclntsh.$SHLIB_SUFFIX_NAME.8.0; then + OCI8_VERSION=8.1 + elif test -f $OCI8_DIR/lib/libclntsh.$SHLIB_SUFFIX_NAME.1.0; then + OCI8_VERSION=8.0 + elif test -f $OCI8_DIR/lib/libclntsh.a; then + if test -f $OCI8_DIR/lib/libcore4.a; then + OCI8_VERSION=8.0 + else + OCI8_VERSION=8.1 + fi + else + AC_MSG_ERROR([Oracle (OCI8) required libraries not found]) + fi + AC_MSG_RESULT($OCI8_VERSION) +]) + +AC_DEFUN([AC_OCI8IC_VERSION],[ + AC_MSG_CHECKING([Oracle Instant Client version]) + if test -f $PHP_OCI8_INSTANT_CLIENT/libociei.$SHLIB_SUFFIX_NAME; then + if test -f $PHP_OCI8_INSTANT_CLIENT/libclntsh.$SHLIB_SUFFIX_NAME.10.1; then + if test ! -f $PHP_OCI8_INSTANT_CLIENT/libclntsh.$SHLIB_SUFFIX_NAME; then + AC_MSG_ERROR([Link from $PHP_OCI8_INSTANT_CLIENT/libclntsh.$SHLIB_SUFFIX_NAME to libclntsh.$SHLIB_SUFFIX_NAME.10.1 not found]) + fi + OCI8_VERSION=10.1 + else + AC_MSG_ERROR([Oracle Instant Client library version not supported]) + fi + else + AC_MSG_ERROR([Oracle Instant Client libraries not found]) + fi + AC_MSG_RESULT([$OCI8_VERSION]) +]) + +PHP_ARG_WITH(oci8, for Oracle (OCI8) support using ORACLE_HOME installation, +[ --with-oci8[=DIR] Include Oracle (OCI8) support using an ORACLE_HOME + install. The default DIR is ORACLE_HOME]) + +if test "$PHP_OCI8" = "no"; then + PHP_ARG_WITH(oci8-instant-client, for Oracle (OCI8) support using Oracle Instant Client, + [ --with-oci8-instant-client[=DIR] + Include Oracle (OCI8) support using + Oracle Instant Client. DIR is the directory with the + Instant Client libraries. On Linux it will default to + /usr/lib/oracle/<most_recent_version>/client/lib + Other platforms will need to have it explicitly specified.]) +else + PHP_OCI8_INSTANT_CLIENT="no"; +fi + +if test "$PHP_OCI8" != "no"; then + + if test "$PHP_OCI8_INSTANT_CLIENT" != "no"; then + AC_MSG_ERROR([--with-oci8 and --with-oci8-instant-client are mutually exclusive]) + fi + + AC_MSG_CHECKING([Oracle Install Directory]) + if test "$PHP_OCI8" = "yes"; then + OCI8_DIR=$ORACLE_HOME + else + OCI8_DIR=$PHP_OCI8 + fi + AC_MSG_RESULT($OCI8_DIR) + + if test -d "$OCI8_DIR/rdbms/public"; then + PHP_ADD_INCLUDE($OCI8_DIR/rdbms/public) + OCI8_INCLUDES="$OCI8_INCLUDES -I$OCI8_DIR/rdbms/public" + fi + if test -d "$OCI8_DIR/rdbms/demo"; then + PHP_ADD_INCLUDE($OCI8_DIR/rdbms/demo) + OCI8_INCLUDES="$OCI8_INCLUDES -I$OCI8_DIR/rdbms/demo" + fi + if test -d "$OCI8_DIR/network/public"; then + PHP_ADD_INCLUDE($OCI8_DIR/network/public) + OCI8_INCLUDES="$OCI8_INCLUDES -I$OCI8_DIR/network/public" + fi + if test -d "$OCI8_DIR/plsql/public"; then + PHP_ADD_INCLUDE($OCI8_DIR/plsql/public) + OCI8_INCLUDES="$OCI8_INCLUDES -I$OCI8_DIR/plsql/public" + fi + + if test -f "$OCI8_DIR/lib/sysliblist"; then + PHP_EVAL_LIBLINE(`cat $OCI8_DIR/lib/sysliblist`, OCI8_SYSLIB) + elif test -f "$OCI8_DIR/rdbms/lib/sysliblist"; then + PHP_EVAL_LIBLINE(`cat $OCI8_DIR/rdbms/lib/sysliblist`, OCI8_SYSLIB) + fi + + AC_OCI8_VERSION($OCI8_DIR) + case $OCI8_VERSION in + 8.0) + PHP_ADD_LIBRARY_WITH_PATH(nlsrtl3, "", OCI8_SHARED_LIBADD) + PHP_ADD_LIBRARY_WITH_PATH(core4, "", OCI8_SHARED_LIBADD) + PHP_ADD_LIBRARY_WITH_PATH(psa, "", OCI8_SHARED_LIBADD) + PHP_ADD_LIBRARY_WITH_PATH(clntsh, $OCI8_DIR/lib, OCI8_SHARED_LIBADD) + ;; + + 8.1) + PHP_ADD_LIBRARY(clntsh, 1, OCI8_SHARED_LIBADD) + PHP_ADD_LIBPATH($OCI8_DIR/lib, OCI8_SHARED_LIBADD) + + dnl + dnl OCI_ATTR_STATEMENT is not available in all 8.1.x versions + dnl + PHP_OCI_IF_DEFINED(OCI_ATTR_STATEMENT, [AC_DEFINE(HAVE_OCI8_ATTR_STATEMENT,1,[ ])], $OCI8_INCLUDES) + ;; + + 9.0) + PHP_ADD_LIBRARY(clntsh, 1, OCI8_SHARED_LIBADD) + PHP_ADD_LIBPATH($OCI8_DIR/lib, OCI8_SHARED_LIBADD) + AC_DEFINE(HAVE_OCI8_ATTR_STATEMENT,1,[ ]) + + dnl These functions are only available in version >= 9.2 + PHP_CHECK_LIBRARY(clntsh, OCIEnvNlsCreate, + [ + PHP_CHECK_LIBRARY(clntsh, OCINlsCharSetNameToId, + [ + AC_DEFINE(HAVE_OCI_9_2,1,[ ]) + OCI8_VERSION=9.2 + ], [], [ + -L$OCI8_DIR/lib $OCI8_SHARED_LIBADD + ]) + ], [], [ + -L$OCI8_DIR/lib $OCI8_SHARED_LIBADD + ]) + ;; + + 10.1) + PHP_ADD_LIBRARY(clntsh, 1, OCI8_SHARED_LIBADD) + PHP_ADD_LIBPATH($OCI8_DIR/lib, OCI8_SHARED_LIBADD) + AC_DEFINE(HAVE_OCI8_ATTR_STATEMENT,1,[ ]) + AC_DEFINE(HAVE_OCI_9_2,1,[ ]) + AC_DEFINE(HAVE_OCI8_TEMP_LOB,1,[ ]) + AC_DEFINE(PHP_OCI8_HAVE_COLLECTIONS,1,[ ]) + ;; + + *) + AC_MSG_ERROR([Unsupported Oracle version]) + ;; + esac + + dnl + dnl Check if we need to add -locijdbc8 + dnl + PHP_CHECK_LIBRARY(clntsh, OCILobIsTemporary, + [ + AC_DEFINE(HAVE_OCI8_TEMP_LOB,1,[ ]) + ], [ + PHP_CHECK_LIBRARY(ocijdbc8, OCILobIsTemporary, + [ + PHP_ADD_LIBRARY(ocijdbc8, 1, OCI8_SHARED_LIBADD) + AC_DEFINE(HAVE_OCI8_TEMP_LOB,1,[ ]) + ], [], [ + -L$OCI8_DIR/lib $OCI8_SHARED_LIBADD + ]) + ], [ + -L$OCI8_DIR/lib $OCI8_SHARED_LIBADD + ]) + + dnl + dnl Check if we have collections + dnl + PHP_CHECK_LIBRARY(clntsh, OCICollAssign, + [ + AC_DEFINE(PHP_OCI8_HAVE_COLLECTIONS,1,[ ]) + ], [], [ + -L$OCI8_DIR/lib $OCI8_SHARED_LIBADD + ]) + + + PHP_NEW_EXTENSION(oci8, oci8.c, $ext_shared) + AC_DEFINE(HAVE_OCI8,1,[ ]) + + PHP_SUBST_OLD(OCI8_SHARED_LIBADD) + PHP_SUBST_OLD(OCI8_DIR) + PHP_SUBST_OLD(OCI8_VERSION) + +elif test "$PHP_OCI8_INSTANT_CLIENT" != "no"; then + + AC_MSG_CHECKING([Oracle Instant Client directory]) + if test "$PHP_OCI8_INSTANT_CLIENT" = "yes"; then +dnl Generally the Instant Client can be anywhere so the user must pass in the +dnl directory to the libraries. But on Linux we default to the most recent +dnl version in /usr/lib + PHP_OCI8_INSTANT_CLIENT=`ls -d /usr/lib/oracle/*/client/lib 2> /dev/null | tail -1` + if test -z "$PHP_OCI8_INSTANT_CLIENT"; then + AC_MSG_ERROR([Oracle Instant Client directory not found. Try --with-oci8-instant-client=DIR]) + fi + fi + AC_MSG_RESULT($PHP_OCI8_INSTANT_CLIENT) + + OCI8_DIR=$PHP_OCI8_INSTANT_CLIENT + + AC_MSG_CHECKING([Oracle Instant Client SDK header directory]) + +dnl Header directory for Instant Client SDK RPM install + OCISDKRPMINC=`echo "$PHP_OCI8_INSTANT_CLIENT" | sed -e 's!^/usr/lib/oracle/\(.*\)/client/lib[[/]]*$!/usr/include/oracle/\1/client!'` + +dnl Header directory for Instant Client SDK zip file install + OCISDKZIPINC=$PHP_OCI8_INSTANT_CLIENT/sdk/include + + if test -f "$OCISDKRPMINC/oci.h"; then + AC_MSG_RESULT($OCISDKRPMINC) + PHP_ADD_INCLUDE($OCISDKRPMINC) + OCI8INCDIR=$OCISDKRPMINC + elif test -f "$OCISDKZIPINC/oci.h"; then + AC_MSG_RESULT($OCISDKZIPINC) + PHP_ADD_INCLUDE($OCISDKZIPINC) + OCI8INCDIR=$OCISDKZIPINC + else + AC_MSG_ERROR([Oracle Instant Client SDK header files not found]) + fi + + OCISYSLIBLIST=`echo "$OCI8INCDIR" | sed -e 's!\(.*\)/include$!\1/demo/sysliblist!'` + if test -f "$OCISYSLIBLIST"; then + PHP_EVAL_LIBLINE(`cat $OCISYSLIBLIST`, OCI8_SYSLIB) + fi + + AC_OCI8IC_VERSION($PHP_OCI8_INSTANT_CLIENT) + case $OCI8_VERSION in + 10.1) + PHP_ADD_LIBRARY(clntsh, 1, OCI8_SHARED_LIBADD) + PHP_ADD_LIBPATH($PHP_OCI8_INSTANT_CLIENT, OCI8_SHARED_LIBADD) + ;; + + *) + AC_MSG_ERROR([Unsupported Oracle Instant Client version]) + ;; + esac + + AC_DEFINE(HAVE_OCI8_ATTR_STATEMENT,1,[ ]) + AC_DEFINE(HAVE_OCI_9_2,1,[ ]) + AC_DEFINE(HAVE_OCI8_TEMP_LOB,1,[ ]) + AC_DEFINE(PHP_OCI8_HAVE_COLLECTIONS,1,[ ]) + AC_DEFINE(HAVE_OCI_INSTANT_CLIENT,1,[ ]) + + PHP_NEW_EXTENSION(oci8, oci8.c, $ext_shared) + AC_DEFINE(HAVE_OCI8,1,[ ]) + + PHP_SUBST_OLD(OCI8_SHARED_LIBADD) + PHP_SUBST_OLD(OCI8_DIR) + PHP_SUBST_OLD(OCI8_VERSION) + +fi diff --git a/ext/oci8/config.w32 b/ext/oci8/config.w32 new file mode 100644 index 000000000..10ca4a9d1 --- /dev/null +++ b/ext/oci8/config.w32 @@ -0,0 +1,23 @@ +// $Id: config.w32,v 1.3 2004/01/22 14:42:17 tony2001 Exp $ +// vim:ft=javascript + +ARG_WITH("oci8", "OCI8 support", "no"); + +if (PHP_OCI8 != "no") { + + if (CHECK_HEADER_ADD_INCLUDE("oci.h", "CFLAGS_OCI8", PHP_PHP_BUILD + "\\oci805\\include;" + PHP_OCI8) && + CHECK_LIB("oci.lib", "oci8", PHP_OCI8) + + ) { + EXTENSION('oci8', 'oci8.c'); + + AC_DEFINE('HAVE_OCI8_TEMP_LOB', 1); + AC_DEFINE('HAVE_OCI8', 1); + AC_DEFINE('HAVE_OCI8_ATTR_STATEMENT', 1); + AC_DEFINE('PHP_OCI8_HAVE_COLLECTIONS', 1); + + } else { + WARNING("oci8 not enabled; libraries and headers not found"); + } +} + diff --git a/ext/oci8/oci8.c b/ext/oci8/oci8.c new file mode 100644 index 000000000..48350d674 --- /dev/null +++ b/ext/oci8/oci8.c @@ -0,0 +1,7154 @@ +/* + +----------------------------------------------------------------------+ + | PHP Version 5 | + +----------------------------------------------------------------------+ + | Copyright (c) 1997-2004 The PHP Group | + +----------------------------------------------------------------------+ + | This source file is subject to version 3.0 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_0.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. | + +----------------------------------------------------------------------+ + | Authors: Stig Sæther Bakken <ssb@php.net> | + | Thies C. Arntzen <thies@thieso.net> | + | Maxim Maletsky <maxim@maxim.cx> | + | | + | Collection support by Andy Sautins <asautins@veripost.net> | + | Temporary LOB support by David Benson <dbenson@mancala.com> | + | ZTS per process OCIPLogon by Harald Radi <harald.radi@nme.at> | + +----------------------------------------------------------------------+ + */ + +/* $Id: oci8.c,v 1.257.2.6 2005/01/20 18:42:40 tony2001 Exp $ */ + +/* TODO list: + * + * - php.ini flags + * especialliy important for things like oci_ping + * allowpconns + * timeout + * maxlifetime + * maxpconns + * - Change return-value for OCIFetch*() (1-row read, 0-Normal end, false-error) + * - Error mode (print or shut up?) + * - binding of arrays + * - Character sets for NCLOBS + * - split the module into an upper (php-callable) and lower (c-callable) layer! + * - remove all XXXs + * - clean up and documentation + * - make OCIInternalDebug accept a mask of flags.... + * - have one ocifree() call. + * - make it possible to have persistent statements? + * - failover + * - change all the lob stuff to work without classes (optional)! + * - make sure that the callbacks terminate the strings with \0 + * - cleanup the ociexecute semantics for refcursors + * - make $lob->savefile use O_BINARY + * - line 2728: ub4 length = -1; needs fixing + * - delay OCIInitialize() as far as we can. + * - add PHP Array <-> OCICollection conversion + * - add Collection iterator object for INDEX BY tables + * - make auto-rollback only happen if we have an outstanding transaction + * - implement ocidisconnect + * - add OCI9-specific functions and separate them from OCI8 with ifdefs + */ + +/* {{{ includes & stuff */ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include "php.h" +#include "ext/standard/info.h" +#include "php_ini.h" + +#if HAVE_OCI8 + +#include "php_oci8.h" + +/* True globals, only used by thread safe functions */ +static TsHashTable *persistent_servers; +static TsHashTable *persistent_sessions; + +static long num_persistent = 0; +static long num_links = 0; + +/* True globals, no need for thread safety */ +static int le_conn; +static int le_stmt; +static int le_desc; +#ifdef PHP_OCI8_HAVE_COLLECTIONS +static int le_coll; +#endif +static int le_server; +static int le_session; +static zend_class_entry *oci_lob_class_entry_ptr; +#ifdef PHP_OCI8_HAVE_COLLECTIONS +static zend_class_entry *oci_coll_class_entry_ptr; +#endif + +#ifndef SQLT_BFILEE +#define SQLT_BFILEE 114 +#endif +#ifndef SQLT_CFILEE +#define SQLT_CFILEE 115 +#endif + +#define SAFE_STRING(s) ((s)?(s):"") + +#ifdef ZTS +MUTEX_T mx_lock; + +#define mutex_alloc(mutex) mutex = tsrm_mutex_alloc() +#define mutex_free(mutex) tsrm_mutex_free(mutex) +#define mutex_lock(mutex) tsrm_mutex_lock(mutex) +#define mutex_unlock(mutex) tsrm_mutex_unlock(mutex) +#define thread_id() tsrm_thread_id() +#else +#define mutex_alloc(mutex) +#define mutex_free(mutex) +#define mutex_lock(mutex) +#define mutex_unlock(mutex) +#define thread_id() 1 +#endif + +/* dirty marcos to make sure we _never_ call oracle-functions recursivly + * + * i'm well aware that we should _never_ call exit directly - this core is for + * pure testing and commented out - as you can see;-) + * thies@thieso.net 20010723 + */ + +#define CALL_OCI(call) \ +{ \ + if (OCI(in_call)) { \ + php_error_docref(NULL TSRMLS_CC, E_WARNING, "OCI8 Recursive call!\n"); \ + exit(-1); \ + } else { \ + OCI(in_call)=1; \ + call; \ + OCI(in_call)=0; \ + } \ +} + +#define CALL_OCI_RETURN(retcode,call) \ +{ \ + if (OCI(in_call)) { \ + retcode=-1; \ + php_error_docref(NULL TSRMLS_CC, E_WARNING, "OCI8 Recursive call!\n"); \ + exit(-1); \ + } else { \ + OCI(in_call)=1; \ + retcode=call; \ + OCI(in_call)=0; \ + } \ +} + +#include <fcntl.h> + +#ifndef O_BINARY +#define O_BINARY 0 +#endif + +/* }}} */ +/* {{{ thread safety stuff */ + +#ifdef ZTS +int oci_globals_id; +#else +PHP_OCI_API php_oci_globals oci_globals; +#endif + +/* }}} */ +/* {{{ dynamically loadable module stuff */ + +#ifdef COMPILE_DL_OCI8 +ZEND_GET_MODULE(oci8) +# ifdef PHP_WIN32 +# include "zend_arg_defs.c" +# endif +#endif /* COMPILE_DL */ + +/* }}} */ +/* {{{ startup/shutdown/info/internal function prototypes */ + +PHP_MINIT_FUNCTION(oci); +PHP_RINIT_FUNCTION(oci); +PHP_MSHUTDOWN_FUNCTION(oci); +PHP_RSHUTDOWN_FUNCTION(oci); +PHP_MINFO_FUNCTION(oci); + +static ub4 oci_handle_error(oci_connection *connection, ub4 errcode); +static ub4 oci_error(OCIError *err_p, char *what, sword status); +static int oci_ping(oci_server *server); +static void oci_debug(const char *format, ...); + +static void _oci_conn_list_dtor(oci_connection *connection TSRMLS_DC); +static void _oci_stmt_list_dtor(zend_rsrc_list_entry *rsrc TSRMLS_DC); +static void _oci_descriptor_list_dtor(zend_rsrc_list_entry *rsrc TSRMLS_DC); +#ifdef PHP_OCI8_HAVE_COLLECTIONS +static void _oci_coll_list_dtor(zend_rsrc_list_entry *rsrc TSRMLS_DC); +#endif +static void _oci_server_list_dtor(zend_rsrc_list_entry *rsrc TSRMLS_DC); +static void _oci_session_list_dtor(zend_rsrc_list_entry *rsrc TSRMLS_DC); +static void php_oci_free_conn_list(zend_rsrc_list_entry *rsrc TSRMLS_DC); + +static void _oci_column_hash_dtor(void *data); +static void _oci_define_hash_dtor(void *data); +static void _oci_bind_hash_dtor(void *data); +static void _oci_desc_flush_hash_dtor(void *data); + +static oci_connection *oci_get_conn(zval ** TSRMLS_DC); +static oci_statement *oci_get_stmt(zval ** TSRMLS_DC); +static oci_descriptor *oci_get_desc(int TSRMLS_DC); +#ifdef PHP_OCI8_HAVE_COLLECTIONS +/* Questionable name. Very close to oci_get_col */ +static oci_collection *oci_get_coll(int TSRMLS_DC); +#endif +static oci_out_column *oci_get_col(oci_statement *, int, zval **); + +static int _oci_make_zval(zval *, oci_statement *, oci_out_column *, char *, int mode TSRMLS_DC); +static oci_statement *oci_parse(oci_connection *, char *, int); +static int oci_execute(oci_statement *, char *, ub4 mode); +static int oci_fetch(oci_statement *, ub4, char * TSRMLS_DC); +static int oci_lobgetlen(oci_connection *, oci_descriptor *, ub4 *length); +static int oci_loadlob(oci_connection *, oci_descriptor *, char **, ub4 *length); +static int oci_readlob(oci_connection *, oci_descriptor *, char **, ub4 *len); +static int oci_setprefetch(oci_statement *statement, int size); + +static void oci_do_connect(INTERNAL_FUNCTION_PARAMETERS,int persistent,int exclusive); + +static oci_server *_oci_open_server(char *dbname,int persistent); +static void _oci_close_server(oci_server *server); + +static oci_session *_oci_open_session(oci_server* server,char *username,char *password,int persistent,int exclusive, char *charset); +static void _oci_close_session(oci_session *session); + +static sb4 oci_bind_in_callback(dvoid *, OCIBind *, ub4, ub4, dvoid **, ub4 *, ub1 *, dvoid **); +static sb4 oci_bind_out_callback(dvoid *, OCIBind *, ub4, ub4, dvoid **, ub4 **, ub1 *, dvoid **, ub2 **); + +#if 0 +static sb4 oci_failover_callback(dvoid *svchp,dvoid* envhp,dvoid *fo_ctx,ub4 fo_type, ub4 fo_event); +#endif + +static int oci_lob_flush(oci_descriptor *, int flush_flag TSRMLS_DC); + +/* }}} */ + +/* {{{ extension macros +*/ +#define OCI_GET_STMT(statement,value) \ + statement = oci_get_stmt(value TSRMLS_CC); \ + if (statement == NULL) { \ + RETURN_FALSE; \ + } + +#define OCI_GET_CONN(connection,value) \ + connection = oci_get_conn(value TSRMLS_CC); \ + if (connection == NULL) { \ + RETURN_FALSE; \ + } + +#define OCI_GET_DESC(descriptor,index) \ + descriptor = oci_get_desc(index TSRMLS_CC); \ + if (descriptor == NULL) { \ + RETURN_FALSE; \ + } + +#ifdef PHP_OCI8_HAVE_COLLECTIONS +#define OCI_GET_COLL(collection,index) \ + collection = oci_get_coll(index TSRMLS_CC); \ + if (collection == NULL) { \ + RETURN_FALSE; \ + } +#endif + +#define IS_LOB_INTERNAL(lob) \ + if (lob->type != OCI_DTYPE_LOB) { \ + switch (lob->type) { \ + case OCI_DTYPE_FILE: \ + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "internal LOB was expected, FILE locator is given"); \ + break; \ + case OCI_DTYPE_ROWID: \ + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "internal LOB was expected, ROWID locator is given"); \ + break; \ + default: \ + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "internal LOB was expected, locator of unknown type is given"); \ + break; \ + } \ + RETURN_FALSE; \ + } +/* }}} */ + +/* {{{ extension function prototypes +*/ +PHP_FUNCTION(oci_bind_by_name); +PHP_FUNCTION(oci_define_by_name); +PHP_FUNCTION(oci_field_is_null); +PHP_FUNCTION(oci_field_name); +PHP_FUNCTION(oci_field_size); +PHP_FUNCTION(oci_field_scale); +PHP_FUNCTION(oci_field_precision); +PHP_FUNCTION(oci_field_type); +PHP_FUNCTION(oci_field_type_raw); +PHP_FUNCTION(oci_execute); +PHP_FUNCTION(oci_fetch); +PHP_FUNCTION(oci_cancel); +PHP_FUNCTION(ocifetchinto); +PHP_FUNCTION(oci_fetch_object); +PHP_FUNCTION(oci_fetch_row); +PHP_FUNCTION(oci_fetch_assoc); +PHP_FUNCTION(oci_fetch_array); +PHP_FUNCTION(ocifetchstatement); +PHP_FUNCTION(oci_fetch_all); +PHP_FUNCTION(oci_free_statement); +PHP_FUNCTION(oci_internal_debug); +PHP_FUNCTION(oci_close); +PHP_FUNCTION(oci_connect); +PHP_FUNCTION(oci_new_connect); +PHP_FUNCTION(oci_pconnect); +PHP_FUNCTION(oci_error); +PHP_FUNCTION(oci_free_descriptor); +PHP_FUNCTION(oci_commit); +PHP_FUNCTION(oci_rollback); +PHP_FUNCTION(oci_new_descriptor); +PHP_FUNCTION(oci_num_fields); +PHP_FUNCTION(oci_parse); +PHP_FUNCTION(oci_new_cursor); +PHP_FUNCTION(oci_result); +PHP_FUNCTION(oci_server_version); +PHP_FUNCTION(oci_statement_type); +PHP_FUNCTION(oci_num_rows); +PHP_FUNCTION(oci_set_prefetch); +PHP_FUNCTION(oci_password_change); +PHP_FUNCTION(oci_lob_save); +PHP_FUNCTION(oci_lob_import); +PHP_FUNCTION(oci_lob_export); +PHP_FUNCTION(oci_lob_load); +PHP_FUNCTION(oci_lob_tell); +PHP_FUNCTION(oci_lob_write); +PHP_FUNCTION(oci_lob_append); +PHP_FUNCTION(oci_lob_copy); +PHP_FUNCTION(oci_lob_truncate); +PHP_FUNCTION(oci_lob_erase); +PHP_FUNCTION(oci_lob_flush); +PHP_FUNCTION(ocisetbufferinglob); +PHP_FUNCTION(ocigetbufferinglob); +PHP_FUNCTION(oci_lob_is_equal); +PHP_FUNCTION(oci_lob_rewind); +PHP_FUNCTION(oci_lob_read); +PHP_FUNCTION(oci_lob_eof); +PHP_FUNCTION(oci_lob_seek); +PHP_FUNCTION(oci_lob_size); +#ifdef HAVE_OCI8_TEMP_LOB +PHP_FUNCTION(oci_lob_write_temporary); +PHP_FUNCTION(oci_lob_close); +#endif +#ifdef PHP_OCI8_HAVE_COLLECTIONS +PHP_FUNCTION(oci_new_collection); +PHP_FUNCTION(oci_free_collection); +PHP_FUNCTION(oci_collection_append); +PHP_FUNCTION(oci_collection_element_get); +PHP_FUNCTION(oci_collection_element_assign); +PHP_FUNCTION(oci_collection_assign); +PHP_FUNCTION(oci_collection_size); +PHP_FUNCTION(oci_collection_max); +PHP_FUNCTION(oci_collection_trim); +#endif +/* }}} */ + +/* {{{ extension definition structures +*/ +#define OCI_ASSOC 1<<0 +#define OCI_NUM 1<<1 +#define OCI_BOTH (OCI_ASSOC|OCI_NUM) + +#define OCI_RETURN_NULLS 1<<2 +#define OCI_RETURN_LOBS 1<<3 + +#define OCI_FETCHSTATEMENT_BY_COLUMN 1<<4 +#define OCI_FETCHSTATEMENT_BY_ROW 1<<5 +#define OCI_FETCHSTATEMENT_BY (OCI_FETCHSTATEMENT_BY_COLUMN | OCI_FETCHSTATEMENT_BY_ROW) + +static zend_function_entry php_oci_functions[] = { + PHP_FE(oci_define_by_name, third_arg_force_ref) + PHP_FE(oci_bind_by_name, third_arg_force_ref) + PHP_FE(oci_field_is_null, NULL) + PHP_FE(oci_field_name, NULL) + PHP_FE(oci_field_size, NULL) + PHP_FE(oci_field_scale, NULL) + PHP_FE(oci_field_precision, NULL) + PHP_FE(oci_field_type, NULL) + PHP_FE(oci_field_type_raw, NULL) + PHP_FE(oci_execute, NULL) + PHP_FE(oci_cancel, NULL) + PHP_FE(oci_fetch, NULL) + PHP_FE(oci_fetch_object, NULL) + PHP_FE(oci_fetch_row, NULL) + PHP_FE(oci_fetch_assoc, NULL) + PHP_FE(oci_fetch_array, NULL) + PHP_FE(ocifetchinto, second_arg_force_ref) + PHP_FE(oci_fetch_all, second_arg_force_ref) + PHP_FE(oci_free_statement, NULL) + PHP_FE(oci_internal_debug, NULL) + PHP_FE(oci_num_fields, NULL) + PHP_FE(oci_parse, NULL) + PHP_FE(oci_new_cursor, NULL) + PHP_FE(oci_result, NULL) + PHP_FE(oci_server_version, NULL) + PHP_FE(oci_statement_type, NULL) + PHP_FE(oci_num_rows, NULL) + PHP_FE(oci_close, NULL) + PHP_FE(oci_connect, NULL) + PHP_FE(oci_new_connect, NULL) + PHP_FE(oci_pconnect, NULL) + PHP_FE(oci_error, NULL) + PHP_FE(oci_free_descriptor, NULL) + PHP_FE(oci_lob_save, NULL) + PHP_FE(oci_lob_import, NULL) + PHP_FE(oci_lob_size, NULL) + PHP_FE(oci_lob_load, NULL) + PHP_FE(oci_lob_read, NULL) + PHP_FE(oci_lob_eof, NULL) + PHP_FE(oci_lob_tell, NULL) + PHP_FE(oci_lob_truncate, NULL) + PHP_FE(oci_lob_erase, NULL) + PHP_FE(oci_lob_flush, NULL) + PHP_FE(ocisetbufferinglob, NULL) + PHP_FE(ocigetbufferinglob, NULL) + PHP_FE(oci_lob_is_equal, NULL) + PHP_FE(oci_lob_rewind, NULL) + PHP_FE(oci_lob_write, NULL) + PHP_FE(oci_lob_append, NULL) + PHP_FE(oci_lob_copy, NULL) + PHP_FE(oci_lob_export, NULL) + PHP_FE(oci_commit, NULL) + PHP_FE(oci_rollback, NULL) + PHP_FE(oci_new_descriptor, NULL) + PHP_FE(oci_set_prefetch, NULL) + PHP_FE(oci_password_change, NULL) +#ifdef PHP_OCI8_HAVE_COLLECTIONS + PHP_FE(oci_free_collection, NULL) + PHP_FE(oci_collection_append, NULL) + PHP_FE(oci_collection_element_get, NULL) + PHP_FE(oci_collection_element_assign, NULL) + PHP_FE(oci_collection_assign, NULL) + PHP_FE(oci_collection_size, NULL) + PHP_FE(oci_collection_max, NULL) + PHP_FE(oci_collection_trim, NULL) + PHP_FE(oci_new_collection, NULL) +#endif + + PHP_FALIAS(oci_free_cursor, oci_free_statement, NULL) + PHP_FALIAS(ocifreecursor, oci_free_statement, NULL) + PHP_FALIAS(ocibindbyname, oci_bind_by_name, third_arg_force_ref) + PHP_FALIAS(ocidefinebyname, oci_define_by_name, third_arg_force_ref) + PHP_FALIAS(ocicolumnisnull, oci_field_is_null, NULL) + PHP_FALIAS(ocicolumnname, oci_field_name, NULL) + PHP_FALIAS(ocicolumnsize, oci_field_size, NULL) + PHP_FALIAS(ocicolumnscale, oci_field_scale, NULL) + PHP_FALIAS(ocicolumnprecision, oci_field_precision, NULL) + PHP_FALIAS(ocicolumntype, oci_field_type, NULL) + PHP_FALIAS(ocicolumntyperaw, oci_field_type_raw, NULL) + PHP_FALIAS(ociexecute, oci_execute, NULL) + PHP_FALIAS(ocicancel, oci_cancel, NULL) + PHP_FALIAS(ocifetch, oci_fetch, NULL) + PHP_FALIAS(ocifetchstatement, oci_fetch_all, second_arg_force_ref) + PHP_FALIAS(ocifreestatement, oci_free_statement, NULL) + PHP_FALIAS(ociinternaldebug, oci_internal_debug, NULL) + PHP_FALIAS(ocinumcols, oci_num_fields, NULL) + PHP_FALIAS(ociparse, oci_parse, NULL) + PHP_FALIAS(ocinewcursor, oci_new_cursor, NULL) + PHP_FALIAS(ociresult, oci_result, NULL) + PHP_FALIAS(ociserverversion, oci_server_version, NULL) + PHP_FALIAS(ocistatementtype, oci_statement_type, NULL) + PHP_FALIAS(ocirowcount, oci_num_rows, NULL) + PHP_FALIAS(ocilogoff, oci_close, NULL) + PHP_FALIAS(ocilogon, oci_connect, NULL) + PHP_FALIAS(ocinlogon, oci_new_connect, NULL) + PHP_FALIAS(ociplogon, oci_pconnect, NULL) + PHP_FALIAS(ocierror, oci_error, NULL) + PHP_FALIAS(ocifreedesc, oci_free_descriptor, NULL) + PHP_FALIAS(ocisavelob, oci_lob_save, NULL) + PHP_FALIAS(ocisavelobfile, oci_lob_import, NULL) + PHP_FALIAS(ociwritelobtofile, oci_lob_export, NULL) + PHP_FALIAS(ociloadlob, oci_lob_load, NULL) + PHP_FALIAS(ocicommit, oci_commit, NULL) + PHP_FALIAS(ocirollback, oci_rollback, NULL) + PHP_FALIAS(ocinewdescriptor, oci_new_descriptor, NULL) + PHP_FALIAS(ocisetprefetch, oci_set_prefetch, NULL) + PHP_FALIAS(ocipasswordchange, oci_password_change, NULL) +#ifdef PHP_OCI8_HAVE_COLLECTIONS + PHP_FALIAS(ocifreecollection, oci_free_collection, NULL) + PHP_FALIAS(ocinewcollection, oci_new_collection, NULL) + PHP_FALIAS(ocicollappend, oci_collection_append, NULL) + PHP_FALIAS(ocicollgetelem, oci_collection_element_get, NULL) + PHP_FALIAS(ocicollassignelem, oci_collection_element_assign, NULL) + PHP_FALIAS(ocicollsize, oci_collection_size, NULL) + PHP_FALIAS(ocicollmax, oci_collection_max, NULL) + PHP_FALIAS(ocicolltrim, oci_collection_trim, NULL) +#endif + {NULL,NULL,NULL} +}; + +static zend_function_entry php_oci_lob_class_functions[] = { + PHP_FALIAS(load, oci_lob_load, NULL) + PHP_FALIAS(tell, oci_lob_tell, NULL) + PHP_FALIAS(truncate, oci_lob_truncate, NULL) + PHP_FALIAS(erase, oci_lob_erase, NULL) + PHP_FALIAS(flush, oci_lob_flush, NULL) + PHP_FALIAS(setbuffering,ocisetbufferinglob, NULL) + PHP_FALIAS(getbuffering,ocigetbufferinglob, NULL) + PHP_FALIAS(rewind, oci_lob_rewind, NULL) + PHP_FALIAS(read, oci_lob_read, NULL) + PHP_FALIAS(eof, oci_lob_eof, NULL) + PHP_FALIAS(seek, oci_lob_seek, NULL) + PHP_FALIAS(write, oci_lob_write, NULL) + PHP_FALIAS(append, oci_lob_append, NULL) + PHP_FALIAS(size, oci_lob_size, NULL) + PHP_FALIAS(writetofile, oci_lob_export, NULL) +#ifdef HAVE_OCI8_TEMP_LOB + PHP_FALIAS(writetemporary, oci_lob_write_temporary, NULL) + PHP_FALIAS(close, oci_lob_close, NULL) +#endif + PHP_FALIAS(save, oci_lob_save, NULL) + PHP_FALIAS(savefile, oci_lob_import, NULL) + PHP_FALIAS(free, oci_free_descriptor, NULL) + {NULL,NULL,NULL} +}; + +#ifdef PHP_OCI8_HAVE_COLLECTIONS +static zend_function_entry php_oci_coll_class_functions[] = { + PHP_FALIAS(append, oci_collection_append, NULL) + PHP_FALIAS(getelem, oci_collection_element_get, NULL) + PHP_FALIAS(assignelem, oci_collection_element_assign, NULL) + PHP_FALIAS(assign, oci_collection_assign, NULL) + PHP_FALIAS(size, oci_collection_size, NULL) + PHP_FALIAS(max, oci_collection_max, NULL) + PHP_FALIAS(trim, oci_collection_trim, NULL) + PHP_FALIAS(free, oci_free_collection, NULL) + {NULL,NULL,NULL} +}; +#endif + +zend_module_entry oci8_module_entry = { + STANDARD_MODULE_HEADER, + "oci8", /* extension name */ + php_oci_functions, /* extension function list */ + PHP_MINIT(oci), /* extension-wide startup function */ + PHP_MSHUTDOWN(oci), /* extension-wide shutdown function */ + PHP_RINIT(oci), /* per-request startup function */ + PHP_RSHUTDOWN(oci), /* per-request shutdown function */ + PHP_MINFO(oci), /* information function */ + NO_VERSION_YET, + STANDARD_MODULE_PROPERTIES +}; +/* }}} */ + +/* {{{ startup, shutdown and info functions +*/ +static void php_oci_init_globals(php_oci_globals *oci_globals_p TSRMLS_DC) +{ + OCI(shutdown) = 0; + OCI(in_call) = 0; + + CALL_OCI( + OCIEnvInit( + &OCI(pEnv), + OCI_DEFAULT, + 0, + NULL + ) + ); + + CALL_OCI( + OCIHandleAlloc( + OCI(pEnv), + (dvoid **)&OCI(pError), + OCI_HTYPE_ERROR, + 0, + NULL + ) + ); +} + +static int _sessions_pcleanup(zend_llist *session_list TSRMLS_DC) +{ + zend_llist_destroy(session_list); + + return 1; +} + +static int _session_pcleanup(oci_session *session TSRMLS_DC) +{ + _oci_close_session(session); + + return 1; +} + +static int _server_pcleanup(oci_server *server TSRMLS_DC) +{ + _oci_close_server(server); + + return 1; +} + +PHP_MINIT_FUNCTION(oci) +{ + zend_class_entry oci_lob_class_entry; +#ifdef PHP_OCI8_HAVE_COLLECTIONS + zend_class_entry oci_coll_class_entry; +#endif + +#ifdef HAVE_OCI8_SHARED_MODE + +#ifdef PHP_OCI8_HAVE_COLLECTIONS +#define PHP_OCI_INIT_MODE_TMP OCI_SHARED | OCI_OBJECT +#else +#define PHP_OCI_INIT_MODE_TMP OCI_SHARED +#endif + +#else + +#ifdef PHP_OCI8_HAVE_COLLECTIONS +#define PHP_OCI_INIT_MODE_TMP OCI_DEFAULT | OCI_OBJECT +#else +#define PHP_OCI_INIT_MODE_TMP OCI_DEFAULT +#endif + +#endif + +#ifdef ZTS +#define PHP_OCI_INIT_MODE PHP_OCI_INIT_MODE_TMP | OCI_THREADED +#else +#define PHP_OCI_INIT_MODE PHP_OCI_INIT_MODE_TMP +#endif + + mutex_alloc(mx_lock); + + persistent_servers = malloc(sizeof(TsHashTable)); + persistent_sessions = malloc(sizeof(TsHashTable)); + zend_ts_hash_init(persistent_servers, 13, NULL, (dtor_func_t) _server_pcleanup, 1); + zend_ts_hash_init(persistent_sessions, 13, NULL, (dtor_func_t) _sessions_pcleanup, 1); + + OCIInitialize(PHP_OCI_INIT_MODE, NULL, NULL, NULL, NULL); + +#ifdef ZTS + ts_allocate_id(&oci_globals_id, sizeof(php_oci_globals), (ts_allocate_ctor) php_oci_init_globals, NULL); +#else + php_oci_init_globals(&oci_globals TSRMLS_CC); +#endif + + le_stmt = zend_register_list_destructors_ex(_oci_stmt_list_dtor, NULL, "oci8 statement", module_number); + le_conn = zend_register_list_destructors_ex(php_oci_free_conn_list, NULL, "oci8 connection", module_number); + le_desc = zend_register_list_destructors_ex(_oci_descriptor_list_dtor, NULL, "oci8 descriptor", module_number); +#ifdef PHP_OCI8_HAVE_COLLECTIONS + le_coll = zend_register_list_destructors_ex(_oci_coll_list_dtor, NULL, "oci8 collection", module_number); +#endif + le_server = zend_register_list_destructors_ex(_oci_server_list_dtor, NULL, "oci8 server", module_number); + le_session = zend_register_list_destructors_ex(_oci_session_list_dtor, NULL, "oci8 session", module_number); + + INIT_CLASS_ENTRY(oci_lob_class_entry, "OCI-Lob", php_oci_lob_class_functions); +#ifdef PHP_OCI8_HAVE_COLLECTIONS + INIT_CLASS_ENTRY(oci_coll_class_entry, "OCI-Collection", php_oci_coll_class_functions); +#endif + + oci_lob_class_entry_ptr = zend_register_internal_class(&oci_lob_class_entry TSRMLS_CC); +#ifdef PHP_OCI8_HAVE_COLLECTIONS + oci_coll_class_entry_ptr = zend_register_internal_class(&oci_coll_class_entry TSRMLS_CC); +#endif + +/* thies@thieso.net 990203 i do not think that we will need all of them - just in here for completeness for now! */ + REGISTER_LONG_CONSTANT("OCI_DEFAULT",OCI_DEFAULT, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_DESCRIBE_ONLY",OCI_DESCRIBE_ONLY, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_COMMIT_ON_SUCCESS",OCI_COMMIT_ON_SUCCESS, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_EXACT_FETCH",OCI_EXACT_FETCH, CONST_CS | CONST_PERSISTENT); + +/* for $LOB->seek() */ + REGISTER_LONG_CONSTANT("OCI_SEEK_SET",OCI_SEEK_SET, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_SEEK_CUR",OCI_SEEK_CUR, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_SEEK_END",OCI_SEEK_END, CONST_CS | CONST_PERSISTENT); + +/* for $LOB->flush() */ + REGISTER_LONG_CONSTANT("OCI_LOB_BUFFER_FREE",OCI_LOB_BUFFER_FREE, CONST_CS | CONST_PERSISTENT); + +/* for OCIBindByName (real "oci" names + short "php" names*/ + REGISTER_LONG_CONSTANT("SQLT_BFILEE",SQLT_BFILEE, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("SQLT_CFILEE",SQLT_CFILEE, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("SQLT_CLOB",SQLT_CLOB, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("SQLT_BLOB",SQLT_BLOB, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("SQLT_RDD",SQLT_RDD, CONST_CS | CONST_PERSISTENT); + +#ifdef PHP_OCI8_HAVE_COLLECTIONS + REGISTER_LONG_CONSTANT("OCI_B_NTY",SQLT_NTY, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("SQLT_NTY",SQLT_NTY, CONST_CS | CONST_PERSISTENT); + REGISTER_STRING_CONSTANT("OCI_SYSDATE","SYSDATE",CONST_PERSISTENT); +#endif + + REGISTER_LONG_CONSTANT("OCI_B_BFILE",SQLT_BFILEE, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_B_CFILEE",SQLT_CFILEE, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_B_CLOB",SQLT_CLOB, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_B_BLOB",SQLT_BLOB, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_B_ROWID",SQLT_RDD, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_B_CURSOR",SQLT_RSET, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_B_BIN",SQLT_BIN, CONST_CS | CONST_PERSISTENT); + +/* for OCIFetchStatement */ + REGISTER_LONG_CONSTANT("OCI_FETCHSTATEMENT_BY_COLUMN", OCI_FETCHSTATEMENT_BY_COLUMN, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_FETCHSTATEMENT_BY_ROW", OCI_FETCHSTATEMENT_BY_ROW, CONST_CS | CONST_PERSISTENT); + +/* for OCIFetchInto & OCIResult */ + REGISTER_LONG_CONSTANT("OCI_ASSOC",OCI_ASSOC, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_NUM",OCI_NUM, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_BOTH",OCI_BOTH, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_RETURN_NULLS",OCI_RETURN_NULLS, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_RETURN_LOBS",OCI_RETURN_LOBS, CONST_CS | CONST_PERSISTENT); + +/* for OCINewDescriptor (real "oci" names + short "php" names*/ + REGISTER_LONG_CONSTANT("OCI_DTYPE_FILE",OCI_DTYPE_FILE, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_DTYPE_LOB",OCI_DTYPE_LOB, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_DTYPE_ROWID",OCI_DTYPE_ROWID, CONST_CS | CONST_PERSISTENT); + + REGISTER_LONG_CONSTANT("OCI_D_FILE",OCI_DTYPE_FILE, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_D_LOB",OCI_DTYPE_LOB, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_D_ROWID",OCI_DTYPE_ROWID, CONST_CS | CONST_PERSISTENT); + +/* for OCIWriteTemporaryLob */ +#ifdef HAVE_OCI8_TEMP_LOB + REGISTER_LONG_CONSTANT("OCI_TEMP_CLOB",OCI_TEMP_CLOB, CONST_CS | CONST_PERSISTENT); + REGISTER_LONG_CONSTANT("OCI_TEMP_BLOB",OCI_TEMP_BLOB, CONST_CS | CONST_PERSISTENT); +#endif + + return SUCCESS; +} + +/* ----------------------------------------------------------------- */ + +PHP_RINIT_FUNCTION(oci) +{ + OCI(debug_mode) = 0; /* start "fresh" */ +/* OCI(in_call) = 0; i don't think we want this! */ + + oci_debug("php_rinit_oci"); + + return SUCCESS; +} + +PHP_MSHUTDOWN_FUNCTION(oci) +{ + OCI(shutdown) = 1; + + oci_debug("START php_mshutdown_oci"); + + zend_ts_hash_destroy(persistent_sessions); + zend_ts_hash_destroy(persistent_servers); + + free(persistent_sessions); + free(persistent_servers); + + mutex_free(mx_lock); + + CALL_OCI( + OCIHandleFree( + (dvoid *) OCI(pEnv), + OCI_HTYPE_ENV + ) + ); + + oci_debug("END php_mshutdown_oci"); + + return SUCCESS; +} + +PHP_RSHUTDOWN_FUNCTION(oci) +{ + oci_debug("START php_rshutdown_oci"); + +#if 0 + /* XXX free all statements, rollback all outstanding transactions */ + + zend_ts_hash_apply(persistent_sessions, (apply_func_t) _session_cleanup TSRMLS_CC); + zend_ts_hash_apply(persistent_servers, (apply_func_t) _server_cleanup TSRMLS_CC); +#endif + + oci_debug("END php_rshutdown_oci"); + + return SUCCESS; +} + +PHP_MINFO_FUNCTION(oci) +{ + char buf[32]; + + php_info_print_table_start(); + php_info_print_table_row(2, "OCI8 Support", "enabled"); + php_info_print_table_row(2, "Revision", "$Revision: 1.257.2.6 $"); + + sprintf(buf, "%ld", num_persistent); + php_info_print_table_row(2, "Active Persistent Links", buf); + sprintf(buf, "%ld", num_links); + php_info_print_table_row(2, "Active Links", buf); + +#ifndef PHP_WIN32 + php_info_print_table_row(2, "Oracle Version", PHP_OCI8_VERSION ); + php_info_print_table_row(2, "Compile-time ORACLE_HOME", PHP_OCI8_DIR ); + php_info_print_table_row(2, "Libraries Used", PHP_OCI8_SHARED_LIBADD ); +#endif + +#ifdef HAVE_OCI8_TEMP_LOB + php_info_print_table_row(2, "Temporary Lob support", "enabled" ); +#else + php_info_print_table_row(2, "Temporary Lob support", "disabled" ); +#endif + +#ifdef PHP_OCI8_HAVE_COLLECTIONS + php_info_print_table_row(2, "Collections support", "enabled" ); +#else + php_info_print_table_row(2, "Collections support", "disabled" ); +#endif + + php_info_print_table_end(); +} +/* }}} */ + +/* {{{ _oci_define_hash_dtor() +*/ +static void _oci_define_hash_dtor(void *data) +{ + oci_define *define = (oci_define *) data; + + oci_debug("_oci_define_hash_dtor: %s",define->name); + + zval_ptr_dtor(&define->zval); + + if (define->name) { + efree(define->name); + define->name = 0; + } +} +/* }}} */ + +/* {{{ _oci_desc_flush_hash_dtor() + */ +static void _oci_desc_flush_hash_dtor(void *data) +{ + oci_descriptor *descr = *(oci_descriptor **)data; + TSRMLS_FETCH(); + + if (descr->buffering == 2 && (descr->type == OCI_DTYPE_LOB || descr->type == OCI_DTYPE_FILE)) { + oci_lob_flush(descr,OCI_LOB_BUFFER_FREE TSRMLS_CC); + descr->buffering = 1; + } +} +/* }}} */ + +/* {{{ _oci_bind_hash_dtor() +*/ +static void _oci_bind_hash_dtor(void *data) +{ + oci_bind *bind = (oci_bind *) data; + + oci_debug("_oci_bind_hash_dtor:"); + + zval_ptr_dtor(&(bind->zval)); +} +/* }}} */ + +/* {{{ _oci_bind_pre_exec() +*/ +static int _oci_bind_pre_exec(void *data TSRMLS_DC) +{ + oci_bind *bind = (oci_bind *) data; + + /* reset all bind stuff to a normal state..-. */ + + bind->indicator = 0; + + return 0; +} +/* }}} */ + +/* {{{ _oci_bind_post_exec() +*/ +static int _oci_bind_post_exec(void *data TSRMLS_DC) +{ + oci_bind *bind = (oci_bind *) data; + + if (bind->indicator == -1) { /* NULL */ + zval *val = bind->zval; + if (Z_TYPE_P(val) == IS_STRING && (Z_STRVAL_P(val) != empty_string)) { + *Z_STRVAL_P(val) = '\0'; /* XXX avoid warning in debug mode */ + } + zval_dtor(val); + ZVAL_NULL(val); + } else if (Z_TYPE_P(bind->zval) == IS_STRING && (Z_STRVAL_P(bind->zval) != empty_string)) { + Z_STRVAL_P(bind->zval) = erealloc(Z_STRVAL_P(bind->zval), Z_STRLEN_P(bind->zval)+1); + Z_STRVAL_P(bind->zval)[ Z_STRLEN_P(bind->zval) ] = '\0'; + } + + + return 0; +} +/* }}} */ + +/* {{{ _oci_column_hash_dtor() +*/ +static void _oci_column_hash_dtor(void *data) +{ + oci_out_column *column = (oci_out_column *) data; + TSRMLS_FETCH(); + + oci_debug("START _oci_column_hash_dtor: %s",column->name); + + if (column->stmtid) { + zend_list_delete(column->stmtid); + } + + if (column->is_descr) { + zend_list_delete(column->descid); + } else { + if (column->data) { + efree(column->data); + } + } + + oci_debug("END _oci_column_hash_dtor: %s",column->name); + + if (column->name) { + efree(column->name); + } +} +/* }}} */ + +/* {{{ _oci_stmt_list_dtor() +*/ +static void _oci_stmt_list_dtor(zend_rsrc_list_entry *rsrc TSRMLS_DC) +{ + oci_statement *statement = (oci_statement *)rsrc->ptr; + oci_debug("START _oci_stmt_list_dtor: id=%d last_query=\"%s\"",statement->id,SAFE_STRING(statement->last_query)); + + if (statement->pStmt) { + CALL_OCI( + OCIHandleFree( + statement->pStmt, + OCI_HTYPE_STMT + ) + ); + + statement->pStmt = 0; + } + + if (statement->pError) { + CALL_OCI( + OCIHandleFree( + statement->pError, + OCI_HTYPE_ERROR + ) + ); + + statement->pError = 0; + } + + if (statement->last_query) { + efree(statement->last_query); + } + + if (statement->columns) { + zend_hash_destroy(statement->columns); + efree(statement->columns); + } + + if (statement->binds) { + zend_hash_destroy(statement->binds); + efree(statement->binds); + } + + if (statement->defines) { + zend_hash_destroy(statement->defines); + efree(statement->defines); + } + + oci_debug("END _oci_stmt_list_dtor: id=%d",statement->id); + + efree(statement); +} +/* }}} */ + +/* {{{ _oci_conn_list_dtor() +*/ +static void _oci_conn_list_dtor(oci_connection *connection TSRMLS_DC) +{ + /* + as the connection is "only" a in memory service context we do not disconnect from oracle. + */ + + oci_debug("START _oci_conn_list_dtor: id=%d",connection->id); + + if (connection->pServiceContext) { + + if (connection->needs_commit) { + oci_debug("OCITransRollback"); + CALL_OCI_RETURN(connection->error, + OCITransRollback( + connection->pServiceContext, + connection->pError, + (ub4)0 + ) + ); + + if (connection->error) { + oci_error(connection->pError, "failed to rollback outstanding transactions!", connection->error); + } + connection->needs_commit = 0; + } else { + oci_debug("nothing to do.."); + } + + CALL_OCI( + OCIHandleFree( + (dvoid *) connection->pServiceContext, + (ub4) OCI_HTYPE_SVCCTX + ) + ); + } + + if (connection->session && connection->session->exclusive) { + /* close associated session when destructed */ + zend_list_delete(connection->session->num); + } + + if (connection->descriptors) { + zend_hash_destroy(connection->descriptors); + efree(connection->descriptors); + } + + if (connection->pError) { + CALL_OCI( + OCIHandleFree( + (dvoid *) connection->pError, + (ub4) OCI_HTYPE_ERROR + ) + ); + } + + oci_debug("END _oci_conn_list_dtor: id=%d",connection->id); + + efree(connection); +} +/* }}} */ + +/* {{{ php_oci_free_conn_list + */ +static void php_oci_free_conn_list(zend_rsrc_list_entry *rsrc TSRMLS_DC) +{ + oci_connection *conn = (oci_connection *)rsrc->ptr; + _oci_conn_list_dtor(conn TSRMLS_CC); +} +/* }}} */ + +#ifdef PHP_OCI8_HAVE_COLLECTIONS + +/* {{{ _oci_coll_list_dtor() + */ +static void _oci_coll_list_dtor(zend_rsrc_list_entry *rsrc TSRMLS_DC) +{ + oci_collection *coll = (oci_collection *)rsrc->ptr; + oci_debug("START _oci_coll_list_dtor: %d",coll->id); + + /* Note sure if we need to free the object. Have an + oracle TAR out on this one. + OCIDescriptorFree(descr->ocidescr, descr->type); */ + + oci_debug("END _oci_coll_list_dtor: %d",coll->id); + + efree(coll); +} +/* }}} */ +#endif + +/* {{{ _oci_descriptor_list_dtor() + */ +static void _oci_descriptor_list_dtor(zend_rsrc_list_entry *rsrc TSRMLS_DC) +{ + oci_descriptor *descr = (oci_descriptor *)rsrc->ptr; + oci_debug("START _oci_descriptor_list_dtor: %d",descr->id); + + /* flushing Lobs & Files with buffering enabled */ + if ((descr->type == OCI_DTYPE_FILE || descr->type == OCI_DTYPE_LOB) && descr->buffering == 2) { + oci_debug("descriptor #%d needs to be flushed. flushing..",descr->id); + oci_lob_flush(descr,OCI_LOB_BUFFER_FREE TSRMLS_CC); + } + + CALL_OCI( + OCIDescriptorFree( + descr->ocidescr, + descr->type + ) + ); + + oci_debug("END _oci_descriptor_list_dtor: %d",descr->id); + + efree(descr); +} +/* }}} */ + +/* {{{ _oci_server_list_dtor() + */ +static void _oci_server_list_dtor(zend_rsrc_list_entry *rsrc TSRMLS_DC) +{ +#if 0 + oci_server *server = (oci_server *)rsrc->ptr; + if (server->persistent) + return; + + _oci_close_server(server); +#endif +} +/* }}} */ + +/* {{{ _oci_session_list_dtor() + */ +static void _oci_session_list_dtor(zend_rsrc_list_entry *rsrc TSRMLS_DC) +{ + oci_session *session = (oci_session *)rsrc->ptr; + if (session->persistent) { + /* clear thread assignment */ + session->thread = 0; + return; + } + + _oci_close_session(session); +} +/* }}} */ + +/* {{{ oci_handle_error + */ +static ub4 oci_handle_error(oci_connection *connection, ub4 errcode) +{ + switch (errcode) { + case 1013: /* user requested cancel of current operation */ + zend_bailout(); + break; + + case 22: /* ORA-00022 Invalid session id */ + case 1012: /* ORA-01012: */ + case 3113: /* ORA-03113: end-of-file on communication channel */ + case 604: + case 1041: + connection->is_open = 0; + connection->session->is_open = 0; + connection->session->server->is_open = 0; + return 1; /* fatal error */ + } + + return 0; /* no fatal error */ +} +/* }}} */ + +/* {{{ oci_error() +*/ +static ub4 oci_error(OCIError *err_p, char *what, sword status) +{ + text errbuf[512]; + sb4 errcode = 0; + TSRMLS_FETCH(); + + switch (status) { + case OCI_SUCCESS: + break; + case OCI_SUCCESS_WITH_INFO: + CALL_OCI( + OCIErrorGet( + err_p, + (ub4)1, + NULL, + &errcode, + errbuf, + (ub4)sizeof(errbuf), + (ub4)OCI_HTYPE_ERROR + ) + ); + php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s: OCI_SUCCESS_WITH_INFO: %s", what, errbuf); + break; + case OCI_NEED_DATA: + php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s: OCI_NEED_DATA", what); + break; + case OCI_NO_DATA: + php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s: OCI_NO_DATA", what); + break; + case OCI_ERROR: { + CALL_OCI( + OCIErrorGet( + err_p, + (ub4)1, + NULL, + &errcode, + errbuf, + (ub4)sizeof(errbuf), + (ub4)OCI_HTYPE_ERROR + ) + ); + php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s: %s", what, errbuf); + break; + } + case OCI_INVALID_HANDLE: + php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s: OCI_INVALID_HANDLE", what); + break; + case OCI_STILL_EXECUTING: + php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s: OCI_STILL_EXECUTING", what); + break; + case OCI_CONTINUE: + php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s: OCI_CONTINUE", what); + break; + default: + break; + } + return errcode; +} +/* }}} */ + +/* {{{ oci_ping() +*/ +static int oci_ping(oci_server *server) +{ + char version[256]; + TSRMLS_FETCH(); + + CALL_OCI_RETURN(OCI(error), + OCIServerVersion( + server->pServer, + OCI(pError), + (text*)version, + sizeof(version), + OCI_HTYPE_SERVER + ) + ); + + if (OCI(error) == OCI_SUCCESS) { + return 1; + } + + oci_error(OCI(pError), "oci_ping", OCI(error)); + + return 0; +} +/* }}} */ + +/************************* INTERNAL FUNCTIONS *************************/ + +/* {{{ oci_debug() +*/ +static void oci_debug(const char *format, ...) +{ + TSRMLS_FETCH(); + + if (OCI(debug_mode)) { + char buffer[1024]; + va_list args; + + va_start(args, format); + vsnprintf(buffer, sizeof(buffer)-1, format, args); + va_end(args); + buffer[sizeof(buffer)-1] = '\0'; + php_printf("OCIDebug: %s<br />\n", buffer); + } +} +/* }}} */ + +/* {{{ oci_get_conn() +*/ +static oci_connection *oci_get_conn(zval **conn TSRMLS_DC) +{ + oci_connection *connection; + + connection = (oci_connection *) zend_fetch_resource(conn TSRMLS_CC, -1, "OCI8-Connection", NULL, 1, le_conn); + + if (connection && connection->is_open) { + return connection; + } else { + return (oci_connection *) NULL; + } +} +/* }}} */ + +/* {{{ oci_get_stmt() +*/ +static oci_statement *oci_get_stmt(zval **stmt TSRMLS_DC) +{ + oci_statement *statement; + + statement = (oci_statement *) zend_fetch_resource(stmt TSRMLS_CC, -1, "OCI8-Statement", NULL, 1, le_stmt); + + if (statement && statement->conn->is_open) { + return statement; + } else { + return (oci_statement *) NULL; + } +} +/* }}} */ + +/* {{{ oci_get_desc() +*/ +static oci_descriptor *oci_get_desc(int ind TSRMLS_DC) +{ + oci_descriptor *descriptor; + int actual_resource_type; + + descriptor = (oci_descriptor *) zend_list_find(ind, &actual_resource_type); + + if (descriptor && (actual_resource_type == le_desc)) { + return descriptor; + } else { + return (oci_descriptor *) NULL; + } +} +/* }}} */ + +/* {{{ oci_get_col() +*/ +static oci_out_column *oci_get_col(oci_statement *statement, int col, zval **value) +{ + oci_out_column *outcol = NULL; + int i; + TSRMLS_FETCH(); + + if (statement->columns == 0) { /* we release the columns at the end of a fetch */ + return NULL; + } + + if (value) { + if (Z_TYPE_PP(value) == IS_STRING) { + for (i = 0; i < statement->ncolumns; i++) { + outcol = oci_get_col(statement, i + 1, 0); + if (outcol == NULL) { + continue; + } else if (((int) outcol->name_len == Z_STRLEN_PP(value)) && + (!strncmp(outcol->name, Z_STRVAL_PP(value), Z_STRLEN_PP(value))) + ) { + return outcol; + } + } + } else { + convert_to_long_ex(value); + return oci_get_col(statement,Z_LVAL_PP(value),0); + } + } else if (col != -1) { + if (zend_hash_index_find(statement->columns, col, (void **)&outcol) == FAILURE) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid column %d", col); + return NULL; + } + return outcol; + } + + return NULL; +} +/* }}} */ + +/* {{{ oci_new_desc() +*/ +static oci_descriptor *oci_new_desc(int type, oci_connection *connection) +{ + oci_descriptor *descr; + TSRMLS_FETCH(); + + descr = emalloc(sizeof(oci_descriptor)); + + descr->type = type; + + switch (descr->type) { + case OCI_DTYPE_FILE: + case OCI_DTYPE_LOB: + case OCI_DTYPE_ROWID: + break; + + default: + php_error_docref(NULL TSRMLS_CC, E_WARNING, "Unknown descriptor type %d.", descr->type); + efree(descr); + return 0; + } + + CALL_OCI_RETURN(OCI(error), + OCIDescriptorAlloc( + connection->session->pEnv, + (dvoid*)&(descr->ocidescr), + descr->type, + (size_t) 0, + (dvoid **) 0 + ) + ); + + if (OCI(error)) { + ub4 error; + error = oci_error(OCI(pError),"OCIDescriptorAlloc %d",OCI(error)); + oci_handle_error(connection, error); + efree(descr); + return 0; + } + + descr->id = zend_list_insert(descr,le_desc); + descr->conn = connection; + descr->lob_current_position = 0; + descr->lob_size = -1; /* we should set it to -1 to know, that it's just not initialized */ + descr->buffering = 0; /* buffering is off by default */ + zend_list_addref(connection->id); + + if (descr->type == OCI_DTYPE_LOB || descr->type == OCI_DTYPE_FILE) { + /* add Lobs & Files to hash. we'll flush them ate the end */ + if (!connection->descriptors) { + ALLOC_HASHTABLE(connection->descriptors); + zend_hash_init(connection->descriptors, 13, NULL, _oci_desc_flush_hash_dtor, 0); + } + + zend_hash_next_index_insert(connection->descriptors,&descr,sizeof(oci_descriptor *),NULL); + } + oci_debug("oci_new_desc %d",descr->id); + + return descr; +} +/* }}} */ + +#ifdef PHP_OCI8_HAVE_COLLECTIONS + +/* {{{ _oci_get_ocicoll() +*/ +static int _oci_get_ocicoll(zval *id,oci_collection **collection TSRMLS_DC) +{ + zval **coll; + + if (zend_hash_find(Z_OBJPROP_P(id), "collection", sizeof("collection"), (void **)&coll) == FAILURE) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "cannot find collection"); + return 0; + } + if ((*collection = oci_get_coll(Z_LVAL_PP(coll) TSRMLS_CC)) == NULL) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "collection not found"); + return 0; + } + + return Z_LVAL_PP(coll); +} +/* }}} */ + +#endif + +/* {{{ _oci_get_ocidesc() +*/ +static int _oci_get_ocidesc(zval *id,oci_descriptor **descriptor TSRMLS_DC) +{ + zval **desc; + + if (zend_hash_find(Z_OBJPROP_P(id), "descriptor", sizeof("descriptor"), (void **)&desc) == FAILURE) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "cannot find descriptor"); + return 0; + } + + if ((*descriptor = oci_get_desc(Z_LVAL_PP(desc) TSRMLS_CC)) == NULL) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "descriptor not found"); + return 0; + } + + return Z_LVAL_PP(desc); +} +/* }}} */ + +/* {{{ _oci_make_zval() +*/ +static int _oci_make_zval(zval *value,oci_statement *statement,oci_out_column *column, char *func, int mode TSRMLS_DC) +{ + oci_descriptor *descr; + ub4 loblen; + int size; + char *buffer; + + if (column->indicator || column->retcode) + if ((column->indicator != -1) && (column->retcode != 1405)) + oci_debug("_oci_make_zval: %16s,retlen = %4d,retlen4 = %d,storage_size4 = %4d,indicator %4d, retcode = %4d", + column->name,column->retlen,column->retlen4,column->storage_size4,column->indicator,column->retcode); + + if ((!statement->has_data) || (column->indicator == -1)) { /* column is NULL or statment has no current data */ + ZVAL_NULL(value); + return 0; + } + + if (column->is_cursor) { /* REFCURSOR -> simply return the statement id */ + ZVAL_RESOURCE(value,column->stmtid); + zend_list_addref(column->stmtid); + } else if (column->is_descr) { + if ((column->data_type != SQLT_RDD) && (mode & OCI_RETURN_LOBS)) { + /* OCI_RETURN_LOBS means that we want the content of the LOB back instead of the locator */ + + descr = oci_get_desc(column->descid TSRMLS_CC); + if (!descr) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "unable to find my descriptor %p",column->data); + return -1; + } + + if (oci_loadlob(statement->conn,descr,&buffer,&loblen)) { + ZVAL_FALSE(value); + } else { + ZVAL_STRINGL(value,buffer,loblen,0); + } + } else { + /* return the locator */ + object_init_ex(value, oci_lob_class_entry_ptr); + add_property_resource(value, "descriptor", column->descid); + zend_list_addref(column->descid); + } + } else { + switch (column->retcode) { + case 0: + /* intact value */ + if (column->piecewise) { + size = column->retlen4; + } else { + size = column->retlen; + } + break; + + default: + /* XXX we SHOULD maybe have a different behaviour for unknown results! */ + ZVAL_FALSE(value); + return 0; + } + + ZVAL_STRINGL(value,column->data,size,1); + } + + return 0; +} +/* }}} */ + +/* {{{ oci_setprefetch() +*/ +static int oci_setprefetch(oci_statement *statement,int size) +{ + ub4 prefetch; + sword error; + TSRMLS_FETCH(); + + prefetch = size * 1024; + + CALL_OCI_RETURN(error, + OCIAttrSet( + statement->pStmt, + OCI_HTYPE_STMT, + &prefetch, + 0, + OCI_ATTR_PREFETCH_MEMORY, + statement->pError + ) + ); + + statement->error = oci_error(statement->pError, "OCIAttrSet OCI_ATTR_PREFETCH_MEMORY", error); + + if (statement->error) { + oci_handle_error(statement->conn, statement->error); + } + + prefetch = size; + CALL_OCI_RETURN(error, + OCIAttrSet( + statement->pStmt, + OCI_HTYPE_STMT, + &prefetch, + 0, + OCI_ATTR_PREFETCH_ROWS, + statement->pError + ) + ); + + statement->error = oci_error(statement->pError, "OCIAttrSet OCI_ATTR_PREFETCH_MEMORY", error); + if (statement->error) { + oci_handle_error(statement->conn, statement->error); + } + + return 1; +} +/* }}} */ + +/* {{{ oci_parse() +*/ +static oci_statement *oci_parse(oci_connection *connection, char *query, int len) +{ + oci_statement *statement; + sword error; + TSRMLS_FETCH(); + + statement = ecalloc(1,sizeof(oci_statement)); + + CALL_OCI( + OCIHandleAlloc( + connection->session->pEnv, + (dvoid **)&statement->pStmt, + OCI_HTYPE_STMT, + 0, + NULL + ) + ); + + CALL_OCI( + OCIHandleAlloc( + connection->session->pEnv, + (dvoid **)&statement->pError, + OCI_HTYPE_ERROR, + 0, + NULL + ) + ); + + if (len > 0) { + CALL_OCI_RETURN(error, + OCIStmtPrepare( + statement->pStmt, + connection->pError, + (text*)query, + len, + OCI_NTV_SYNTAX, + OCI_DEFAULT + ) + ); + + connection->error = oci_error(connection->pError, "OCIParse", error); + if (connection->error) { + CALL_OCI( + OCIHandleFree( + statement->pStmt, + OCI_HTYPE_STMT + ) + ); + + CALL_OCI( + OCIHandleFree( + statement->pError, + OCI_HTYPE_ERROR + ) + ); + + efree(statement); + oci_handle_error(connection, connection->error); + return 0; + } + } + + if (query) { + statement->last_query = estrdup(query); + } + + statement->conn = connection; + statement->has_data = 0; + + statement->id = zend_list_insert(statement,le_stmt); + + oci_debug("oci_parse \"%s\" id=%d conn=%d", + SAFE_STRING(query), + statement->id, + statement->conn->id); + + zend_list_addref(statement->conn->id); + + return statement; +} +/* }}} */ + +/* {{{ oci_execute() +*/ +static int oci_execute(oci_statement *statement, char *func,ub4 mode) +{ + oci_out_column *outcol; + oci_out_column column; + OCIParam *param = 0; + text *colname; + ub4 counter; + ub2 define_type; + ub4 iters; + ub4 colcount; + ub2 dynamic; + int dtype; + dvoid *buf; + oci_descriptor *descr; + sword error; + TSRMLS_FETCH(); + + if (!statement->stmttype) { + CALL_OCI_RETURN(error, + OCIAttrGet( + (dvoid *)statement->pStmt, + OCI_HTYPE_STMT, + (ub2 *)&statement->stmttype, + (ub4 *)0, + OCI_ATTR_STMT_TYPE, + statement->pError + ) + ); + + statement->error = oci_error(statement->pError, "OCIAttrGet OCI_HTYPE_STMT/OCI_ATTR_STMT_TYPE", error); + + if (statement->error) { + oci_handle_error(statement->conn, statement->error); + return 0; + } + } + + if (statement->stmttype == OCI_STMT_SELECT) { + iters = 0; + } else { + iters = 1; + } + + if (statement->last_query) { + /* if we execute refcursors we don't have a query and + we don't want to execute!!! */ + + if (statement->binds) { + zend_hash_apply(statement->binds, (apply_func_t) _oci_bind_pre_exec TSRMLS_CC); + } + + CALL_OCI_RETURN(error, + OCIStmtExecute( + statement->conn->pServiceContext, + statement->pStmt, + statement->pError, + iters, + 0, + NULL, + NULL, + mode + ) + ); + + statement->error = oci_error(statement->pError, "OCIStmtExecute", error); + + if (statement->binds) { + zend_hash_apply(statement->binds, (apply_func_t) _oci_bind_post_exec TSRMLS_CC); + } + + oci_handle_error(statement->conn, statement->error); + + if (statement->error) { + return 0; + } + + if (mode & OCI_COMMIT_ON_SUCCESS) { + statement->conn->needs_commit = 0; + } else { + statement->conn->needs_commit = 1; + } + } + + if ((statement->stmttype == OCI_STMT_SELECT) && (statement->executed == 0)) { + /* we only need to do the define step is this very statement is executed the first time! */ + statement->executed = 1; + + ALLOC_HASHTABLE(statement->columns); + zend_hash_init(statement->columns, 13, NULL, _oci_column_hash_dtor, 0); + + counter = 1; + + CALL_OCI_RETURN(error, + OCIAttrGet( + (dvoid *)statement->pStmt, + OCI_HTYPE_STMT, + (dvoid *)&colcount, + (ub4 *)0, + OCI_ATTR_PARAM_COUNT, + statement->pError + ) + ); + + statement->error = oci_error(statement->pError, "OCIAttrGet OCI_HTYPE_STMT/OCI_ATTR_PARAM_COUNT", error); + if (statement->error) { + oci_handle_error(statement->conn, statement->error); + return 0; /* XXX we loose memory!!! */ + } + + statement->ncolumns = colcount; + + for (counter = 1; counter <= colcount; counter++) { + memset(&column,0,sizeof(oci_out_column)); + + if (zend_hash_index_update(statement->columns, counter, &column, + sizeof(oci_out_column), (void**) &outcol) == FAILURE) { + efree(statement->columns); + /* out of memory */ + return 0; + } + + outcol->statement = statement; + + CALL_OCI_RETURN(error, + OCIParamGet( + (dvoid *)statement->pStmt, + OCI_HTYPE_STMT, + statement->pError, + (dvoid**)¶m, + counter + ) + ); + + statement->error = oci_error(statement->pError, "OCIParamGet OCI_HTYPE_STMT", error); + if (statement->error) { + oci_handle_error(statement->conn, statement->error); + return 0; /* XXX we loose memory!!! */ + } + + CALL_OCI_RETURN(error, + OCIAttrGet( + (dvoid *)param, + OCI_DTYPE_PARAM, + (dvoid *)&outcol->data_type, + (ub4 *)0, + OCI_ATTR_DATA_TYPE, + statement->pError + ) + ); + + statement->error = oci_error(statement->pError, "OCIAttrGet OCI_DTYPE_PARAM/OCI_ATTR_DATA_TYPE", error); + if (statement->error) { + oci_handle_error(statement->conn, statement->error); + return 0; /* XXX we loose memory!!! */ + } + + CALL_OCI_RETURN(error, + OCIAttrGet( + (dvoid *)param, + OCI_DTYPE_PARAM, + (dvoid *)&outcol->data_size, + (dvoid *)0, + OCI_ATTR_DATA_SIZE, + statement->pError + ) + ); + + statement->error = oci_error(statement->pError, "OCIAttrGet OCI_DTYPE_PARAM/OCI_ATTR_DATA_SIZE", error); + if (statement->error) { + oci_handle_error(statement->conn, statement->error); + return 0; /* XXX we loose memory!!! */ + } + + outcol->storage_size4 = outcol->data_size; + outcol->retlen = outcol->data_size; + + CALL_OCI_RETURN(error, + OCIAttrGet( + (dvoid *)param, + OCI_DTYPE_PARAM, + (dvoid *)&outcol->scale, + (dvoid *)0, + OCI_ATTR_SCALE, + statement->pError + ) + ); + + statement->error = oci_error(statement->pError, "OCIAttrGet OCI_DTYPE_PARAM/OCI_ATTR_SCALE", error); + if (statement->error) { + oci_handle_error(statement->conn, statement->error); + return 0; /* XXX we lose memory!!! */ + } + + CALL_OCI_RETURN(error, + OCIAttrGet( + (dvoid *)param, + OCI_DTYPE_PARAM, + (dvoid *)&outcol->precision, + (dvoid *)0, + OCI_ATTR_PRECISION, + statement->pError + ) + ); + + statement->error = oci_error(statement->pError, "OCIAttrGet OCI_DTYPE_PARAM/OCI_ATTR_PRECISION", error); + if (statement->error) { + oci_handle_error(statement->conn, statement->error); + return 0; /* XXX we lose memory!!! */ + } + + CALL_OCI_RETURN(error, + OCIAttrGet( + (dvoid *)param, + OCI_DTYPE_PARAM, + (dvoid **)&colname, /* XXX this string is NOT zero terminated!!!! */ + (ub4 *)&outcol->name_len, + (ub4)OCI_ATTR_NAME, + statement->pError + ) + ); + + statement->error = oci_error(statement->pError, "OCIAttrGet OCI_DTYPE_PARAM/OCI_ATTR_NAME", error); + if (statement->error) { + oci_handle_error(statement->conn, statement->error); + return 0; /* XXX we loose memory!!! */ + } + + outcol->name = estrndup((char*) colname,outcol->name_len); + + /* find a user-setted define */ + if (statement->defines) { + zend_hash_find(statement->defines,outcol->name,outcol->name_len,(void **) &outcol->define); + } + + buf = 0; + switch (outcol->data_type) { + case SQLT_RSET: + outcol->pstmt = oci_parse(statement->conn,0,0); + outcol->stmtid = outcol->pstmt->id; + + define_type = SQLT_RSET; + outcol->is_cursor = 1; + outcol->storage_size4 = -1; + outcol->retlen = -1; + dynamic = OCI_DEFAULT; + buf = &(outcol->pstmt->pStmt); + break; + + case SQLT_RDD: /* ROWID */ + case SQLT_BLOB: /* binary LOB */ + case SQLT_CLOB: /* character LOB */ + case SQLT_BFILE: /* binary file LOB */ + define_type = outcol->data_type; + outcol->is_descr = 1; + outcol->storage_size4 = -1; + dynamic = OCI_DEFAULT; + + if (outcol->data_type == SQLT_BFILE) { + dtype = OCI_DTYPE_FILE; + } else if (outcol->data_type == SQLT_RDD ) { + dtype = OCI_DTYPE_ROWID; + } else { + dtype = OCI_DTYPE_LOB; + } + + descr = oci_new_desc(dtype,statement->conn); + if (!descr) { + /* need better error checking XXX */ + } + outcol->descid = descr->id; + buf = &(descr->ocidescr); + break; + + case SQLT_LNG: + case SQLT_LBI: + if (outcol->data_type == SQLT_LBI) { + define_type = SQLT_BIN; + } else { + define_type = SQLT_CHR; + } + outcol->storage_size4 = OCI_MAX_DATA_SIZE; + outcol->piecewise = 1; + dynamic = OCI_DYNAMIC_FETCH; + break; + + case SQLT_BIN: + default: + define_type = SQLT_CHR; + if ((outcol->data_type == SQLT_DAT) || (outcol->data_type == SQLT_NUM) +#ifdef SQLT_TIMESTAMP + || (outcol->data_type == SQLT_TIMESTAMP) +#endif +#ifdef SQLT_TIMESTAMP_TZ + || (outcol->data_type == SQLT_TIMESTAMP_TZ) +#endif + ) { + outcol->storage_size4 = 512; /* XXX this should fit "most" NLS date-formats and Numbers */ + } else { + outcol->storage_size4++; /* add one for string terminator */ + } + if (outcol->data_type == SQLT_BIN) { + outcol->storage_size4 *= 3; + } + dynamic = OCI_DEFAULT; + buf = outcol->data = (text *) emalloc(outcol->storage_size4); + break; + } + + if (dynamic == OCI_DYNAMIC_FETCH) { + CALL_OCI_RETURN(error, + OCIDefineByPos( + statement->pStmt, /* IN/OUT handle to the requested SQL query */ + (OCIDefine **)&outcol->pDefine, /* IN/OUT pointer to a pointer to a define handle */ + statement->pError, /* IN/OUT An error handle */ + counter, /* IN position in the select list */ + (dvoid *)NULL, /* IN/OUT pointer to a buffer */ + outcol->storage_size4, /* IN The size of each valuep buffer in bytes */ + define_type, /* IN The data type */ + (dvoid *)&outcol->indicator, /* IN pointer to an indicator variable or arr */ + (ub2 *)NULL, /* IN/OUT Pointer to array of length of data fetched */ + (ub2 *)NULL, /* OUT Pointer to array of column-level return codes */ + OCI_DYNAMIC_FETCH /* IN mode (OCI_DEFAULT, OCI_DYNAMIC_FETCH) */ + ) + ); + + statement->error = oci_error(statement->pError, "OCIDefineByPos", error); + } else { + CALL_OCI_RETURN(error, + OCIDefineByPos( + statement->pStmt, /* IN/OUT handle to the requested SQL query */ + (OCIDefine **)&outcol->pDefine, /* IN/OUT pointer to a pointer to a define handle */ + statement->pError, /* IN/OUT An error handle */ + counter, /* IN position in the select list */ + (dvoid *)buf, /* IN/OUT pointer to a buffer */ + outcol->storage_size4, /* IN The size of each valuep buffer in bytes */ + define_type, /* IN The data type */ + (dvoid *)&outcol->indicator, /* IN pointer to an indicator variable or arr */ + (ub2 *)&outcol->retlen, /* IN/OUT Pointer to array of length of data fetched */ + (ub2 *)&outcol->retcode, /* OUT Pointer to array of column-level return codes */ + OCI_DEFAULT /* IN mode (OCI_DEFAULT, OCI_DYNAMIC_FETCH) */ + ) + ); + + statement->error = oci_error(statement->pError, "OCIDefineByPos", error); + } + if (statement->error) { + oci_handle_error(statement->conn, statement->error); + return 0; /* XXX we loose memory!!! */ + } + } + } + + return 1; +} +/* }}} */ + +/* {{{ oci_fetch() +*/ +static int _oci_column_pre_fetch(void *data TSRMLS_DC) +{ + oci_out_column *col = (oci_out_column *) data; + + if (col->piecewise) { + col->retlen4 = 0; + } + + return 0; +} + +static int oci_fetch(oci_statement *statement, ub4 nrows, char *func TSRMLS_DC) +{ + int i; + oci_out_column *column; + + if (statement->columns) { + zend_hash_apply(statement->columns, (apply_func_t) _oci_column_pre_fetch TSRMLS_CC); + } + + + CALL_OCI_RETURN(statement->error, + OCIStmtFetch( + statement->pStmt, + statement->pError, + nrows, + OCI_FETCH_NEXT, + OCI_DEFAULT + ) + ); + + if ((statement->error == OCI_NO_DATA) || (nrows == 0)) { + if (statement->last_query == 0) { + /* reset define-list for refcursors */ + if (statement->columns) { + zend_hash_destroy(statement->columns); + efree(statement->columns); + statement->columns = 0; + statement->ncolumns = 0; + } + statement->executed = 0; + } + + statement->error = 0; /* OCI_NO_DATA is NO error for us!!! */ + statement->has_data = 0; + + return 0; + } + + while (statement->error == OCI_NEED_DATA) { + for (i = 0; i < statement->ncolumns; i++) { + column = oci_get_col(statement, i + 1, 0); + if (column->piecewise) { + if (!column->data) { + column->data = (text *) emalloc(OCI_PIECE_SIZE); + } else { + column->data = erealloc(column->data,column->retlen4 + OCI_PIECE_SIZE); + } + + column->cb_retlen = OCI_PIECE_SIZE; + + CALL_OCI( + OCIStmtSetPieceInfo( + (void *) column->pDefine, + OCI_HTYPE_DEFINE, + statement->pError, + ((char*)column->data) + column->retlen4, + &(column->cb_retlen), + OCI_NEXT_PIECE, + &column->indicator, + &column->retcode + ) + ); + } + } + + CALL_OCI_RETURN(statement->error, + OCIStmtFetch( + statement->pStmt, + statement->pError, + nrows, + OCI_FETCH_NEXT, + OCI_DEFAULT + ) + ); + + for (i = 0; i < statement->ncolumns; i++) { + column = oci_get_col(statement, i + 1, 0); + if (column->piecewise) { + column->retlen4 += column->cb_retlen; + } + } + } + + if (statement->error == OCI_SUCCESS_WITH_INFO || statement->error == OCI_SUCCESS) { + statement->has_data = 1; + + /* do the stuff needed for OCIDefineByName */ + for (i = 0; i < statement->ncolumns; i++) { + column = oci_get_col(statement, i + 1, 0); + if (column == NULL) { + continue; + } + + if (!column->define) { + continue; + } + + zval_dtor(column->define->zval); + _oci_make_zval(column->define->zval,statement,column,"OCIFetch",0 TSRMLS_CC); + } + + return 1; + } + + oci_error(statement->pError, func, statement->error); + oci_handle_error(statement->conn, statement->error); + + statement->has_data = 0; + + return 0; +} +/* }}} */ + +/* {{{ oci_lobgetlen() +*/ +static int oci_lobgetlen(oci_connection *connection, oci_descriptor *mydescr, ub4 *loblen) +{ + TSRMLS_FETCH(); + + *loblen = 0; + + /* do we need to ask oracle about LOB's length, if we do already know it? I think no. */ + if (mydescr->lob_size >= 0) { + *loblen = mydescr->lob_size; + } else { + if (mydescr->type == OCI_DTYPE_FILE) { + CALL_OCI_RETURN(connection->error, + OCILobFileOpen( + connection->pServiceContext, + connection->pError, + mydescr->ocidescr, + OCI_FILE_READONLY + ) + ); + if (connection->error) { + oci_error(connection->pError, "OCILobFileOpen",connection->error); + oci_handle_error(connection, connection->error); + return -1; + } + } + + CALL_OCI_RETURN(connection->error, + OCILobGetLength( + connection->pServiceContext, + connection->pError, + mydescr->ocidescr, + loblen + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobGetLength",connection->error); + oci_handle_error(connection, connection->error); + return -1; + } + mydescr->lob_size = *loblen; + + if (mydescr->type == OCI_DTYPE_FILE) { + CALL_OCI_RETURN(connection->error, + OCILobFileClose( + connection->pServiceContext, + connection->pError, + mydescr->ocidescr + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobFileClose", connection->error); + oci_handle_error(connection, connection->error); + return -1; + } + } + } + + oci_debug("oci_lobgetlen: len=%d",*loblen); + + return 0; +} +/* }}} */ + +/* {{{ oci_loadlob() +*/ +#define LOBREADSIZE 1048576l /* 1MB */ +static int oci_loadlob(oci_connection *connection, oci_descriptor *mydescr, char **buffer, ub4 *loblen) +{ + ub4 siz = 0; + ub4 readlen = 0; + char *buf; + TSRMLS_FETCH(); + + *loblen = 0; + + if (mydescr->type == OCI_DTYPE_FILE) { + CALL_OCI_RETURN(connection->error, + OCILobFileOpen( + connection->pServiceContext, + connection->pError, + mydescr->ocidescr, + OCI_FILE_READONLY + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobFileOpen",connection->error); + oci_handle_error(connection, connection->error); + return -1; + } + } + + CALL_OCI_RETURN(connection->error, + OCILobGetLength( + connection->pServiceContext, + connection->pError, + mydescr->ocidescr, + &readlen + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobGetLength",connection->error); + oci_handle_error(connection, connection->error); + return -1; + } + + buf = emalloc(readlen + 1); + + while (readlen > 0) { /* thies loop should not be entered on readlen == 0 */ + CALL_OCI_RETURN(connection->error, + OCILobRead( + connection->pServiceContext, + connection->pError, + mydescr->ocidescr, + &readlen, /* IN/OUT bytes toread/read */ + siz + 1, /* offset (starts with 1) */ + (dvoid *) ((char *) buf + siz), + readlen, /* size of buffer */ + (dvoid *)0, + (OCICallbackLobRead) 0, /* callback... */ + (ub2) connection->session->charsetId, /* The character set ID of the buffer data. */ + (ub1) SQLCS_IMPLICIT /* The character set form of the buffer data. */ + ) + ); + + siz += readlen; + readlen = LOBREADSIZE; + + if (connection->error == OCI_NEED_DATA) { + buf = erealloc(buf,siz + LOBREADSIZE + 1); + continue; + } else { + break; + } + } + + if (connection->error) { + oci_error(connection->pError, "OCILobRead", connection->error); + oci_handle_error(connection, connection->error); + efree(buf); + return -1; + } + + if (mydescr->type == OCI_DTYPE_FILE) { + CALL_OCI_RETURN(connection->error, + OCILobFileClose( + connection->pServiceContext, + connection->pError, + mydescr->ocidescr + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobFileClose", connection->error); + oci_handle_error(connection, connection->error); + efree(buf); + return -1; + } + } + + buf = erealloc(buf,siz+1); + buf[ siz ] = 0; + + *buffer = buf; + *loblen = siz; + + oci_debug("oci_loadlob: size=%d",siz); + + return 0; +} +/* }}} */ + +/* {{{ oci_readlob() +*/ +static int oci_readlob(oci_connection *connection, oci_descriptor *mydescr, char **buffer, ub4 *len) +{ + ub4 siz = 0; + ub4 readlen = 0; + ub4 loblen = 0; + ub4 bytes = 0; + char *buf; + TSRMLS_FETCH(); + + /* we're not going to read LOB, if length is not known */ + if (!len || (int)*len <= 0) { + return -1; + } + + if (mydescr->type == OCI_DTYPE_FILE) { + CALL_OCI_RETURN(connection->error, + OCILobFileOpen( + connection->pServiceContext, + connection->pError, + mydescr->ocidescr, + OCI_FILE_READONLY + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobFileOpen",connection->error); + oci_handle_error(connection, connection->error); + return -1; + } + } + + if (oci_lobgetlen(connection, mydescr, &loblen) != 0) { + *len = 0; + return -1; + } + + /* check if we're in LOB's borders */ + if ((mydescr->lob_current_position + *len) > loblen) { + *len = loblen - mydescr->lob_current_position; + } + + if ((int)*len > 0) { + buf = emalloc(*len + 1); + + /* set offset to current LOB's position */ + siz = mydescr->lob_current_position; + + /* check if len is smaller, if not - using LOBREADSIZE' sized buffer */ + if (*len > LOBREADSIZE) { + readlen = LOBREADSIZE; + } else { + readlen = *len; + } + } else { + *len = 0; + return -1; + } + + while (readlen > 0 && bytes < *len && siz < loblen) { /* paranoia */ + CALL_OCI_RETURN(connection->error, + OCILobRead( + connection->pServiceContext, + connection->pError, + mydescr->ocidescr, + &readlen, /* IN/OUT bytes toread/read */ + siz + 1, /* offset (starts with 1) */ + (dvoid *) ((char *) buf + bytes), + readlen, /* size of buffer */ + (dvoid *)0, + (OCICallbackLobRead) 0, /* callback... */ + (ub2) connection->session->charsetId, /* The character set ID of the buffer data. */ + (ub1) SQLCS_IMPLICIT /* The character set form of the buffer data. */ + ) + ); + + siz += readlen; + bytes += readlen; + + if ((*len - bytes) > LOBREADSIZE) { + readlen = LOBREADSIZE; + } else { + readlen = *len - bytes; + } + + if (connection->error == OCI_NEED_DATA) { + buf = erealloc(buf,bytes + LOBREADSIZE + 1); + continue; + } else { + break; + } + } + + /* moving current position */ + mydescr->lob_current_position = siz; + + if (connection->error) { + oci_error(connection->pError, "OCILobRead", connection->error); + oci_handle_error(connection, connection->error); + efree(buf); + return -1; + } + + if (mydescr->type == OCI_DTYPE_FILE) { + CALL_OCI_RETURN(connection->error, + OCILobFileClose( + connection->pServiceContext, + connection->pError, + mydescr->ocidescr + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobFileClose", connection->error); + oci_handle_error(connection, connection->error); + efree(buf); + return -1; + } + } + + buf = erealloc(buf,bytes+1); + buf[ bytes ] = 0; + + *buffer = buf; + *len = bytes; + + oci_debug("oci_readlob: size=%d",bytes); + + return 0; +} +/* }}} */ + +/* {{{ oci_failover_callback() +*/ +#if 0 /* not needed yet ! */ +static sb4 oci_failover_callback(dvoid *svchp, dvoid *envhp, dvoid *fo_ctx, ub4 fo_type, ub4 fo_event) +{ + /* + this stuff is from an oci sample - it will get cleaned up as soon as i understand it!!! (thies@thieso.net 990420) + right now i cant get oracle to even call it;-((((((((((( + */ + + switch (fo_event) { + case OCI_FO_BEGIN: { + printf(" Failing Over ... Please stand by \n"); + printf(" Failover type was found to be %s \n", + ((fo_type==OCI_FO_NONE) ? "NONE" + :(fo_type==OCI_FO_SESSION) ? "SESSION" + :(fo_type==OCI_FO_SELECT) ? "SELECT" + : "UNKNOWN!")); + printf(" Failover Context is :%s\n", + (fo_ctx?(char *)fo_ctx:"NULL POINTER!")); + break; + } + + case OCI_FO_ABORT: { + printf(" Failover aborted. Failover will not take place.\n"); + break; + } + + case OCI_FO_END: { + printf(" Failover ended ...resuming services\n"); + break; + } + + case OCI_FO_REAUTH: { + printf(" Failed over user. Resuming services\n"); + + /* Application can check the OCI_ATTR_SESSION attribute of + the service handle to find out the user being + re-authenticated. + + After this, the application can replay any ALTER SESSION + commands associated with this session. These must have + been saved by the application in the fo_ctx + */ + break; + } + + case OCI_FO_ERROR: { + printf(" Failover error gotten. Sleeping...\n"); + php_sleep(3); + /* cannot find this blody define !!! return OCI_FO_RETRY; */ + break; + } + + default: { + printf("Bad Failover Event: %ld.\n", fo_event); + break; + } + } + + return 0; +} +#endif +/* }}} */ + +/* {{{ oci_bind_in_callback() +*/ +static sb4 oci_bind_in_callback( + dvoid *ictxp, /* context pointer */ + OCIBind *bindp, /* bind handle */ + ub4 iter, /* 0-based execute iteration value */ + ub4 index, /* index of current array for PL/SQL or row index for SQL */ + dvoid **bufpp, /* pointer to data */ + ub4 *alenp, /* size after value/piece has been read */ + ub1 *piecep, /* which piece */ + dvoid **indpp) /* indicator value */ +{ + oci_bind *phpbind; + zval *val; + TSRMLS_FETCH(); + + if (!(phpbind=(oci_bind *)ictxp) || !(val = phpbind->zval)) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "!phpbind || !phpbind->val"); + return OCI_ERROR; + } + + if (ZVAL_IS_NULL(val)) { + /* we're going to insert a NULL column */ + phpbind->indicator = -1; + *bufpp = 0; + *alenp = -1; + *indpp = (dvoid *)&phpbind->indicator; + } else if ((phpbind->descr == 0) && (phpbind->pStmt == 0)) { + /* "normal string bind */ + convert_to_string(val); + + *bufpp = Z_STRVAL_P(val); + *alenp = Z_STRLEN_P(val); + *indpp = (dvoid *)&phpbind->indicator; + } else if (phpbind->pStmt != 0) { + /* RSET */ + *bufpp = phpbind->pStmt; + *alenp = -1; /* seems to be allright */ + *indpp = (dvoid *)&phpbind->indicator; + } else { + /* descriptor bind */ + *bufpp = phpbind->descr; + *alenp = -1; /* seems to be allright */ + *indpp = (dvoid *)&phpbind->indicator; + } + + *piecep = OCI_ONE_PIECE; /* pass all data in one go */ + + return OCI_CONTINUE; +} +/* }}} */ + +/* {{{ oci_bind_out_callback() */ +static sb4 oci_bind_out_callback( + dvoid *octxp, /* context pointer */ + OCIBind *bindp, /* bind handle */ + ub4 iter, /* 0-based execute iteration value */ + ub4 index, /* index of current array for PL/SQL or row index for SQL */ + dvoid **bufpp, /* pointer to data */ + ub4 **alenpp, /* size after value/piece has been read */ + ub1 *piecep, /* which piece */ + dvoid **indpp, /* indicator value */ + ub2 **rcodepp) /* return code */ +{ + oci_bind *phpbind; + zval *val; + sb4 retval = OCI_ERROR; + TSRMLS_FETCH(); + + if (!(phpbind=(oci_bind *)octxp) || !(val = phpbind->zval)) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "!phpbind || !phpbind->val"); + return retval; + } + + if ((Z_TYPE_P(val) == IS_OBJECT) || (Z_TYPE_P(val) == IS_RESOURCE)) { + retval = OCI_CONTINUE; + } else { + convert_to_string(val); + zval_dtor(val); + + Z_STRLEN_P(val) = OCI_PIECE_SIZE; /* 64K-1 is max XXX */ + Z_STRVAL_P(val) = emalloc(Z_STRLEN_P(phpbind->zval)); + + /* XXX we assume that zend-zval len has 4 bytes */ + *alenpp = (ub4*) &Z_STRLEN_P(phpbind->zval); + *bufpp = Z_STRVAL_P(phpbind->zval); + *piecep = OCI_ONE_PIECE; + *rcodepp = &phpbind->retcode; + *indpp = &phpbind->indicator; + retval = OCI_CONTINUE; + } + + return retval; +} +/* }}} */ + +/* {{{ _oci_open_session() +*/ +#include "ext/standard/php_smart_str.h" +static oci_session *_oci_open_session(oci_server* server,char *username,char *password,int persistent,int exclusive,char *charset) +{ + zend_llist *session_list; + oci_session *session = NULL; + OCISvcCtx *svchp = NULL; + smart_str hashed_details = {0}; +#ifdef HAVE_OCI_9_2 + ub2 charsetid = 0; +#endif + TSRMLS_FETCH(); + + /* + check if we already have this user authenticated + + we will reuse authenticated users within a request no matter if the user requested a persistent + connections or not! + + but only as persistent requested connections will be kept between requests! + */ + +#if defined(HAVE_OCI_9_2) + if (*charset) { + smart_str_appends_ex(&hashed_details, charset, 1); + } else { + size_t rsize; + + /* Safe, charsetid is initialized to 0 */ + CALL_OCI( + OCINlsEnvironmentVariableGet( + &charsetid, + 2, + OCI_NLS_CHARSET_ID, + 0, + &rsize + ) + ); + + smart_str_append_unsigned_ex(&hashed_details, charsetid, 1); + + charsetid = 0; + } +#else + { + char *nls_lang = getenv("NLS_LANG"); + + /* extract charset from NLS_LANG=LANUAGE_TERRITORY.CHARSET */ + if (nls_lang) { + char *p = strchr(nls_lang, '.'); + + if (p) { + smart_str_appends_ex(&hashed_details, p + 1, 1); + } + } + } +#endif + + smart_str_appends_ex(&hashed_details, SAFE_STRING(username), 1); + smart_str_appends_ex(&hashed_details, SAFE_STRING(password), 1); + smart_str_appends_ex(&hashed_details, SAFE_STRING(server->dbname), 1); + smart_str_0(&hashed_details); + + if (!exclusive) { + mutex_lock(mx_lock); + if (zend_ts_hash_find(persistent_sessions, hashed_details.c, hashed_details.len+1, (void **) &session_list) != SUCCESS) { + zend_llist tmp; + /* first session, set up a session list */ + zend_llist_init(&tmp, sizeof(oci_session), (llist_dtor_func_t) _session_pcleanup, 1); + zend_ts_hash_update(persistent_sessions, hashed_details.c, hashed_details.len+1, &tmp, sizeof(zend_llist), (void **) &session_list); + } else { + + /* session list found, search for an idle session or an already opened session by the current thread */ + session = zend_llist_get_first(session_list); + while ((session != NULL) && session->thread && (session->thread != thread_id())) { + session = zend_llist_get_next(session_list); + } + + if (session != NULL) { + /* mark session as busy */ + session->thread = thread_id(); + } + + } + + if (session) { + if (session->is_open) { + if (persistent) { + session->persistent = 1; + } + smart_str_free_ex(&hashed_details, 1); + mutex_unlock(mx_lock); + return session; + } else { + _oci_close_session(session); + /* breakthru to open */ + } + } + mutex_unlock(mx_lock); + } + + session = ecalloc(1,sizeof(oci_session)); + + if (!session) { + goto CLEANUP; + } + + session->persistent = persistent; + session->server = server; + session->exclusive = exclusive; + session->sessions_list = session_list; + session->thread = thread_id(); + +#ifdef HAVE_OCI_9_2 + + /* following chunk is Oracle 9i+ ONLY */ + if (*charset) { + /* + get ub2 charset id based on charset + this is pretty secure, since if we don't have a valid character set name, + 0 comes back and we can still use the 0 in all further statements -> OCI uses NLS_LANG + setting in that case + */ + CALL_OCI_RETURN(charsetid, + OCINlsCharSetNameToId( + OCI(pEnv), + charset + ) + ); + + oci_debug("oci_do_connect: using charset id=%d",charsetid); + } + + session->charsetId = charsetid; + + /* create an environment using the character set id, Oracle 9i+ ONLY */ + CALL_OCI( + OCIEnvNlsCreate( + &session->pEnv, + PHP_OCI_INIT_MODE, + 0, + NULL, + NULL, + NULL, + 0, + NULL, + charsetid, + charsetid + ) + ); + +#else + + /* fallback solution (simply use global env and charset, same behaviour as always been) */ + session->pEnv = OCI(pEnv); + session->charsetId = 0; + +#endif /* HAVE_OCI_9_2 */ + + /* allocate temporary Service Context */ + CALL_OCI_RETURN(OCI(error), + OCIHandleAlloc( + session->pEnv, + (dvoid **)&svchp, + OCI_HTYPE_SVCCTX, + 0, + NULL + ) + ); + + if (OCI(error) != OCI_SUCCESS) { + oci_error(OCI(pError), "_oci_open_session: OCIHandleAlloc OCI_HTYPE_SVCCTX", OCI(error)); + goto CLEANUP; + } + + /* allocate private session-handle */ + CALL_OCI_RETURN(OCI(error), + OCIHandleAlloc( + session->pEnv, + (dvoid **)&session->pSession, + OCI_HTYPE_SESSION, + 0, + NULL + ) + ); + + if (OCI(error) != OCI_SUCCESS) { + oci_error(OCI(pError), "_oci_open_session: OCIHandleAlloc OCI_HTYPE_SESSION", OCI(error)); + goto CLEANUP; + } + + /* Set the server handle in service handle */ + CALL_OCI_RETURN(OCI(error), + OCIAttrSet( + svchp, + OCI_HTYPE_SVCCTX, + server->pServer, + 0, + OCI_ATTR_SERVER, + OCI(pError) + ) + ); + + if (OCI(error) != OCI_SUCCESS) { + oci_error(OCI(pError), "_oci_open_session: OCIAttrSet OCI_ATTR_SERVER", OCI(error)); + goto CLEANUP; + } + + /* set the username in user handle */ + CALL_OCI_RETURN(OCI(error), + OCIAttrSet( + (dvoid *) session->pSession, + (ub4) OCI_HTYPE_SESSION, + (dvoid *) username, + (ub4) strlen(username), + (ub4) OCI_ATTR_USERNAME, + OCI(pError) + ) + ); + + if (OCI(error) != OCI_SUCCESS) { + oci_error(OCI(pError), "OCIAttrSet OCI_ATTR_USERNAME", OCI(error)); + goto CLEANUP; + } + + /* set the password in user handle */ + CALL_OCI_RETURN(OCI(error), + OCIAttrSet( + (dvoid *) session->pSession, + (ub4) OCI_HTYPE_SESSION, + (dvoid *) password, + (ub4) strlen(password), + (ub4) OCI_ATTR_PASSWORD, + OCI(pError) + ) + ); + + if (OCI(error) != OCI_SUCCESS) { + oci_error(OCI(pError), "OCIAttrSet OCI_ATTR_PASSWORD", OCI(error)); + goto CLEANUP; + } + + CALL_OCI_RETURN(OCI(error), + OCISessionBegin( + svchp, + OCI(pError), + session->pSession, + (ub4) OCI_CRED_RDBMS, + (ub4) OCI_DEFAULT + ) + ); + + if (OCI(error) != OCI_SUCCESS) { + oci_error(OCI(pError), "OCISessionBegin", OCI(error)); + /* OCISessionBegin returns OCI_SUCCESS_WITH_INFO when + * user's password has expired, but is still usable. + * */ + if (OCI(error) != OCI_SUCCESS_WITH_INFO) { + goto CLEANUP; + } + } + + /* Free Temporary Service Context */ + CALL_OCI( + OCIHandleFree( + (dvoid *) svchp, + (ub4) OCI_HTYPE_SVCCTX + ) + ); + + session->num = zend_list_insert(session, le_session); + session->is_open = 1; + + mutex_lock(mx_lock); + num_links++; + if (!exclusive) { + zend_llist_add_element(session_list, session); + efree(session); + session = (oci_session*) session_list->tail->data; + num_persistent++; + } + mutex_unlock(mx_lock); + + oci_debug("_oci_open_session new sess=%d user=%s",session->num,username); + + return session; + +CLEANUP: + oci_debug("_oci_open_session: FAILURE -> CLEANUP called"); + + _oci_close_session(session); + + return 0; +} +/* }}} */ + +/* {{{ _oci_close_session() +*/ +static int _session_compare(void *a, void *b) +{ + oci_session *sess1 = (oci_session*) a; + oci_session *sess2 = (oci_session*) b; + + return sess1->num == sess2->num; +} + +static void _oci_close_session(oci_session *session) +{ + OCISvcCtx *svchp; + TSRMLS_FETCH(); + + if (!session) { + return; + } + + oci_debug("START _oci_close_session: logging-off sess=%d",session->num); + + if (session->is_open) { + /* Temporary Service Context */ + CALL_OCI_RETURN(OCI(error), + OCIHandleAlloc( + session->pEnv, + (dvoid **) &svchp, + (ub4) OCI_HTYPE_SVCCTX, + (size_t) 0, + (dvoid **) 0 + ) + ); + + if (OCI(error) != OCI_SUCCESS) { + oci_error(OCI(pError), "_oci_close_session OCIHandleAlloc OCI_HTYPE_SVCCTX", OCI(error)); + } + + /* Set the server handle in service handle */ + CALL_OCI_RETURN(OCI(error), + OCIAttrSet( + svchp, + OCI_HTYPE_SVCCTX, + session->server->pServer, + 0, + OCI_ATTR_SERVER, + OCI(pError) + ) + ); + + if (OCI(error) != OCI_SUCCESS) { + oci_error(OCI(pError), "_oci_close_session: OCIAttrSet OCI_ATTR_SERVER", OCI(error)); + } + + /* Set the Authentication handle in the service handle */ + CALL_OCI_RETURN(OCI(error), + OCIAttrSet( + svchp, + OCI_HTYPE_SVCCTX, + session->pSession, + 0, + OCI_ATTR_SESSION, + OCI(pError) + ) + ); + + if (OCI(error) != OCI_SUCCESS) { + oci_error(OCI(pError), "_oci_close_session: OCIAttrSet OCI_ATTR_SESSION", OCI(error)); + } + + CALL_OCI_RETURN(OCI(error), + OCISessionEnd( + svchp, + OCI(pError), + session->pSession, + (ub4) 0 + ) + ); + + if (OCI(error) != OCI_SUCCESS) { + oci_error(OCI(pError), "_oci_close_session: OCISessionEnd", OCI(error)); + } + + CALL_OCI( + OCIHandleFree( + (dvoid *) svchp, + (ub4) OCI_HTYPE_SVCCTX + ) + ); + + } else { + oci_debug("_oci_close_session: logging-off DEAD session"); + } + + if (session->pSession) { + CALL_OCI( + OCIHandleFree( + (dvoid *) session->pSession, + (ub4) OCI_HTYPE_SESSION + ) + ); + } + + mutex_lock(mx_lock); + num_links--; + if (!OCI(shutdown) && session->persistent) { + zend_llist_del_element(session->sessions_list, session, _session_compare); + num_persistent--; + } + mutex_unlock(mx_lock); + +#ifdef HAVE_OCI_9_2 + /* free environment handle (and fix bug #29652 with growing .msb FD number under weirdie Solarises) */ + CALL_OCI( + OCIHandleFree( + (dvoid *) session->pEnv, + OCI_HTYPE_ENV + ) + ); +#endif + + if (session->exclusive) { + efree(session); + } +} +/* }}} */ + +/* {{{ _oci_open_server() +*/ +static oci_server *_oci_open_server(char *dbname,int persistent) +{ + oci_server *server, *pserver = NULL; + TSRMLS_FETCH(); + + /* + check if we already have this server open + + we will reuse servers within a request no matter if the user requested persistent + connections or not! + + but only as pesistent requested connections will be kept between requests! + */ + + /* TODO either keep servers global or don't reuse them at all */ + zend_ts_hash_find(persistent_servers, dbname, strlen(dbname)+1, (void **) &pserver); + + if (pserver) { + /* XXX ini-flag */ + /* + if (!oci_ping(pserver)) { + pserver->is_open = 0; + } + */ + if (pserver->is_open) { + /* if our new users uses this connection persistent, we're keeping it! */ + if (persistent) { + pserver->persistent = persistent; + } + + return pserver; + } else { /* server "died" in the meantime - try to reconnect! */ + _oci_close_server(pserver); + /* breakthru to open */ + } + } + + server = ecalloc(1,sizeof(oci_server)); + + server->persistent = persistent; + server->dbname = strdup(SAFE_STRING(dbname)); + + CALL_OCI( + OCIHandleAlloc( + OCI(pEnv), + (dvoid **)&server->pServer, + OCI_HTYPE_SERVER, + 0, + NULL + ) + ); + + CALL_OCI_RETURN(OCI(error), + OCIServerAttach( + server->pServer, + OCI(pError), + (text*)server->dbname, + strlen(server->dbname), + (ub4) OCI_DEFAULT + ) + ); + + if (OCI(error)) { + oci_error(OCI(pError), "_oci_open_server", OCI(error)); + goto CLEANUP; + } + + zend_ts_hash_update(persistent_servers, + server->dbname, + strlen(server->dbname)+1, + (void *)server, + sizeof(oci_server), + (void**)&pserver); + + pserver->num = zend_list_insert(pserver,le_server); + pserver->is_open = 1; + + oci_debug("_oci_open_server new conn=%d dname=%s",server->num,server->dbname); + + efree(server); + + return pserver; + +CLEANUP: + oci_debug("_oci_open_server: FAILURE -> CLEANUP called"); + + _oci_close_server(server); + + return 0; +} + +#if 0 + server->failover.fo_ctx = (dvoid *) server; + server->failover.callback_function = oci_failover_callback; + + error = OCIAttrSet((dvoid *)server->pServer, + (ub4) OCI_HTYPE_SERVER, + (dvoid *) &server->failover, + (ub4) 0, + (ub4) OCI_ATTR_FOCBK, + OCI(pError)); + + if (error) { + oci_error(OCI(pError), "_oci_open_server OCIAttrSet OCI_ATTR_FOCBK", error); + goto CLEANUP; + } +#endif +/* }}} */ + +/* {{{ _oci_close_server() +*/ +static int _oci_session_cleanup(void *data TSRMLS_DC) +{ + zend_rsrc_list_entry *le = (zend_rsrc_list_entry *) data; + + if (le->type == le_session) { + oci_server *server = ((oci_session*) le->ptr)->server; + if (server && server->is_open == 2) + return 1; + } + return 0; +} + +static void _oci_close_server(oci_server *server) +{ + char *dbname; + int oldopen; + TSRMLS_FETCH(); + + oldopen = server->is_open; + server->is_open = 2; + if (!OCI(shutdown)) { + zend_hash_apply(&EG(regular_list), (apply_func_t) _oci_session_cleanup TSRMLS_CC); + } + server->is_open = oldopen; + + oci_debug("START _oci_close_server: detaching conn=%d dbname=%s",server->num,server->dbname); + + /* XXX close server here */ + + if (server->is_open) { + if (server->pServer && OCI(pError)) { + CALL_OCI_RETURN(OCI(error), + OCIServerDetach( + server->pServer, + OCI(pError), + OCI_DEFAULT + ) + ); + + if (OCI(error)) { + oci_error(OCI(pError), "oci_close_server OCIServerDetach", OCI(error)); + } + } + } else { + oci_debug("_oci_close_server: closing DEAD server"); + } + + if (server->pServer) { + CALL_OCI( + OCIHandleFree( + (dvoid *) server->pServer, + (ub4) OCI_HTYPE_SERVER + ) + ); + } + + dbname = server->dbname; + + if (!OCI(shutdown)) { + zend_ts_hash_del(persistent_servers, dbname, strlen(dbname)+1); + } + + free(dbname); +} +/* }}} */ + +/* {{{ oci_do_connect() + Connect to an Oracle database and log on. returns a new session. */ +static void oci_do_connect(INTERNAL_FUNCTION_PARAMETERS,int persistent,int exclusive) +{ + char *username, *password, *dbname, *charset; + zval **userParam, **passParam, **dbParam, **charParam; + oci_server *server = 0; + oci_session *session = 0; + oci_connection *connection = 0; + + /* if a forth parameter is handed over, it is the charset identifier (but is only used in Oracle 9i+) */ + if (zend_get_parameters_ex(4, &userParam, &passParam, &dbParam, &charParam) == SUCCESS) { + convert_to_string_ex(userParam); + convert_to_string_ex(passParam); + convert_to_string_ex(dbParam); + convert_to_string_ex(charParam); + + username = Z_STRVAL_PP(userParam); + password = Z_STRVAL_PP(passParam); + dbname = Z_STRVAL_PP(dbParam); + charset = Z_STRVAL_PP(charParam); + oci_debug("oci_do_connect: using charset=%s",charset); + } else if (zend_get_parameters_ex(3, &userParam, &passParam, &dbParam) == SUCCESS) { + convert_to_string_ex(userParam); + convert_to_string_ex(passParam); + convert_to_string_ex(dbParam); + + username = Z_STRVAL_PP(userParam); + password = Z_STRVAL_PP(passParam); + dbname = Z_STRVAL_PP(dbParam); + charset = ""; + } else if (zend_get_parameters_ex(2, &userParam, &passParam) == SUCCESS) { + convert_to_string_ex(userParam); + convert_to_string_ex(passParam); + + username = Z_STRVAL_PP(userParam); + password = Z_STRVAL_PP(passParam); + dbname = ""; + charset = ""; + } else { + WRONG_PARAM_COUNT; + } + + connection = (oci_connection *) ecalloc(1,sizeof(oci_connection)); + + if (!connection) { + goto CLEANUP; + } + + server = _oci_open_server(dbname,persistent); + + if (!server) { + goto CLEANUP; + } + + if (exclusive) { + /* exlusive session can never be persistent!*/ + persistent = 0; + } else { + /* if our server-context is not persistent we can't */ + persistent = (server->persistent) ? persistent : 0; + } + + session = _oci_open_session(server,username,password,persistent,exclusive,charset); + + if (!session) { + goto CLEANUP; + } + + /* set our session */ + connection->session = session; + + /* allocate our private error-handle */ + CALL_OCI_RETURN(OCI(error), + OCIHandleAlloc( + connection->session->pEnv, + (dvoid **)&connection->pError, + OCI_HTYPE_ERROR, + 0, + NULL + ) + ); + + if (OCI(error) != OCI_SUCCESS) { + oci_error(OCI(pError), "oci_do_connect: OCIHandleAlloc OCI_HTYPE_ERROR",OCI(error)); + goto CLEANUP; + } + + /* allocate our service-context */ + CALL_OCI_RETURN(OCI(error), + OCIHandleAlloc( + connection->session->pEnv, + (dvoid **)&connection->pServiceContext, + OCI_HTYPE_SVCCTX, + 0, + NULL + ) + ); + + if (OCI(error) != OCI_SUCCESS) { + oci_error(OCI(pError), "oci_do_connect: OCIHandleAlloc OCI_HTYPE_SVCCTX",OCI(error)); + goto CLEANUP; + } + + /* Set the server handle in service handle */ + CALL_OCI_RETURN(connection->error, + OCIAttrSet( + connection->pServiceContext, + OCI_HTYPE_SVCCTX, + server->pServer, + 0, + OCI_ATTR_SERVER, + connection->pError + ) + ); + + if (connection->error != OCI_SUCCESS) { + oci_error(connection->pError, "oci_do_connect: OCIAttrSet OCI_ATTR_SERVER", connection->error); + goto CLEANUP; + } + + /* Set the Authentication handle in the service handle */ + CALL_OCI_RETURN(connection->error, + OCIAttrSet( + connection->pServiceContext, + OCI_HTYPE_SVCCTX, + session->pSession, + 0, + OCI_ATTR_SESSION, + connection->pError + ) + ); + + if (connection->error != OCI_SUCCESS) { + oci_error(connection->pError, "oci_do_connect: OCIAttrSet OCI_ATTR_SESSION", connection->error); + goto CLEANUP; + } + + /* + OCIAttrSet((dvoid *)session->server->pServer, + OCI_HTYPE_SERVER, + (dvoid *) "demo", + 0, + OCI_ATTR_EXTERNAL_NAME, + connection->pError); + + OCIAttrSet((dvoid *)session->server->pServer, + OCI_HTYPE_SERVER, + (dvoid *) "txn demo2", + 0, + OCI_ATTR_INTERNAL_NAME, + connection->pError); + */ + + connection->id = zend_list_insert(connection, le_conn); + + connection->is_open = 1; + + oci_debug("oci_do_connect: id=%d",connection->id); + + RETURN_RESOURCE(connection->id); + +CLEANUP: + oci_debug("oci_do_connect: FAILURE -> CLEANUP called"); + + if (connection->id) { + zend_list_delete(connection->id); + } else { + _oci_conn_list_dtor(connection TSRMLS_CC); + } + + RETURN_FALSE; +} +/* }}} */ + +/* {{{ oci_lob_flush() +*/ +static int oci_lob_flush(oci_descriptor* descr, int flush_flag TSRMLS_DC) +{ + OCILobLocator *mylob; + oci_connection *connection; + + mylob = (OCILobLocator *) descr->ocidescr; + + if (!mylob) { + return 0; + } + + /* do not really flush buffer, but reporting success + * to suppress OCI error when flushing not used buffer + * */ + if (descr->buffering != 2) { + return 1; + } + + connection = descr->conn; + + CALL_OCI_RETURN(connection->error, + OCILobFlushBuffer( + connection->pServiceContext, + connection->pError, + mylob, + flush_flag + ) + ); + + oci_debug("oci_lob_flush: flush_flag=%d",flush_flag); + + if (connection->error) { + oci_error(connection->pError, "OCILobFlushBuffer", connection->error); + oci_handle_error(connection, connection->error); + return 0; + } + + /* marking buffer as enabled and not used */ + descr->buffering = 1; + return 1; +} +/* }}} */ + +/* {{{ php_oci_fetch_row() +*/ +static void php_oci_fetch_row (INTERNAL_FUNCTION_PARAMETERS, int mode, int expected_args) +{ + zval **stmt, **arg2, **arg3; + oci_statement *statement; + oci_out_column *column; + ub4 nrows = 1; + int i; + + if (ZEND_NUM_ARGS() > expected_args) { + WRONG_PARAM_COUNT; + } + + if (expected_args > 2) { + /* only for ocifetchinto BC */ + + switch (ZEND_NUM_ARGS()) { + case 2: + if (zend_get_parameters_ex(2, &stmt, &arg2) == FAILURE) { + RETURN_FALSE; + } + if (!mode) { + mode = OCI_NUM; + } + break; + + case 3: + if (zend_get_parameters_ex(3, &stmt, &arg2, &arg3) == FAILURE) { + RETURN_FALSE; + } + convert_to_long_ex(arg3); + mode = Z_LVAL_PP(arg3); + break; + + default: + WRONG_PARAM_COUNT; + break; + } + + } else { + + switch (ZEND_NUM_ARGS()) { + case 1: + if (zend_get_parameters_ex(1, &stmt) == FAILURE) { + RETURN_FALSE; + } + if (!mode) { + mode = OCI_BOTH; + } + break; + + case 2: + if (zend_get_parameters_ex(2, &stmt, &arg2)==FAILURE) { + RETURN_FALSE; + } + convert_to_long_ex(arg2); + mode = Z_LVAL_PP(arg2); + break; + + default: + WRONG_PARAM_COUNT; + break; + } + } + + OCI_GET_STMT(statement,stmt); + + if (!oci_fetch(statement, nrows, "OCIFetchInto" TSRMLS_CC)) { + RETURN_FALSE; + } + + array_init(return_value); + + for (i = 0; i < statement->ncolumns; i++) { + column = oci_get_col(statement, i + 1, 0); + if (column == NULL) { + continue; + } + if ((column->indicator == -1) && ((mode & OCI_RETURN_NULLS) == 0)) { + continue; + } + + if (!(column->indicator == -1)) { + zval *element; + + MAKE_STD_ZVAL(element); + _oci_make_zval(element,statement,column,"OCIFetchInto",mode TSRMLS_CC); + + if (mode & OCI_NUM || !(mode & OCI_ASSOC)) { + add_index_zval(return_value, i, element); + } + if (mode & OCI_ASSOC) { + if (mode & OCI_NUM) { + ZVAL_ADDREF(element); + } + add_assoc_zval(return_value, column->name, element); + } + + } else { + if (mode & OCI_NUM || !(mode & OCI_ASSOC)) { + add_index_null(return_value, i); + } + if (mode & OCI_ASSOC) { + add_assoc_null(return_value, column->name); + } + } + } + + if (expected_args > 2) { + /* only for ocifetchinto BC + * in all other cases we return array, not long + */ + REPLACE_ZVAL_VALUE(arg2, return_value, 1); /* copy return_value to given reference */ + zval_dtor(return_value); + RETURN_LONG(statement->ncolumns); + } +} +/* }}} */ + +/************************* EXTENSION FUNCTIONS *************************/ + +/* {{{ proto bool oci_define_by_name(resource stmt, string name, mixed &var [, int type]) + Define a PHP variable to an Oracle column by name */ +/* if you want to define a LOB/CLOB etc make sure you allocate it via OCINewDescriptor BEFORE defining!!! */ +PHP_FUNCTION(oci_define_by_name) +{ + zval **stmt, **name, **var, **type; + oci_statement *statement; + oci_define *define, *tmp_define; + ub2 ocitype = SQLT_CHR; /* zero terminated string */ + int ac = ZEND_NUM_ARGS(); + + if (ac < 3 || ac > 4 || zend_get_parameters_ex(ac, &stmt, &name, &var, &type) == FAILURE) { + WRONG_PARAM_COUNT; + } + + switch (ac) { + case 4: + convert_to_long_ex(type); + ocitype = (ub2) Z_LVAL_PP(type); + /* possible breakthru */ + } + + OCI_GET_STMT(statement,stmt); + + convert_to_string_ex(name); + + if (statement->defines == NULL) { + ALLOC_HASHTABLE(statement->defines); + zend_hash_init(statement->defines, 13, NULL, _oci_define_hash_dtor, 0); + } + + define = ecalloc(1,sizeof(oci_define)); + + if (zend_hash_add(statement->defines, + Z_STRVAL_PP(name), + Z_STRLEN_PP(name), + define, + sizeof(oci_define), + (void **)&tmp_define) == SUCCESS) { + efree(define); + define = tmp_define; + } else { + efree(define); + RETURN_FALSE; + } + + define->name = (text*) estrndup(Z_STRVAL_PP(name),Z_STRLEN_PP(name)); + define->name_len = Z_STRLEN_PP(name); + define->type = ocitype; + define->zval = *var; + zval_add_ref(var); + + RETURN_TRUE; +} +/* }}} */ + +/* {{{ proto bool oci_bind_by_name(resource stmt, string name, mixed &var, [, int maxlength [, int type]]) + Bind a PHP variable to an Oracle placeholder by name */ +/* if you want to bind a LOB/CLOB etc make sure you allocate it via OCINewDescriptor BEFORE binding!!! */ +PHP_FUNCTION(oci_bind_by_name) +{ + zval **stmt, **name, **var, **maxlen, **type; + oci_statement *statement; + oci_statement *bindstmt; + oci_bind bind, *bindp; + oci_descriptor *descr; +#ifdef PHP_OCI8_HAVE_COLLECTIONS + oci_collection *coll; + dvoid *mycoll = 0; +#endif + int mode = OCI_DATA_AT_EXEC; + ub2 ocitype = SQLT_CHR; /* unterminated string */ + OCIStmt *mystmt = 0; + dvoid *mydescr = 0; + sb4 value_sz = -1; + int ac = ZEND_NUM_ARGS(), inx; + + if (ac < 3 || ac > 5 || zend_get_parameters_ex(ac, &stmt, &name, &var, &maxlen, &type) == FAILURE) { + WRONG_PARAM_COUNT; + } + + switch (ac) { + case 5: + convert_to_long_ex(type); + ocitype = (ub2) Z_LVAL_PP(type); + /* possible breakthru */ + case 4: + convert_to_long_ex(maxlen); + value_sz = Z_LVAL_PP(maxlen); + /* possible breakthru */ + } + + OCI_GET_STMT(statement,stmt); + + switch (ocitype) { +#ifdef PHP_OCI8_HAVE_COLLECTIONS + case SQLT_NTY: + if (Z_TYPE_PP(var) != IS_OBJECT) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "Variable must be allocated using OCINewCollection()"); + RETURN_FALSE; + } + if ((inx = _oci_get_ocicoll(*var,&coll TSRMLS_CC)) == 0) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "Variable must be allocated using OCINewCollection()"); + RETURN_FALSE; + } + if (!(mycoll = (dvoid *) coll->coll)) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "Collection empty"); + RETURN_FALSE; + } + value_sz = sizeof(void*); + mode = OCI_DEFAULT; +break; +#endif + case SQLT_BFILEE: + case SQLT_CFILEE: + case SQLT_CLOB: + case SQLT_BLOB: + case SQLT_RDD: + if (Z_TYPE_PP(var) != IS_OBJECT) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "Variable must be allocated using OCINewDescriptor()"); + RETURN_FALSE; + } + + if ((inx = _oci_get_ocidesc(*var,&descr TSRMLS_CC)) == 0) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "Variable must be allocated using OCINewDescriptor()"); + RETURN_FALSE; + } + + if (!(mydescr = (dvoid *) descr->ocidescr)) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "Descriptor empty"); + RETURN_FALSE; + } + value_sz = sizeof(void*); + break; + + case SQLT_RSET: + OCI_GET_STMT(bindstmt,var); + + if (!(mystmt = bindstmt->pStmt)) { + RETURN_FALSE; + } + value_sz = sizeof(void*); + break; + } + + if ((ocitype == SQLT_CHR) && (value_sz == -1)) { + convert_to_string_ex(var); + value_sz = Z_STRLEN_PP(var); + } + + if (value_sz == 0) { + value_sz = 1; + } + + convert_to_string_ex(name); + + if (!statement->binds) { + ALLOC_HASHTABLE(statement->binds); + zend_hash_init(statement->binds, 13, NULL, _oci_bind_hash_dtor, 0); + } + + memset((void*)&bind,0,sizeof(oci_bind)); + zend_hash_update(statement->binds, Z_STRVAL_PP(name), Z_STRLEN_PP(name) + 1, &bind, sizeof(oci_bind), (void **)&bindp); + + bindp->descr = mydescr; + bindp->pStmt = mystmt; + bindp->zval = *var; + zval_add_ref(var); + + CALL_OCI_RETURN(statement->error, + OCIBindByName( + statement->pStmt, /* statement handle */ + (OCIBind **)&bindp->pBind, /* bind hdl (will alloc) */ + statement->pError, /* error handle */ + (text*) Z_STRVAL_PP(name), /* placeholder name */ + Z_STRLEN_PP(name), /* placeholder length */ + (dvoid *)0, /* in/out data */ + value_sz, /* OCI_MAX_DATA_SIZE, */ /* max size of input/output data */ + (ub2)ocitype, /* in/out data type */ + (dvoid *)&bindp->indicator, /* indicator (ignored) */ + (ub2 *)0, /* size array (ignored) */ + (ub2 *)&bindp->retcode, /* return code (ignored) */ + (ub4)0, /* maxarr_len (PL/SQL only?) */ + (ub4 *)0, /* actual array size (PL/SQL only?) */ + mode /* mode */ + ) + ); + + if (statement->error != OCI_SUCCESS) { + oci_error(statement->pError, "OCIBindByName", statement->error); + oci_handle_error(statement->conn, statement->error); + RETURN_FALSE; + } + + if (mode == OCI_DATA_AT_EXEC) { + CALL_OCI_RETURN(statement->error, + OCIBindDynamic( + bindp->pBind, + statement->pError, + (dvoid *)bindp, + oci_bind_in_callback, + (dvoid *)bindp, + oci_bind_out_callback + ) + ); + + if (statement->error != OCI_SUCCESS) { + oci_error(statement->pError, "OCIBindDynamic", statement->error); + oci_handle_error(statement->conn, statement->error); + RETURN_FALSE; + } + } + +#ifdef PHP_OCI8_HAVE_COLLECTIONS + if (ocitype == SQLT_NTY) { + /* Bind object */ + CALL_OCI_RETURN(statement->error, + OCIBindObject( + bindp->pBind, + statement->pError, + coll->tdo, + (dvoid **) &(coll->coll), + (ub4 *) 0, + (dvoid **) 0, + (ub4 *) 0 + ) + ); + + if (statement->error) { + oci_error(statement->pError, "OCIBindObject", statement->error); + RETURN_FALSE; + } + } +#endif + + RETURN_TRUE; +} +/* }}} */ + +/* {{{ proto bool oci_free_descriptor() + Deletes large object description */ +PHP_FUNCTION(oci_free_descriptor) +{ + zval *id; + int inx; + oci_descriptor *descriptor; + + if ((id = getThis()) != 0) { + inx = _oci_get_ocidesc(id,&descriptor TSRMLS_CC); + if (inx) { + oci_debug("oci_free_descriptor: descr=%d",inx); + zend_list_delete(inx); + RETURN_TRUE; + } + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_free_descriptor() should not be called like this. Use $somelob->free() to free a LOB"); + + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool oci_lob_save( string data [, int offset ]) + Saves a large object */ +PHP_FUNCTION(oci_lob_save) +{ + zval *id, **arg,**oarg; + OCILobLocator *mylob; + oci_connection *connection; + oci_descriptor *descr; + int offparam,inx; + ub4 loblen; + ub4 curloblen; + ub4 offset; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + mylob = (OCILobLocator *) descr->ocidescr; + + if (!mylob) { + RETURN_FALSE; + } + + connection = descr->conn; + + offset = 0; + if (zend_get_parameters_ex(2, &arg, &oarg) == SUCCESS) { + convert_to_long_ex(oarg); + offparam = Z_LVAL_PP(oarg); + + CALL_OCI_RETURN(connection->error, + OCILobGetLength( + connection->pServiceContext, + connection->pError, + mylob, + &curloblen + ) + ); + + oci_debug("oci_lob_save: curloblen=%d",curloblen); + + if (offparam == -1) { + offset = curloblen; + } else if ((ub4)offparam >= curloblen) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "Offset is bigger than current LOB-Size - appending"); + offset = curloblen; + } else { + offset = (ub4)offparam; + } + } else if (zend_get_parameters_ex(1, &arg) == FAILURE) { + WRONG_PARAM_COUNT; + } + + convert_to_string_ex(arg); + loblen = Z_STRLEN_PP(arg); + + if (loblen < 1) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot save a lob which size is less than 1 byte"); + RETURN_FALSE; + } + + if (offset <= 0) { + offset = 1; + } + + CALL_OCI_RETURN(connection->error, + OCILobWrite( + connection->pServiceContext, + connection->pError, + mylob, + &loblen, + (ub4) offset, + (dvoid *) Z_STRVAL_PP(arg), + (ub4) loblen, + OCI_ONE_PIECE, + (dvoid *)0, + (OCICallbackLobWrite) 0, + (ub2) 0, + (ub1) SQLCS_IMPLICIT + ) + ); + + oci_debug("oci_lob_save: size=%d offset=%d",loblen,offset); + + if (connection->error) { + oci_error(connection->pError, "OCILobWrite", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + RETURN_TRUE; + } + + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool oci_lob_import( string filename ) + Saves a large object to file */ +PHP_FUNCTION(oci_lob_import) +{ + zval *id, **arg; + OCILobLocator *mylob; + oci_connection *connection; + oci_descriptor *descr; + char *filename; + int fp,inx; + char buf[8192]; + ub4 offset = 1; + ub4 loblen; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + mylob = (OCILobLocator *) descr->ocidescr; + + if (!mylob) { + RETURN_FALSE; + } + + connection = descr->conn; + + if (zend_get_parameters_ex(1, &arg) == FAILURE) { + WRONG_PARAM_COUNT; + } + + convert_to_string_ex(arg); + + if (php_check_open_basedir(Z_STRVAL_PP(arg) TSRMLS_CC)) { + RETURN_FALSE; + } + + filename = Z_STRVAL_PP(arg); + + if ((fp = VCWD_OPEN(filename, O_RDONLY|O_BINARY)) == -1) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "Can't open file %s", filename); + RETURN_FALSE; + } + + while ((loblen = read(fp, &buf, sizeof(buf))) > 0) { + CALL_OCI_RETURN(connection->error, + OCILobWrite( + connection->pServiceContext, + connection->pError, + mylob, + &loblen, + (ub4) offset, + (dvoid *) &buf, + (ub4) loblen, + OCI_ONE_PIECE, + (dvoid *)0, + (OCICallbackLobWrite) 0, + (ub2) 0, + (ub1) SQLCS_IMPLICIT + ) + ); + + oci_debug("oci_lob_import: size=%d",loblen); + + if (connection->error) { + oci_error(connection->pError, "OCILobWrite", connection->error); + oci_handle_error(connection, connection->error); + close(fp); + RETURN_FALSE; + } + + offset += loblen; + } + close(fp); + + RETURN_TRUE; + } + + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto string oci_lob_load() + Loads a large object */ +PHP_FUNCTION(oci_lob_load) +{ + zval *id; + oci_descriptor *descr; + char *buffer; + int inx; + ub4 loblen; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + if (!oci_loadlob(descr->conn,descr,&buffer,&loblen)) { + RETURN_STRINGL(buffer,loblen,0); + } else { + RETURN_FALSE; + } + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_lob_load() should not be called like this. Use $somelob->load() to load a LOB"); + + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto string oci_lob_read( int length ) + Reads particular part of a large object */ +PHP_FUNCTION(oci_lob_read) +{ + zval *id; + zval **len; + oci_descriptor *descr; + char *buffer; + int inx; + ub4 loblen; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + if (zend_get_parameters_ex(1, &len) == FAILURE) { + WRONG_PARAM_COUNT; + } + + loblen = Z_LVAL_PP(len); + if (oci_readlob(descr->conn,descr,&buffer,&loblen) == 0) { + RETURN_STRINGL(buffer,loblen,0); + } else { + RETURN_FALSE; + } + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_lob_read() should not be called like this. Use $somelob->read($len) to read a LOB"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool oci_lob_eof() + Checks if EOF is reached */ +PHP_FUNCTION(oci_lob_eof) +{ + zval *id; + oci_descriptor *descr; + int inx; + int len; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + if (oci_lobgetlen(descr->conn,descr,&len) == 0 && descr->lob_size >= 0) { + if (descr->lob_size == descr->lob_current_position) { + RETURN_TRUE; + } else { + RETURN_FALSE; + } + } + RETURN_FALSE; + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_lob_eof() should not be called like this. Use $somelob->eof() to check if end of LOB is reached"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto int oci_lob_tell() + Tells LOB pointer position */ +PHP_FUNCTION(oci_lob_tell) +{ + zval *id; + oci_descriptor *descr; + int inx; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + RETURN_LONG(descr->lob_current_position); + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_lob_tell() should not be called like this. Use $somelob->tell() to get current position of LOB pointer"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool oci_lob_rewind() + Rewind pointer of a LOB */ +PHP_FUNCTION(oci_lob_rewind) +{ + zval *id; + oci_descriptor *descr; + int inx; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + descr->lob_current_position = 0; + RETURN_TRUE; + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_lob_rewind() should not be called like this. Use $somelob->rewind() to set current position of LOB pointer to beginning"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool oci_lob_seek( int offset [, int whence ]) + Moves the pointer of a LOB */ +PHP_FUNCTION(oci_lob_seek) +{ + zval *id; + zval **arg1, **arg2; + int argcount = ZEND_NUM_ARGS(), whence = OCI_SEEK_SET; + oci_descriptor *descr; + int inx, len; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + if (argcount < 1 || argcount > 2 || zend_get_parameters_ex(argcount, &arg1, &arg2) == FAILURE) { + WRONG_PARAM_COUNT; + } + + convert_to_long_ex(arg1); + + if (oci_lobgetlen(descr->conn,descr,&len) == 0 && descr->lob_size >= 0) { + if (argcount > 1) { + convert_to_long_ex(arg2); + whence = Z_LVAL_PP(arg2); + switch (whence) { + case OCI_SEEK_CUR: + descr->lob_current_position += Z_LVAL_PP(arg1); + break; + + case OCI_SEEK_END: + if (descr->lob_size + Z_LVAL_PP(arg1) >= 0) { + descr->lob_current_position = descr->lob_size + Z_LVAL_PP(arg1); + } else { + descr->lob_current_position = 0; + } + break; + + case OCI_SEEK_SET: + default: + descr->lob_current_position = Z_LVAL_PP(arg1); + break; + } + } else { + /* OCI_SEEK_SET by default */ + descr->lob_current_position = Z_LVAL_PP(arg1); + } + RETURN_TRUE; + } else { + RETURN_FALSE; + } + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_lob_seek() should not be called like this. Use $somelob->seek($offset) to move pointer"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto int oci_lob_size() + Returns size of a large object */ +PHP_FUNCTION(oci_lob_size) +{ + zval *id; + oci_descriptor *descr; + int inx; + ub4 loblen; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + if (!oci_lobgetlen(descr->conn,descr,&loblen)) { + RETURN_LONG(loblen); + } else { + RETURN_FALSE; + } + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_lob_size() should not be called like this. Use $somelob->size() to get size of a LOB"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto int oci_lob_write( string string [, int length ]) + Writes data to current position of a LOB */ +PHP_FUNCTION(oci_lob_write) +{ + zval *id, **data,**length; + OCILobLocator *mylob; + oci_connection *connection; + oci_descriptor *descr; + int write_length,inx; + ub4 loblen; + ub4 curloblen; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + IS_LOB_INTERNAL(descr); + + mylob = (OCILobLocator *) descr->ocidescr; + + if (!mylob) { + RETURN_FALSE; + } + + connection = descr->conn; + + if (oci_lobgetlen(descr->conn,descr,&curloblen) != 0) { + RETURN_FALSE; + } + + if (zend_get_parameters_ex(2, &data, &length) == SUCCESS) { + convert_to_string_ex(data); + convert_to_long_ex(length); + write_length = Z_LVAL_PP(length); + } else if (zend_get_parameters_ex(1, &data) == SUCCESS) { + convert_to_string_ex(data); + write_length = Z_STRLEN_PP(data); + } else { + WRONG_PARAM_COUNT; + } + + if (write_length < 1) { + RETURN_LONG(0); + } + + loblen = write_length; + + CALL_OCI_RETURN(connection->error, + OCILobWrite( + connection->pServiceContext, + connection->pError, + mylob, + &loblen, + (ub4) descr->lob_current_position+1, + (dvoid *) Z_STRVAL_PP(data), + (ub4) loblen, + OCI_ONE_PIECE, + (dvoid *)0, + (OCICallbackLobWrite) 0, + (ub2) 0, + (ub1) SQLCS_IMPLICIT + ) + ); + + oci_debug("oci_lob_write: size=%d offset=%d",loblen,descr->lob_current_position); + + if (connection->error) { + oci_error(connection->pError, "OCILobWrite", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + descr->lob_current_position += loblen; + + if (descr->lob_current_position > descr->lob_size) { + descr->lob_size = descr->lob_current_position; + } + + /* marking buffer as used */ + if (descr->buffering == 1) { + descr->buffering = 2; + } + RETURN_LONG(loblen); + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_lob_write() should not be called like this. Use $somelob->write($data,$len) to write to a LOB"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool oci_lob_append( object lob ) + Appends data from a LOB to another LOB */ +PHP_FUNCTION(oci_lob_append) +{ + zval *id, **arg; + OCILobLocator *mylob,*my_fromlob; + oci_connection *connection; + oci_descriptor *descr,*from_descr; + int inx; + ub4 curloblen,from_curloblen; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + IS_LOB_INTERNAL(descr); + + mylob = (OCILobLocator *) descr->ocidescr; + + if (!mylob) { + RETURN_FALSE; + } + + connection = descr->conn; + + if (oci_lobgetlen(descr->conn,descr,&curloblen) != 0) { + RETURN_FALSE; + } + + if (zend_get_parameters_ex(1, &arg) == SUCCESS) { + convert_to_object_ex(arg); + if ((inx = _oci_get_ocidesc(*arg,&from_descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + my_fromlob = (OCILobLocator *) from_descr->ocidescr; + + if (!my_fromlob) { + RETURN_FALSE; + } + + if (oci_lobgetlen(from_descr->conn,from_descr,&from_curloblen) != 0) { + RETURN_FALSE; + } + } else { + WRONG_PARAM_COUNT; + } + + if (from_descr->lob_size == 0) { + RETURN_LONG(0); + } + + CALL_OCI_RETURN(connection->error, + OCILobAppend( + connection->pServiceContext, + connection->pError, + mylob, + my_fromlob + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobAppend", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + RETURN_TRUE; + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_lob_append() should not be called like this. Use $somelob->append($LOB_from) to append data from a LOB to another LOB"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool oci_lob_truncate( [ int length ]) + Truncates a LOB */ +PHP_FUNCTION(oci_lob_truncate) +{ + zval *id, **length; + OCILobLocator *mylob; + oci_connection *connection; + oci_descriptor *descr; + int inx; + ub4 trim_length; + ub4 curloblen; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + IS_LOB_INTERNAL(descr); + + mylob = (OCILobLocator *) descr->ocidescr; + + if (!mylob) { + RETURN_FALSE; + } + + connection = descr->conn; + + if (oci_lobgetlen(descr->conn,descr,&curloblen) != 0) { + RETURN_FALSE; + } + + if (zend_get_parameters_ex(1, &length) == SUCCESS) { + convert_to_long_ex(length); + trim_length = Z_LVAL_PP(length); + } else { + trim_length = 0; + } + + if (trim_length < 0) { + /* negative length is not allowed */ + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCILobTrim( + connection->pServiceContext, + connection->pError, + mylob, + trim_length + ) + ); + + oci_debug("oci_lob_truncate: trim_length=%d",trim_length); + + if (connection->error) { + oci_error(connection->pError, "OCILobTrim", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + descr->lob_size = trim_length; + RETURN_TRUE; + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_lob_truncate() should not be called like this. Use $somelob->truncate($length) to truncate a LOB to a specified length"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto int oci_lob_erase( [ int offset [, int length ] ] ) + Erases a specified portion of the internal LOB, starting at a specified offset */ +PHP_FUNCTION(oci_lob_erase) +{ + zval *id, **length, **offset; + OCILobLocator *mylob; + oci_connection *connection; + oci_descriptor *descr; + int inx; + ub4 erase_length, erase_offset; + ub4 curloblen; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + IS_LOB_INTERNAL(descr); + + mylob = (OCILobLocator *) descr->ocidescr; + + if (!mylob) { + RETURN_FALSE; + } + + connection = descr->conn; + + if (oci_lobgetlen(descr->conn,descr,&curloblen) != 0) { + RETURN_FALSE; + } + + if (zend_get_parameters_ex(2, &offset, &length) == SUCCESS) { + convert_to_long_ex(offset); + convert_to_long_ex(length); + + erase_offset = Z_LVAL_PP(offset); + erase_length = Z_LVAL_PP(length); + } else if (zend_get_parameters_ex(1, &offset) == SUCCESS) { + convert_to_long_ex(offset); + + erase_offset = Z_LVAL_PP(offset); + erase_length = descr->lob_size - erase_offset; + } else { + erase_offset = 0; + erase_length = descr->lob_size; + } + + if (erase_length < 1) { + RETURN_LONG(0); + } + + if (erase_offset > descr->lob_size) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "oci_lob_erase(): offset is greater than LOB's length"); + } + + CALL_OCI_RETURN(connection->error, + OCILobErase( + connection->pServiceContext, + connection->pError, + mylob, + &erase_length, + erase_offset+1 + ) + ); + + oci_debug("oci_lob_erase: erase_length=%d, erase_offset=%d",erase_length,erase_offset); + + if (connection->error) { + oci_error(connection->pError, "OCILobErase", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + RETURN_LONG(erase_length); + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_lob_erase() should not be called like this. Use $somelob->erase($offset, $length) to erase specified part of LOB"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool oci_lob_flush( [ int flag ] ) + Flushes the LOB buffer */ +PHP_FUNCTION(oci_lob_flush) +{ + zval *id, **flag; + OCILobLocator *mylob; + oci_connection *connection; + oci_descriptor *descr; + int inx, flush_flag; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + IS_LOB_INTERNAL(descr); + + mylob = (OCILobLocator *) descr->ocidescr; + + if (!mylob) { + RETURN_FALSE; + } + + connection = descr->conn; + + if (zend_get_parameters_ex(1, &flag) == SUCCESS) { + convert_to_long_ex(flag); + flush_flag = Z_LVAL_PP(flag); + } else { + flush_flag = 0; + } + + if (descr->buffering == 0) { + /* buffering wasn't enabled, there is nothing to flush */ + RETURN_FALSE; + } + + if (oci_lob_flush(descr,flush_flag TSRMLS_CC) == 1) { + RETURN_TRUE; + } + RETURN_FALSE; + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_lob_flush() should not be called like this. Use $somelob->flush() to flush LOB buffer"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool ocisetbufferinglob( boolean flag ) + Enables/disables buffering for a LOB */ +PHP_FUNCTION(ocisetbufferinglob) +{ + zval *id, **flag; + OCILobLocator *mylob; + oci_connection *connection; + oci_descriptor *descr; + int inx, buffering_flag; + ub4 curloblen; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + IS_LOB_INTERNAL(descr); + + mylob = (OCILobLocator *) descr->ocidescr; + + if (!mylob) { + RETURN_FALSE; + } + + connection = descr->conn; + + if (oci_lobgetlen(descr->conn,descr,&curloblen) != 0) { + RETURN_FALSE; + } + + if (zend_get_parameters_ex(1, &flag) == SUCCESS) { + convert_to_boolean_ex(flag); + buffering_flag = Z_LVAL_PP(flag); + } else { + WRONG_PARAM_COUNT; + } + + /* we'll return true if function was called twice with the same parameter */ + if (buffering_flag == 0 && descr->buffering == 0) { + RETURN_TRUE; + } else if (buffering_flag == 1 && descr->buffering > 0) { + RETURN_TRUE; + } + + switch (buffering_flag) { + case 0: + CALL_OCI_RETURN(connection->error, + OCILobDisableBuffering( + connection->pServiceContext, + connection->pError, + mylob + ) + ); + break; + case 1: + CALL_OCI_RETURN(connection->error, + OCILobEnableBuffering( + connection->pServiceContext, + connection->pError, + mylob + ) + ); + break; + } + + oci_debug("oci_lob_set_buffering: buffering_flag=%d",buffering_flag); + + if (connection->error) { + oci_error(connection->pError, "OCILobFlushBuffer", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + descr->buffering = buffering_flag; + RETURN_TRUE; + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "OCISetBufferingLob() should not be called like this. Use $somelob->setBuffering($flag) to set buffering on/off for a LOB"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool ocigetbufferinglob() + Returns current state of buffering for a LOB */ +PHP_FUNCTION(ocigetbufferinglob) +{ + zval *id; + OCILobLocator *mylob; + oci_descriptor *descr; + int inx; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + IS_LOB_INTERNAL(descr); + + mylob = (OCILobLocator *) descr->ocidescr; + + if (!mylob) { + RETURN_FALSE; + } + + switch (descr->buffering) { + case 1: + case 2: + RETURN_TRUE; + break; + + case 0: + default: + RETURN_FALSE; + break; + } + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "OCIGetBufferingLob() should not be called like this. Use $somelob->getBuffering() to get current state of buffering for a LOB"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool oci_lob_copy( object lob_to, object lob_from [, int length ] ) + Copies data from a LOB to another LOB */ +PHP_FUNCTION(oci_lob_copy) +{ + zval **arg1, **arg2, **arg3; + OCILobLocator *mylob,*my_fromlob; + oci_connection *connection; + oci_descriptor *descr,*from_descr; + int inx, ac = ZEND_NUM_ARGS(); + ub4 curloblen,from_curloblen, copylen; + + if (ac < 2 || ac > 3 || zend_get_parameters_ex(ac, &arg1, &arg2, &arg3) == FAILURE) { + WRONG_PARAM_COUNT; + } + + convert_to_object_ex(arg1); + convert_to_object_ex(arg2); + + if ((inx = _oci_get_ocidesc(*arg1,&descr TSRMLS_CC)) == 0 || (inx = _oci_get_ocidesc(*arg2,&from_descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + IS_LOB_INTERNAL(descr); + IS_LOB_INTERNAL(from_descr); + + mylob = (OCILobLocator *) descr->ocidescr; + my_fromlob = (OCILobLocator *) from_descr->ocidescr; + + if (!mylob || !my_fromlob) { + RETURN_FALSE; + } + + if (oci_lobgetlen(descr->conn,descr,&curloblen) != 0 || oci_lobgetlen(from_descr->conn,from_descr,&from_curloblen) != 0) { + RETURN_FALSE; + } + + if (ac == 3) { + convert_to_long_ex(arg3); + copylen = Z_LVAL_PP(arg3); + } else { + copylen = from_descr->lob_size - from_descr->lob_current_position; + } + + if ((int)copylen <= 0) { + RETURN_FALSE; + } + + connection = descr->conn; + + CALL_OCI_RETURN(connection->error, + OCILobCopy( + connection->pServiceContext, + connection->pError, + mylob, + my_fromlob, + copylen, + descr->lob_current_position+1, + from_descr->lob_current_position+1 + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobCopy", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + RETURN_TRUE; +} +/* }}} */ + +/* {{{ proto bool oci_lob_is_equal( object lob1, object lob2 ) + Tests to see if two LOB/FILE locators are equal */ +PHP_FUNCTION(oci_lob_is_equal) +{ + zval **arg1, **arg2; + OCILobLocator *first_lob,*second_lob; + oci_connection *connection; + oci_descriptor *first_descr,*second_descr; + int inx; + boolean is_equal; + + if (zend_get_parameters_ex(2, &arg1, &arg2) == FAILURE) { + WRONG_PARAM_COUNT; + } + + convert_to_object_ex(arg1); + convert_to_object_ex(arg2); + + if ((inx = _oci_get_ocidesc(*arg1,&first_descr TSRMLS_CC)) == 0 || (inx = _oci_get_ocidesc(*arg2,&second_descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + first_lob = (OCILobLocator *) first_descr->ocidescr; + second_lob = (OCILobLocator *) second_descr->ocidescr; + + if (!first_lob || !second_lob) { + RETURN_FALSE; + } + + connection = first_descr->conn; + + CALL_OCI_RETURN(connection->error, + OCILobIsEqual( + connection->session->pEnv, + first_lob, + second_lob, + &is_equal + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobIsEqual", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + if (is_equal == TRUE) { + RETURN_TRUE; + } else { + RETURN_FALSE; + } +} +/* }}} */ + +/* {{{ proto bool oci_lob_export([string filename [, int start [, int length]]]) + Writes a large object into a file */ +PHP_FUNCTION(oci_lob_export) +{ + zval *id, **zfilename, **zstart, **zlength; + char *filename = NULL; + int start = -1; + ub4 length = -1; + oci_connection *connection; + oci_descriptor *descr; + char *buffer=0; + ub4 loblen; + int ac = ZEND_NUM_ARGS(); + int fp = -1,inx; + OCILobLocator *mylob; + int coffs; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocidesc(id,&descr TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + mylob = (OCILobLocator *) descr->ocidescr; + + if (!mylob) { + RETURN_FALSE; + } + + connection = descr->conn; + + if (ac < 0 || ac > 3 || zend_get_parameters_ex(ac, &zfilename, &zstart, &zlength) == FAILURE) { + WRONG_PARAM_COUNT; + } + + switch (ac) { + case 3: + convert_to_long_ex(zlength); + length = Z_LVAL_PP(zlength); + case 2: + convert_to_long_ex(zstart); + start = Z_LVAL_PP(zstart); + case 1: + convert_to_string_ex(zfilename); + filename = Z_STRVAL_PP(zfilename); + } + + if (filename && *filename) { + if (php_check_open_basedir(filename TSRMLS_CC)) { + goto bail; + } + + if ((fp = VCWD_OPEN_MODE(filename,O_CREAT | O_RDWR | O_BINARY | O_TRUNC, 0600)) == -1) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "Can't create file %s", filename); + goto bail; + } + } + + CALL_OCI_RETURN(connection->error, + OCILobGetLength( + connection->pServiceContext, + connection->pError, + descr->ocidescr, + &loblen + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobGetLength", connection->error); + oci_handle_error(connection, connection->error); + goto bail; + } + + if (descr->type == OCI_DTYPE_FILE) { + CALL_OCI_RETURN(connection->error, + OCILobFileOpen( + connection->pServiceContext, + connection->pError, + descr->ocidescr, + OCI_FILE_READONLY + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobFileOpen",connection->error); + oci_handle_error(connection, connection->error); + goto bail; + } + } + + if (start == -1) { + start = 0; + } + + if (length == -1) { + length = loblen - start; + } + + if ((start + length) > loblen) { + length = loblen - start; + } + +#define OCI_LOB_READ_BUFFER 128*1024 + + buffer = emalloc(OCI_LOB_READ_BUFFER); + + coffs = start; + + oci_debug("oci_lob_export(start = %d, length = %d, loblen = %d",start,length,loblen); + + while (length > 0) { + ub4 toread; + + if (length > OCI_LOB_READ_BUFFER) { + toread = OCI_LOB_READ_BUFFER; + } else { + toread = length; + } + + oci_debug("oci_lob_read(coffs = %d, toread = %d",coffs,toread); + + CALL_OCI_RETURN(connection->error, + OCILobRead( + connection->pServiceContext, + connection->pError, + descr->ocidescr, + &toread, /* IN/OUT bytes toread/read */ + coffs+1, /* offset (starts with 1) */ + (dvoid *) buffer, + toread, /* size of buffer */ + (dvoid *)0, + (OCICallbackLobRead) 0, /* callback... */ + (ub2) 0, /* The character set ID of the buffer data. */ + (ub1) SQLCS_IMPLICIT /* The character set form of the buffer data. */ + ) + ); + + oci_debug("oci_lob_read(read - %d",toread); + + if (connection->error) { + oci_error(connection->pError, "OCILobRead", connection->error); + oci_handle_error(connection, connection->error); + goto bail; + } + + if (fp != -1) { + if ((ub4) write(fp,buffer,toread) != toread) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "cannot write file!"); + goto bail; + } + } else { + PHPWRITE(buffer,toread); + } + + length -= toread; + coffs += toread; + } + + efree(buffer); + buffer = 0; + + if (fp != -1) { + close(fp); + fp = 0; + } + + if (descr->type == OCI_DTYPE_FILE) { + CALL_OCI_RETURN(connection->error, + OCILobFileClose( + connection->pServiceContext, + connection->pError, + descr->ocidescr + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobFileClose", connection->error); + oci_handle_error(connection, connection->error); + goto bail; + } + } + RETURN_TRUE; + } + +bail: + if (fp != -1) { + close(fp); + } + + if (buffer) { + efree(buffer); + } + + RETURN_FALSE; +} +/* }}} */ + +#ifdef HAVE_OCI8_TEMP_LOB +/* {{{ proto bool oci_lob_write_temporary(string var [, int lob_type]) + Writes temporary blob */ +PHP_FUNCTION(oci_lob_write_temporary) +{ + zval *id, *var; + OCILobLocator *mylob; + oci_connection *connection; + oci_descriptor *descr; + ub4 offset = 1; + ub4 loblen; + long lob_type = OCI_TEMP_CLOB; + + oci_debug ("oci_write_temporary_lob"); + + if ((id = getThis()) == 0) { + RETURN_FALSE; + } + + if (_oci_get_ocidesc(id,&descr TSRMLS_CC) == 0) { + RETURN_FALSE; + } + + mylob = (OCILobLocator *) descr->ocidescr; + + if (!mylob) { + RETURN_FALSE; + } + + connection = descr->conn; + + if (ZEND_NUM_ARGS() < 1) WRONG_PARAM_COUNT; + if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z|l", &var, &lob_type) == FAILURE) { + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCILobCreateTemporary( + connection->pServiceContext, + connection->pError, + mylob, + OCI_DEFAULT, + OCI_DEFAULT, + lob_type, + OCI_ATTR_NOCACHE, + OCI_DURATION_SESSION + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobCreateTemporary", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCILobOpen( + connection->pServiceContext, + connection->pError, + mylob, + OCI_LOB_READWRITE + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobOpen", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + convert_to_string_ex(&var); + loblen = Z_STRLEN_P(var); + + if (loblen < 1) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot save a lob that is less than 1 byte"); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCILobWrite( + connection->pServiceContext, + connection->pError, + mylob, + (ub4 *) &loblen, + (ub4) offset, + (dvoid *) Z_STRVAL_P(var), + (ub4) loblen, + OCI_ONE_PIECE, + (dvoid *)0, + (sb4 (*)(dvoid *, dvoid *, ub4 *, ub1 *)) 0, + (ub2) 0, + (ub1) SQLCS_IMPLICIT + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobWrite", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + RETURN_TRUE; +} +/* }}} */ + +/* {{{ proto bool oci_lob_close() + Closes lob descriptor */ +PHP_FUNCTION(oci_lob_close) +{ + zval *id; + int inx; + OCILobLocator *mylob; + oci_connection *connection; + oci_descriptor *descriptor; + int is_temporary; + + if ((id = getThis()) != 0) { + inx = _oci_get_ocidesc(id,&descriptor TSRMLS_CC); + if (inx) { + + mylob = (OCILobLocator *) descriptor->ocidescr; + + if (!mylob) { + RETURN_FALSE; + } + + connection = descriptor->conn; + + CALL_OCI_RETURN(connection->error, + OCILobClose( + connection->pServiceContext, + connection->pError, + mylob + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCILobClose", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + connection->error = + OCILobIsTemporary(connection->session->pEnv, + connection->pError, + mylob, + &is_temporary); + if (is_temporary) { + connection->error = + OCILobFreeTemporary(connection->pServiceContext, + connection->pError, + mylob); + + if (connection->error) { + oci_error(connection->pError, "OCILobFreeTemporary", + connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + oci_debug("oci_lob_free_temporary: descr=%d",inx); + } + + oci_debug("oci_close_lob: descr=%d",inx); + RETURN_TRUE; + } + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_lob_close() should not be called like this. Use $somelob->close() to close a LOB"); + + RETURN_FALSE; +} +/* }}} */ +#endif + +/* {{{ proto object oci_new_descriptor(resource connection [, int type]) + Initialize a new empty descriptor LOB/FILE (LOB is default) */ +PHP_FUNCTION(oci_new_descriptor) +{ + zval **conn, **type; + oci_connection *connection; + oci_descriptor *descr; + int dtype; + + dtype = OCI_DTYPE_LOB; + + if (zend_get_parameters_ex(2, &conn, &type) == SUCCESS) { + convert_to_long_ex(type); + dtype = Z_LVAL_PP(type); + } else if (zend_get_parameters_ex(1, &conn) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_CONN(connection,conn); + + descr = oci_new_desc(dtype,connection); + + if (!descr) { + RETURN_NULL(); + } + + object_init_ex(return_value, oci_lob_class_entry_ptr); + add_property_resource(return_value, "descriptor", descr->id); +} +/* }}} */ + +/* {{{ proto bool oci_rollback(resource conn) + Rollback the current context */ +PHP_FUNCTION(oci_rollback) +{ + zval **conn; + oci_connection *connection; + + if (zend_get_parameters_ex(1, &conn) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_CONN(connection,conn); + + if (connection->descriptors) { + zend_hash_apply(connection->descriptors,(apply_func_t) _oci_desc_flush_hash_dtor TSRMLS_CC); + } + + oci_debug("<OCITransRollback"); + + CALL_OCI_RETURN(connection->error, + OCITransRollback( + connection->pServiceContext, + connection->pError, + (ub4) 0 + ) + ); + + connection->needs_commit = 0; + + oci_debug(">OCITransRollback"); + + if (connection->error) { + oci_error(connection->pError, "OCIRollback", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + RETURN_TRUE; +} +/* }}} */ + +/* {{{ proto bool oci_commit(resource conn) + Commit the current context */ +PHP_FUNCTION(oci_commit) +{ + zval **conn; + oci_connection *connection; + + if (zend_get_parameters_ex(1, &conn) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_CONN(connection,conn); + + if (connection->descriptors) { + zend_hash_apply(connection->descriptors,(apply_func_t) _oci_desc_flush_hash_dtor TSRMLS_CC); + } + + oci_debug("<OCITransCommit"); + + CALL_OCI_RETURN(connection->error, + OCITransCommit( + connection->pServiceContext, + connection->pError, + (ub4) 0 + ) + ); + + connection->needs_commit = 0; + + oci_debug(">OCITransCommit"); + + if (connection->error) { + oci_error(connection->pError, "OCICommit", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + RETURN_TRUE; +} +/* }}} */ + +/* {{{ proto string oci_field_name(resource stmt, int col) + Tell the name of a column */ +PHP_FUNCTION(oci_field_name) +{ + zval **stmt, **col; + oci_statement *statement; + oci_out_column *outcol; + + if (zend_get_parameters_ex(2, &stmt, &col) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + outcol = oci_get_col(statement, -1, col); + if (outcol == NULL) { + RETURN_FALSE; + } + + RETURN_STRINGL(outcol->name, outcol->name_len, 1); +} +/* }}} */ + +/* {{{ proto int oci_field_size(resource stmt, int col) + Tell the maximum data size of a column */ +PHP_FUNCTION(oci_field_size) +{ + zval **stmt, **col; + oci_statement *statement; + oci_out_column *outcol; + + if (zend_get_parameters_ex(2, &stmt, &col) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + outcol = oci_get_col(statement, -1, col); + if (outcol == NULL) { + RETURN_FALSE; + } + + oci_debug("oci_field_size: %16s, retlen = %4d, retlen4 = %d, data_size = %4d, storage_size4 = %4d, indicator %4d, retcode = %4d", + outcol->name,outcol->retlen,outcol->retlen4,outcol->data_size,outcol->storage_size4,outcol->indicator,outcol->retcode); + + /* Handle data type of LONG */ + if (outcol->data_type == SQLT_LNG){ + RETURN_LONG(outcol->storage_size4); + } else { + RETURN_LONG(outcol->data_size); + } +} +/* }}} */ + +/* {{{ proto int oci_field_scale(resource stmt, int col) + Tell the scale of a column */ +PHP_FUNCTION(oci_field_scale) +{ + zval **stmt, **col; + oci_statement *statement; + oci_out_column *outcol; + + if (zend_get_parameters_ex(2, &stmt, &col) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + outcol = oci_get_col(statement, -1, col); + if (outcol == NULL) { + RETURN_FALSE; + } + RETURN_LONG(outcol->scale); +} +/* }}} */ + +/* {{{ proto int oci_field_precision(resource stmt, int col) + Tell the precision of a column */ +PHP_FUNCTION(oci_field_precision) +{ + zval **stmt, **col; + oci_statement *statement; + oci_out_column *outcol; + + if (zend_get_parameters_ex(2, &stmt, &col) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + outcol = oci_get_col(statement, -1, col); + if (outcol == NULL) { + RETURN_FALSE; + } + RETURN_LONG(outcol->precision); +} +/* }}} */ + +/* {{{ proto mixed oci_field_type(resource stmt, int col) + Tell the data type of a column */ +PHP_FUNCTION(oci_field_type) +{ + zval **stmt, **col; + oci_statement *statement; + oci_out_column *outcol; + + if (zend_get_parameters_ex(2, &stmt, &col) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + outcol = oci_get_col(statement, -1, col); + if (outcol == NULL) { + RETURN_FALSE; + } + switch (outcol->data_type) { +#ifdef SQLT_TIMESTAMP + case SQLT_TIMESTAMP: + RETVAL_STRING("TIMESTAMP",1); + break; +#endif +#ifdef SQLT_TIMESTAMP_TZ + case SQLT_TIMESTAMP_TZ: + RETVAL_STRING("TIMESTAMP_TZ",1); + break; +#endif + case SQLT_DAT: + RETVAL_STRING("DATE",1); + break; + case SQLT_NUM: + RETVAL_STRING("NUMBER",1); + break; + case SQLT_LNG: + RETVAL_STRING("LONG",1); + break; + case SQLT_BIN: + RETVAL_STRING("RAW",1); + break; + case SQLT_LBI: + RETVAL_STRING("LONG RAW",1); + break; + case SQLT_CHR: + RETVAL_STRING("VARCHAR",1); + break; + case SQLT_RSET: + RETVAL_STRING("REFCURSOR",1); + break; + case SQLT_AFC: + RETVAL_STRING("CHAR",1); + break; + case SQLT_BLOB: + RETVAL_STRING("BLOB",1); + break; + case SQLT_CLOB: + RETVAL_STRING("CLOB",1); + break; + case SQLT_BFILE: + RETVAL_STRING("BFILE",1); + break; + case SQLT_RDD: + RETVAL_STRING("ROWID",1); + break; + default: + RETVAL_LONG(outcol->data_type); + } +} +/* }}} */ + +/* {{{ proto int oci_field_type_raw(resource stmt, int col) + Tell the raw oracle data type of a column */ +PHP_FUNCTION(oci_field_type_raw) +{ + zval **stmt, **col; + oci_statement *statement; + oci_out_column *outcol; + + if (zend_get_parameters_ex(2, &stmt, &col) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + outcol = oci_get_col(statement, -1, col); + if (outcol == NULL) { + RETURN_FALSE; + } + RETVAL_LONG(outcol->data_type); +} +/* }}} */ + +/* {{{ proto bool oci_field_is_null(resource stmt, int col) + Tell whether a column is NULL */ +PHP_FUNCTION(oci_field_is_null) +{ + zval **stmt, **col; + oci_statement *statement; + oci_out_column *outcol; + + if (zend_get_parameters_ex(2, &stmt, &col) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + outcol = oci_get_col(statement, -1, col); + if (outcol == NULL) { + RETURN_FALSE; + } + if (outcol->indicator == -1) { + RETURN_TRUE; + } else { + RETURN_FALSE; + } +} +/* }}} */ + +/* {{{ proto void oci_internal_debug(int onoff) + Toggle internal debugging output for the OCI extension */ +PHP_FUNCTION(oci_internal_debug) +{ + zval **arg; + + if (zend_get_parameters_ex(1, &arg) == FAILURE) { + WRONG_PARAM_COUNT; + } + convert_to_long_ex(arg); + OCI(debug_mode) = Z_LVAL_PP(arg); +} +/* }}} */ + +/* {{{ proto bool oci_execute(resource stmt [, int mode]) + Execute a parsed statement */ +PHP_FUNCTION(oci_execute) +{ + zval **stmt,**mode; + oci_statement *statement; + ub4 execmode; + + if (zend_get_parameters_ex(2, &stmt, &mode) == SUCCESS) { + convert_to_long_ex(mode); + execmode = Z_LVAL_PP(mode); + } else if (zend_get_parameters_ex(1, &stmt) == SUCCESS) { + execmode = OCI_COMMIT_ON_SUCCESS; + } else { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + if (oci_execute(statement, "OCIExecute",execmode)) { + RETURN_TRUE; + } else { + RETURN_FALSE; + } +} +/* }}} */ + +/* {{{ proto bool oci_cancel(resource stmt) + Cancel reading from a cursor */ +PHP_FUNCTION(oci_cancel) +{ + zval **stmt; + oci_statement *statement; + + if (zend_get_parameters_ex(1, &stmt) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + if (oci_fetch(statement, 0, "OCICancel" TSRMLS_CC)) { + RETURN_TRUE; + } else { + RETURN_FALSE; + } +} +/* }}} */ + +/* {{{ proto bool oci_fetch(resource stmt) + Prepare a new row of data for reading */ +PHP_FUNCTION(oci_fetch) +{ + zval **stmt; + oci_statement *statement; + ub4 nrows = 1; /* only one row at a time is supported for now */ + + if (zend_get_parameters_ex(1, &stmt) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + if (oci_fetch(statement, nrows, "OCIFetch" TSRMLS_CC)) { + RETURN_TRUE; + } else { + RETURN_FALSE; + } +} +/* }}} */ + +/* {{{ proto int ocifetchinto(resource stmt, array &output [, int mode]) + Fetch a row of result data into an array */ +PHP_FUNCTION(ocifetchinto) +{ + php_oci_fetch_row(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0, 3); +} +/* }}} */ + +/* {{{ proto int oci_fetch_all(resource stmt, array &output[, int skip[, int maxrows[, int flags]]]) + Fetch all rows of result data into an array */ +PHP_FUNCTION(oci_fetch_all) +{ + zval **stmt, **array, *element, **zskip, **zmaxrows, **zflags, *tmp; + oci_statement *statement; + oci_out_column **columns; + zval ***outarrs; + ub4 nrows = 1; + int i; + int skip = 0, maxrows = -1; + int flags = 0; + int rows = 0; + int ac = ZEND_NUM_ARGS(); + + if (ac < 2 || ac > 5 || zend_get_parameters_ex(ac, &stmt, &array, &zskip, &zmaxrows, &zflags) == FAILURE) { + WRONG_PARAM_COUNT; + } + + switch (ac) { + case 5: + convert_to_long_ex(zflags); + flags = Z_LVAL_PP(zflags); + case 4: + convert_to_long_ex(zmaxrows); + maxrows = Z_LVAL_PP(zmaxrows); + case 3: + convert_to_long_ex(zskip); + skip = Z_LVAL_PP(zskip); + } + + OCI_GET_STMT(statement,stmt); + + zval_dtor(*array); + array_init(*array); + + while (skip--) { + if (!oci_fetch(statement, nrows, "OCIFetchStatement" TSRMLS_CC)) { + RETURN_LONG(0); + } + } + + if (flags & OCI_FETCHSTATEMENT_BY_ROW) { + columns = safe_emalloc(statement->ncolumns, sizeof(oci_out_column *), 0); + + for (i = 0; i < statement->ncolumns; i++) { + columns[ i ] = oci_get_col(statement, i + 1, 0); + } + + while (oci_fetch(statement, nrows, "OCIFetchStatement" TSRMLS_CC)) { + zval *row; + + MAKE_STD_ZVAL(row); + array_init(row); + + for (i = 0; i < statement->ncolumns; i++) { + MAKE_STD_ZVAL(element); + + _oci_make_zval(element,statement,columns[ i ], "OCIFetchStatement",OCI_RETURN_LOBS TSRMLS_CC); + + if (flags & OCI_NUM) { + zend_hash_next_index_insert(Z_ARRVAL_P(row), &element, sizeof(zval*), NULL); + } else { /* default to ASSOC */ + zend_hash_update(Z_ARRVAL_P(row), + columns[ i ]->name, columns[ i ]->name_len+1, + &element, sizeof(zval*), NULL); + } + } + + zend_hash_next_index_insert(Z_ARRVAL_PP(array), &row, sizeof(zval*), NULL); + + rows++; + + if ((maxrows != -1) && (rows == maxrows)) { + oci_fetch(statement, 0, "OCIFetchStatement" TSRMLS_CC); + break; + } + } + efree(columns); + + } else { /* default to BY_COLUMN */ + columns = safe_emalloc(statement->ncolumns, sizeof(oci_out_column *), 0); + outarrs = safe_emalloc(statement->ncolumns, sizeof(zval*), 0); + + if (flags & OCI_NUM) { + for (i = 0; i < statement->ncolumns; i++) { + columns[ i ] = oci_get_col(statement, i + 1, 0); + + MAKE_STD_ZVAL(tmp); + array_init(tmp); + + zend_hash_next_index_insert(Z_ARRVAL_PP(array), + &tmp, sizeof(zval*), (void **) &(outarrs[ i ])); + } + } else { /* default to ASSOC */ + for (i = 0; i < statement->ncolumns; i++) { + columns[ i ] = oci_get_col(statement, i + 1, 0); + + MAKE_STD_ZVAL(tmp); + array_init(tmp); + + zend_hash_update(Z_ARRVAL_PP(array), + columns[ i ]->name, columns[ i ]->name_len+1, + (void *) &tmp, sizeof(zval*), (void **) &(outarrs[ i ])); + } + } + + while (oci_fetch(statement, nrows, "OCIFetchStatement" TSRMLS_CC)) { + for (i = 0; i < statement->ncolumns; i++) { + MAKE_STD_ZVAL(element); + + _oci_make_zval(element,statement,columns[ i ], "OCIFetchStatement",OCI_RETURN_LOBS TSRMLS_CC); + + zend_hash_index_update((*(outarrs[ i ]))->value.ht, rows, (void *)&element, sizeof(zval*), NULL); + } + + rows++; + + if ((maxrows != -1) && (rows == maxrows)) { + oci_fetch(statement, 0, "OCIFetchStatement" TSRMLS_CC); + break; + } + } + + efree(columns); + efree(outarrs); + } + + RETURN_LONG(rows); +} +/* }}} */ + +/* {{{ proto object oci_fetch_object( resource stmt ) + Fetch a result row as an object */ +PHP_FUNCTION(oci_fetch_object) +{ + php_oci_fetch_row(INTERNAL_FUNCTION_PARAM_PASSTHRU, OCI_ASSOC, 2); + + if (Z_TYPE_P(return_value) == IS_ARRAY) { + object_and_properties_init(return_value, ZEND_STANDARD_CLASS_DEF_PTR, Z_ARRVAL_P(return_value)); + } +} +/* }}} */ + +/* {{{ proto array oci_fetch_row( resource stmt ) + Fetch a result row as an enumerated array */ +PHP_FUNCTION(oci_fetch_row) +{ + php_oci_fetch_row(INTERNAL_FUNCTION_PARAM_PASSTHRU, OCI_NUM, 1); +} +/* }}} */ + +/* {{{ proto array oci_fetch_assoc( resource stmt ) + Fetch a result row as an associative array */ +PHP_FUNCTION(oci_fetch_assoc) +{ + php_oci_fetch_row(INTERNAL_FUNCTION_PARAM_PASSTHRU, OCI_ASSOC, 1); +} +/* }}} */ + +/* {{{ proto array oci_fetch_array( resource stmt [, int mode ]) + Fetch a result row as an array */ +PHP_FUNCTION(oci_fetch_array) +{ + php_oci_fetch_row(INTERNAL_FUNCTION_PARAM_PASSTHRU, OCI_BOTH, 2); +} +/* }}} */ + +/* {{{ proto bool oci_free_statement(resource stmt) + Free all resources associated with a statement */ +PHP_FUNCTION(oci_free_statement) +{ + zval **stmt; + oci_statement *statement; + + if (zend_get_parameters_ex(1, &stmt) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + zend_list_delete(statement->id); + + RETURN_TRUE; +} +/* }}} */ + +/* {{{ proto bool oci_close(resource conn) + Disconnect from database */ +PHP_FUNCTION(oci_close) +{ +#if 0 + this function does nothing any more. server-connections get automagiclly closed on + request-end. connection handles will "dissappear" as soon as they are no longer + referenced. as this module makes heavy use of zends reference-counting mechanism + this is the desired behavior. it has always been a bad idea to close a connection that + has outstanding transactions. this way we have a nice-clean approach. + (thies@thieso.net 20000110) + + oci_connection *connection; + zval **conn; + + if (zend_get_parameters_ex(1, &conn) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_CONN(connection,conn); + + connection->is_open = 0; + + zend_hash_apply(list, (apply_func_t) _stmt_cleanup TSRMLS_CC); + + if (zend_list_delete(connection->id) == SUCCESS) { + RETURN_TRUE; + } else { + RETURN_FALSE; + } +#endif +} +/* }}} */ + +/* {{{ proto resource oci_new_connect(string user, string pass [, string db]) + Connect to an Oracle database and log on. Returns a new session. */ +PHP_FUNCTION(oci_new_connect) +{ + oci_do_connect(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0, 1); +} +/* }}} */ + +/* {{{ proto resource oci_connect(string user, string pass [, string db]) + Connect to an Oracle database and log on. Returns a new session. */ +PHP_FUNCTION(oci_connect) +{ + oci_do_connect(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0, 1); +} +/* }}} */ + +/* {{{ proto resource oci_pconnect(string user, string pass [, string db]) + Connect to an Oracle database using a persistent connection and log on. Returns a new session. */ +PHP_FUNCTION(oci_pconnect) +{ + oci_do_connect(INTERNAL_FUNCTION_PARAM_PASSTHRU,1,0); +} +/* }}} */ + +/* {{{ proto array oci_error([resource stmt|conn|global]) + Return the last error of stmt|conn|global. If no error happened returns false. */ +PHP_FUNCTION(oci_error) +{ + zval **arg; + oci_statement *statement; + oci_connection *connection; + text errbuf[512]; + sb4 errcode = 0; + sword error = 0; + dvoid *errh = NULL; +#ifdef HAVE_OCI8_ATTR_STATEMENT + ub2 errorofs = 0; + text *sqltext = NULL; +#endif + + if (zend_get_parameters_ex(1, &arg) == SUCCESS) { + statement = (oci_statement *) zend_fetch_resource(arg TSRMLS_CC, -1, NULL, NULL, 1, le_stmt); + if (statement) { + errh = statement->pError; + error = statement->error; + +#ifdef HAVE_OCI8_ATTR_STATEMENT + CALL_OCI_RETURN(statement->error, + OCIAttrGet( + (dvoid *)statement->pStmt, + OCI_HTYPE_STMT, + (text *) &sqltext, + (ub4 *)0, + OCI_ATTR_STATEMENT, + statement->pError + ) + ); + + CALL_OCI_RETURN(statement->error, + OCIAttrGet( + (dvoid *)statement->pStmt, + OCI_HTYPE_STMT, + (ub2 *)&errorofs, + (ub4 *)0, + OCI_ATTR_PARSE_ERROR_OFFSET, + statement->pError + ) + ); +#endif + + } else { + connection = (oci_connection *) zend_fetch_resource(arg TSRMLS_CC, -1, NULL, NULL, 1, le_conn); + if (connection) { + errh = connection->pError; + error = connection->error; + } + } + } else { + errh = OCI(pError); + error = OCI(error); + } + + if (!error) { /* no error set in the handle */ + RETURN_FALSE; + } + + if (!errh) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "OCIError: unable to find Error handle"); + RETURN_FALSE; + } + + CALL_OCI( + OCIErrorGet( + errh, + 1, + NULL, + &errcode, + errbuf, + (ub4) sizeof(errbuf), + (ub4) OCI_HTYPE_ERROR + ) + ); + + if (errcode) { + array_init(return_value); + add_assoc_long(return_value, "code", errcode); + add_assoc_string(return_value, "message", (char*) errbuf, 1); +#ifdef HAVE_OCI8_ATTR_STATEMENT + add_assoc_long(return_value, "offset", errorofs); + add_assoc_string(return_value, "sqltext", sqltext ? (char *) sqltext : "", 1); +#endif + } else { + RETURN_FALSE; + } +} +/* }}} */ + +/* {{{ proto int oci_num_fields(resource stmt) + Return the number of result columns in a statement */ +PHP_FUNCTION(oci_num_fields) +{ + zval **stmt; + oci_statement *statement; + + if (zend_get_parameters_ex(1, &stmt) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + RETURN_LONG(statement->ncolumns); +} +/* }}} */ + +/* {{{ proto resource oci_parse(resource conn, string query) + Parse a query and return a statement */ +PHP_FUNCTION(oci_parse) +{ + zval **conn, **query; + oci_connection *connection; + oci_statement *statement; + + if (zend_get_parameters_ex(2, &conn, &query) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_CONN(connection,conn); + + convert_to_string_ex(query); + + statement = oci_parse(connection,Z_STRVAL_PP(query),Z_STRLEN_PP(query)); + + if (statement) { + RETURN_RESOURCE(statement->id); + } else { + RETURN_FALSE; + } +} +/* }}} */ + +/* {{{ proto bool oci_set_prefetch(resource stmt, int prefetch_rows) + Sets the number of rows to be prefetched on execute to prefetch_rows for stmt */ +PHP_FUNCTION(oci_set_prefetch) +{ + zval **stmt, **size; + oci_statement *statement; + + if (zend_get_parameters_ex(2, &stmt, &size) == FAILURE) { + WRONG_PARAM_COUNT; + } + + convert_to_long_ex(size); + + OCI_GET_STMT(statement,stmt); + + oci_setprefetch(statement,Z_LVAL_PP(size)); + + RETURN_TRUE; +} +/* }}} */ + +/* {{{ proto bool oci_password_change(resource conn, string username, string old_password, string new_password) + Changes the password of an account */ +PHP_FUNCTION(oci_password_change) +{ + zval **conn, **user_param, **pass_old_param, **pass_new_param; + text *user, *pass_old, *pass_new; + oci_connection *connection; + + /* Disable in Safe Mode */ + if (PG(safe_mode)) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "is disabled in Safe Mode"); + RETURN_FALSE; + } + + if (zend_get_parameters_ex(4, &conn, &user_param, &pass_old_param, &pass_new_param) == FAILURE) { + WRONG_PARAM_COUNT; + } + + convert_to_string_ex(user_param); + convert_to_string_ex(pass_old_param); + convert_to_string_ex(pass_new_param); + + user = Z_STRVAL_PP(user_param); + pass_old = Z_STRVAL_PP(pass_old_param); + pass_new = Z_STRVAL_PP(pass_new_param); + + OCI_GET_CONN(connection, conn); + + CALL_OCI_RETURN(connection->error, + OCIPasswordChange( + connection->pServiceContext, + connection->pError, + user, + strlen(user)+1, + pass_old, + strlen(pass_old)+1, + pass_new, + strlen(pass_new)+1, + OCI_DEFAULT + ) + ); + + if (connection->error == OCI_SUCCESS) { + RETURN_TRUE; + } else { + oci_error(connection->pError, "OCIPasswordChange", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } +} +/* }}} */ + +/* {{{ proto resource oci_new_cursor(resource conn) + Return a new cursor (Statement-Handle) - use this to bind ref-cursors! */ +PHP_FUNCTION(oci_new_cursor) +{ + zval **conn; + oci_connection *connection; + oci_statement *statement; + + if (zend_get_parameters_ex(1, &conn) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_CONN(connection,conn); + + statement = oci_parse(connection,0,0); + + RETURN_RESOURCE(statement->id); +} +/* }}} */ + +/* {{{ proto string oci_result(resource stmt, mixed column) + Return a single column of result data */ +PHP_FUNCTION(oci_result) +{ + zval **stmt, **col; + oci_statement *statement; + oci_out_column *outcol = NULL; + + if (zend_get_parameters_ex(2, &stmt, &col) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + outcol = oci_get_col(statement, -1, col); + + if (outcol == NULL) { + RETURN_FALSE; + } + + _oci_make_zval(return_value,statement,outcol, "OCIResult",0 TSRMLS_CC); +} +/* }}} */ + +/* {{{ proto string oci_server_version(resource conn) + Return a string containing server version information */ +PHP_FUNCTION(oci_server_version) +{ + oci_connection *connection; + zval **conn; + char version[256]; + + if (zend_get_parameters_ex(1, &conn) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_CONN(connection,conn); + + CALL_OCI_RETURN(connection->error, + OCIServerVersion( + connection->pServiceContext, + connection->pError, + (text*)version, + sizeof(version), + OCI_HTYPE_SVCCTX + ) + ); + + if (connection->error != OCI_SUCCESS) { + oci_error(connection->pError, "OCIServerVersion", connection->error); + oci_handle_error(connection, connection->error); + RETURN_FALSE; + } + + RETURN_STRING(version,1); +} +/* }}} */ + +/* {{{ proto string oci_statement_type(resource stmt) + Return the query type of an OCI statement */ +/* XXX it would be better with a general interface to OCIAttrGet() */ +PHP_FUNCTION(oci_statement_type) +{ + zval **stmt; + oci_statement *statement; + ub2 stmttype; + + if (zend_get_parameters_ex(1, &stmt) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + CALL_OCI_RETURN(statement->error, + OCIAttrGet( + (dvoid *)statement->pStmt, + OCI_HTYPE_STMT, + (ub2 *)&stmttype, + (ub4 *)0, + OCI_ATTR_STMT_TYPE, + statement->pError + ) + ); + + if (statement->error != OCI_SUCCESS) { + oci_error(statement->pError, "OCIStatementType", statement->error); + oci_handle_error(statement->conn, statement->error); + RETURN_FALSE; + } + + switch (stmttype) { + case OCI_STMT_SELECT: + RETVAL_STRING("SELECT",1); + break; + case OCI_STMT_UPDATE: + RETVAL_STRING("UPDATE",1); + break; + case OCI_STMT_DELETE: + RETVAL_STRING("DELETE",1); + break; + case OCI_STMT_INSERT: + RETVAL_STRING("INSERT",1); + break; + case OCI_STMT_CREATE: + RETVAL_STRING("CREATE",1); + break; + case OCI_STMT_DROP: + RETVAL_STRING("DROP",1); + break; + case OCI_STMT_ALTER: + RETVAL_STRING("ALTER",1); + break; + case OCI_STMT_BEGIN: + RETVAL_STRING("BEGIN",1); + break; + case OCI_STMT_DECLARE: + RETVAL_STRING("DECLARE",1); + break; + default: + RETVAL_STRING("UNKNOWN",1); + } +} +/* }}} */ + +/* {{{ proto int oci_num_rows(resource stmt) + Return the row count of an OCI statement */ +PHP_FUNCTION(oci_num_rows) +{ + zval **stmt; + oci_statement *statement; + ub4 rowcount; + + if (zend_get_parameters_ex(1, &stmt) == FAILURE) { + WRONG_PARAM_COUNT; + } + + OCI_GET_STMT(statement,stmt); + + CALL_OCI_RETURN(statement->error, + OCIAttrGet( + (dvoid *)statement->pStmt, + OCI_HTYPE_STMT, + (ub2 *)&rowcount, + (ub4 *)0, + OCI_ATTR_ROW_COUNT, + statement->pError + ) + ); + + if (statement->error != OCI_SUCCESS) { + oci_error(statement->pError, "OCIRowCount", statement->error); + oci_handle_error(statement->conn, statement->error); + RETURN_FALSE; + } + + RETURN_LONG(rowcount); +} +/* }}} */ + +#ifdef PHP_OCI8_HAVE_COLLECTIONS +/* {{{ oci_get_coll() */ +static oci_collection *oci_get_coll(int ind TSRMLS_DC) +{ + oci_collection *collection; + int actual_resource_type; + + collection = (oci_collection *) zend_list_find(ind, &actual_resource_type); + + if (collection && (actual_resource_type == le_coll)) { + return collection; + } else { + return (oci_collection *) NULL; + } +} +/* }}} */ + +/* {{{ proto bool oci_free_collection() + Deletes collection object*/ +PHP_FUNCTION(oci_free_collection) +{ + zval *id; + int inx; + oci_collection *coll; + oci_connection *connection; + + if ((id = getThis()) != 0) { + inx = _oci_get_ocicoll(id,&coll TSRMLS_CC); + if (inx) { + /* + * Do we need to free the object? + * + */ + connection = coll->conn; + oci_debug("oci_free_collection: coll=%d",inx); + + CALL_OCI_RETURN(connection->error, + OCIObjectFree( + connection->session->pEnv, + connection->pError, + (dvoid *)coll->coll, + (ub2)(OCI_OBJECTFREE_FORCE) + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCIObjectFree", connection->error); + RETURN_FALSE; + } + + zend_list_delete(inx); + RETURN_TRUE; + } + } + + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool oci_collection_append(string value) + Append an object to the collection */ +PHP_FUNCTION(oci_collection_append) +{ + zval *id, **arg; + oci_connection *connection; + oci_collection *coll; + OCINumber num; + OCIString *ocistr = (OCIString *)0; + OCIInd new_ind = OCI_IND_NOTNULL; + OCIInd null_ind = OCI_IND_NULL; + OCIDate dt; + int inx; + double ndx; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocicoll(id,&coll TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + connection = coll->conn; + if (zend_get_parameters_ex(1, &arg) == FAILURE) { + WRONG_PARAM_COUNT; + } + + /* + * Handle NULLS. For consistency with the rest of the OCI8 library, when + * a value passed in is a 0 length string, consider it a null + */ + convert_to_string_ex(arg); + if (Z_STRLEN_PP(arg) == 0) { + CALL_OCI_RETURN(connection->error, + OCICollAppend( + connection->session->pEnv, + connection->pError, + (dvoid *)0, + &null_ind, + coll->coll + ) + ); + if (connection->error) { + oci_error(connection->pError, "OCICollAppend - NULL", connection->error); + RETURN_FALSE; + } + + RETURN_TRUE; + } + + switch(coll->element_typecode) { + case OCI_TYPECODE_DATE: + convert_to_string_ex(arg); + + CALL_OCI_RETURN(connection->error, + OCIDateFromText( + connection->pError, + Z_STRVAL_PP(arg), + Z_STRLEN_PP(arg), + 0, + 0, + 0, + 0, + &dt + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCIDateFromText", connection->error); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCICollAppend( + connection->session->pEnv, + connection->pError, + (dvoid *) &dt, + (dvoid *) &new_ind, + (OCIColl *) coll->coll + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCICollAppend", connection->error); + RETURN_FALSE; + } + RETURN_TRUE; + break; + case OCI_TYPECODE_VARCHAR2 : + convert_to_string_ex(arg); + + CALL_OCI_RETURN(connection->error, + OCIStringAssignText( + connection->session->pEnv, + connection->pError, + Z_STRVAL_PP(arg), + Z_STRLEN_PP(arg), + &ocistr + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCIStringAssignText", connection->error); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCICollAppend( + connection->session->pEnv, + connection->pError, + (dvoid *) ocistr, + (dvoid *) &new_ind, + (OCIColl *) coll->coll + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCICollAppend", connection->error); + RETURN_FALSE; + } + RETURN_TRUE; + break; + case OCI_TYPECODE_UNSIGNED16 : /* UNSIGNED SHORT */ + case OCI_TYPECODE_UNSIGNED32 : /* UNSIGNED LONG */ + case OCI_TYPECODE_REAL : /* REAL */ + case OCI_TYPECODE_DOUBLE : /* DOUBLE */ + case OCI_TYPECODE_INTEGER : /* INT */ + case OCI_TYPECODE_SIGNED16 : /* SHORT */ + case OCI_TYPECODE_SIGNED32 : /* LONG */ + case OCI_TYPECODE_DECIMAL : /* DECIMAL */ + case OCI_TYPECODE_FLOAT : /* FLOAT */ + case OCI_TYPECODE_NUMBER : /* NUMBER */ + case OCI_TYPECODE_SMALLINT : /* SMALLINT */ + convert_to_double_ex(arg); + ndx = (double)Z_DVAL_PP(arg); + + CALL_OCI_RETURN(connection->error, + OCINumberFromReal( + connection->pError, + &ndx, + sizeof(double), + &num + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCINumberFromReal", connection->error); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCICollAppend( + connection->session->pEnv, + connection->pError, + (dvoid *) &num, + (dvoid *) &new_ind, + (OCIColl *) coll->coll + ) + ); + + RETURN_TRUE; + break; + default: + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "Unknown or unsupported type of element"); + RETURN_FALSE; + break; + } + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_collection_append() should not be called like this. Use $collection->append($element) to append an element to the collection"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto string oci_collection_element_get(int ndx) + Retrieve the value at collection index ndx */ +PHP_FUNCTION(oci_collection_element_get) +{ + zval *id,**arg; + oci_connection *connection; + oci_collection *coll; + ub4 ndx; + int inx; + dvoid *elem; + OCIInd *elemind; + boolean exists; + OCIString *ocistr = (OCIString *)0; + text *str; + char buff[1024]; + int len; + double dnum; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocicoll(id,&coll TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + if (zend_get_parameters_ex(1, &arg) == FAILURE) { + WRONG_PARAM_COUNT; + } + + convert_to_long_ex(arg); + ndx = Z_LVAL_PP(arg); + + connection = coll->conn; + + CALL_OCI_RETURN(connection->error, + OCICollGetElem( + connection->session->pEnv, + connection->pError, + coll->coll, + ndx, + &exists, + &elem, + (dvoid **)&elemind + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCICollGetElem", connection->error); + RETURN_NULL(); + } + + /* Return false if value does not exist at that location */ + if (exists == 0) { + php_error_docref(NULL TSRMLS_CC, E_WARNING, "OCICollGetElem - Invalid index %d", ndx); + RETURN_FALSE; + } + + /* Return null if the value is null */ + if (*elemind == OCI_IND_NULL) { + RETURN_NULL(); + } + + switch (coll->element_typecode) { + case OCI_TYPECODE_DATE: + len = 1024; + CALL_OCI( + OCIDateToText( + connection->pError, + elem, + 0, /* fmt */ + 0, /* fmt_length */ + 0, /* lang_name */ + 0, /* lang_length */ + &len, + buff + ) + ); + + RETURN_STRINGL(buff,len,1); + case OCI_TYPECODE_VARCHAR2 : + ocistr = *(OCIString **)elem; + str = OCIStringPtr(connection->session->pEnv,ocistr); /* XXX not protected against recursion! */ + RETURN_STRINGL(str,strlen(str),1); + break; + case OCI_TYPECODE_UNSIGNED16 : /* UNSIGNED SHORT */ + case OCI_TYPECODE_UNSIGNED32 : /* UNSIGNED LONG */ + case OCI_TYPECODE_REAL : /* REAL */ + case OCI_TYPECODE_DOUBLE : /* DOUBLE */ + case OCI_TYPECODE_INTEGER : /* INT */ + case OCI_TYPECODE_SIGNED16 : /* SHORT */ + case OCI_TYPECODE_SIGNED32 : /* LONG */ + case OCI_TYPECODE_DECIMAL : /* DECIMAL */ + case OCI_TYPECODE_FLOAT : /* FLOAT */ + case OCI_TYPECODE_NUMBER : /* NUMBER */ + case OCI_TYPECODE_SMALLINT : /* SMALLINT */ + CALL_OCI_RETURN(connection->error, + OCINumberToReal( + connection->pError, + (CONST OCINumber *) elem, + (uword) sizeof(dnum), + (dvoid *) &dnum + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCINumberToReal", connection->error); + RETURN_FALSE; + } + RETURN_DOUBLE(dnum); + break; + default: + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "Unknown or unsupported type of element"); + RETURN_FALSE; + break; + } + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_collection_element_get() should not be called like this. Use $collection->getelem($index) to get an element of the collection with the given index"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool oci_collection_assign(object from) + Assign a collection from another existing collection */ +PHP_FUNCTION(oci_collection_assign) +{ + zval *id,**from; + oci_connection *connection; + oci_collection *coll,*from_coll; + int inx; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocicoll(id,&coll TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + if (zend_get_parameters_ex(1, &from) == FAILURE) { + WRONG_PARAM_COUNT; + } + + if ((inx = _oci_get_ocicoll(*from,&from_coll TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + connection = coll->conn; + + CALL_OCI_RETURN(connection->error, + OCICollAssign( + connection->session->pEnv, + connection->pError, + from_coll->coll, + coll->coll + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCICollAssignElem", connection->error); + RETURN_FALSE; + } + RETURN_TRUE; + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_collection_assign() should not be called like this. Use $collection->assign($collection_value) to assign value to the collection"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool oci_collection_element_assign(int index, string val) + Assign element val to collection at index ndx */ +PHP_FUNCTION(oci_collection_element_assign) +{ + zval *id,**index,**val; + oci_connection *connection; + oci_collection *coll; + OCINumber num; + OCIInd new_ind = OCI_IND_NOTNULL; + OCIInd null_ind = OCI_IND_NULL; + ub4 ndx; + int inx; + OCIString *ocistr = (OCIString *)0; + OCIDate dt; + double dnum; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocicoll(id,&coll TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + + if (zend_get_parameters_ex(2, &index,&val) == FAILURE) { + WRONG_PARAM_COUNT; + } + + convert_to_long_ex(index); + ndx = Z_LVAL_PP(index); + + connection = coll->conn; + + if (connection->error) { + oci_error(connection->pError, "OCICollAssignElem", connection->error); + RETURN_FALSE; + } + + /* + * Handle NULLs. For consistency with the rest of the OCI8 library, when + * a value passed in is a 0 length string, consider it a null + */ + convert_to_string_ex(val); + + if (Z_STRLEN_PP(val) == 0) { + CALL_OCI_RETURN(connection->error, + OCICollAssignElem( + connection->session->pEnv, + connection->pError, + ndx, + (dvoid *)0, + &null_ind, + coll->coll + ) + ); + if (connection->error) { + oci_error(connection->pError, "OCICollAssignElem - NULL", connection->error); + RETURN_FALSE; + } + + RETURN_TRUE; + } + + switch (coll->element_typecode) { + case OCI_TYPECODE_DATE: + convert_to_string_ex(val); + CALL_OCI_RETURN(connection->error, + OCIDateFromText( + connection->pError, + Z_STRVAL_PP(val), + Z_STRLEN_PP(val), + 0, + 0, + 0, + 0, + &dt + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCIDateFromText", connection->error); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCICollAssignElem( + connection->session->pEnv, + connection->pError, + ndx, + (dvoid *)&dt, + &new_ind, + coll->coll + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCICollAssignElem", connection->error); + RETURN_FALSE; + } + break; + case OCI_TYPECODE_VARCHAR2 : + convert_to_string_ex(val); + + CALL_OCI_RETURN(connection->error, + OCIStringAssignText( + connection->session->pEnv, + connection->pError, + Z_STRVAL_PP(val), + Z_STRLEN_PP(val), + &ocistr + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCIStringAssignText", connection->error); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCICollAssignElem( + connection->session->pEnv, + connection->pError, + ndx, + (dvoid *)ocistr, + &new_ind, + coll->coll + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCICollAssignElem", connection->error); + RETURN_FALSE; + } + RETURN_TRUE; + break; + case OCI_TYPECODE_UNSIGNED16 : /* UNSIGNED SHORT */ + case OCI_TYPECODE_UNSIGNED32 : /* UNSIGNED LONG */ + case OCI_TYPECODE_REAL : /* REAL */ + case OCI_TYPECODE_DOUBLE : /* DOUBLE */ + case OCI_TYPECODE_INTEGER : /* INT */ + case OCI_TYPECODE_SIGNED16 : /* SHORT */ + case OCI_TYPECODE_SIGNED32 : /* LONG */ + case OCI_TYPECODE_DECIMAL : /* DECIMAL */ + case OCI_TYPECODE_FLOAT : /* FLOAT */ + case OCI_TYPECODE_NUMBER : /* NUMBER */ + case OCI_TYPECODE_SMALLINT : /* SMALLINT */ + convert_to_double_ex(val); + dnum = (double)Z_DVAL_PP(val); + + CALL_OCI_RETURN(connection->error, + OCINumberFromReal( + connection->pError, + &dnum, + sizeof(double), + &num + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCINumberFromReal", connection->error); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCICollAssignElem( + connection->session->pEnv, + connection->pError, + ndx, + (dvoid *)&num, + &new_ind, + coll->coll + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCICollAssignElem", connection->error); + RETURN_FALSE; + } + RETURN_TRUE; + break; + } + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_collection_element_assign() should not be called like this. Use $collection->assignelem($index, $value) to assign value to an element of the collection"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto int oci_collection_size() + Return the size of a collection */ +PHP_FUNCTION(oci_collection_size) +{ + zval *id; + oci_connection *connection; + oci_collection *coll; + sb4 sz; + int inx; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocicoll(id,&coll TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + connection = coll->conn; + + CALL_OCI_RETURN(connection->error, + OCICollSize( + connection->session->pEnv, + coll->conn->pError, + coll->coll, + &sz + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCICollSize", connection->error); + RETURN_FALSE; + } + + RETURN_LONG(sz); + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_collection_size() should not be called like this. Use $collection->size() to get size of the collection"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto int oci_collection_max() + Return the max value of a collection. For a varray this is the maximum length of the array */ +PHP_FUNCTION(oci_collection_max) +{ + zval *id; + oci_collection *coll; + sb4 sz; + int inx; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocicoll(id,&coll TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + sz = OCICollMax(OCI(pEnv),coll->coll); /* XXX not protected against recursion */ + + RETURN_LONG(sz); + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_collection_max() should not be called like this. Use $collection->max() to get maximum number of elements in the collection"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto bool oci_collection_trim(int num) + Trim num elements from the end of a collection */ +PHP_FUNCTION(oci_collection_trim) +{ + zval *id,**arg; + oci_collection *coll; + int inx; + + if ((id = getThis()) != 0) { + if ((inx = _oci_get_ocicoll(id,&coll TSRMLS_CC)) == 0) { + RETURN_FALSE; + } + if (zend_get_parameters_ex(1, &arg) == FAILURE) { + WRONG_PARAM_COUNT; + } + convert_to_long_ex(arg); + + CALL_OCI_RETURN(coll->conn->error, + OCICollTrim( + OCI(pEnv), + coll->conn->pError, + Z_LVAL_PP(arg), + coll->coll + ) + ); + + if (coll->conn->error) { + oci_error(coll->conn->pError, "OCICollTrim", coll->conn->error); + RETURN_FALSE; + } + RETURN_TRUE; + } + + php_error_docref(NULL TSRMLS_CC, E_NOTICE, "oci_collection_trim() should not be called like this. Use $collection->trim($length) to trim collection to the given length"); + RETURN_FALSE; +} +/* }}} */ + +/* {{{ proto object oci_new_collection(resource connection, string tdo [, string schema]) + Initialize a new collection */ +PHP_FUNCTION(oci_new_collection) +{ + dvoid *dschp1; + dvoid *parmp1; + dvoid *parmp2; + zval **conn, **tdo, **schema; + oci_connection *connection; + oci_collection *coll; + int ac = ZEND_NUM_ARGS(); + + if (ac < 2 || ac > 3 || zend_get_parameters_ex(ac, &conn, &tdo, &schema) == FAILURE) { + WRONG_PARAM_COUNT; + } + + convert_to_string_ex(tdo); + + if (ac == 3) { + convert_to_string_ex(schema); + } + + coll = emalloc(sizeof(oci_collection)); + + OCI_GET_CONN(connection,conn); + + coll->conn = connection; + coll->id = zend_list_insert(coll,le_coll); + zend_list_addref(connection->id); + + CALL_OCI_RETURN(connection->error, + OCITypeByName( + connection->session->pEnv, + connection->pError, + connection->pServiceContext, + ac == 3 ? (text *) Z_STRVAL_PP(schema) : (text *) 0, + ac == 3 ? (ub4) Z_STRLEN_PP(schema) : (ub4) 0, + (text *) Z_STRVAL_PP(tdo), + (ub4) Z_STRLEN_PP(tdo), + (CONST text *) 0, + (ub4) 0, + OCI_DURATION_SESSION, + OCI_TYPEGET_ALL, + &(coll->tdo) + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCITypeByName", connection->error); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCIHandleAlloc( + connection->session->pEnv, + (dvoid **) &dschp1, + (ub4) OCI_HTYPE_DESCRIBE, + (size_t) 0, + (dvoid **) 0 + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCI_HTYPE_DESCRIBE", connection->error); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCIDescribeAny( + connection->pServiceContext, + connection->pError, + (dvoid *) coll->tdo, + (ub4) 0, + OCI_OTYPE_PTR, + (ub1)1, + (ub1) OCI_PTYPE_TYPE, + dschp1 + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCI_OTYPE_PTR", connection->error); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCIAttrGet( + (dvoid *) dschp1, + (ub4) OCI_HTYPE_DESCRIBE, + (dvoid *)&parmp1, + (ub4 *)0, + (ub4)OCI_ATTR_PARAM, + connection->pError + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCI_ATTR_PARAM", connection->error); + RETURN_FALSE; + } + + /* get the collection type code of the attribute */ + + CALL_OCI_RETURN(connection->error, + OCIAttrGet( + (dvoid*) parmp1, + (ub4) OCI_DTYPE_PARAM, + (dvoid*) &(coll->coll_typecode), + (ub4 *) 0, + (ub4) OCI_ATTR_COLLECTION_TYPECODE, + connection->pError + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCI_ATTR_COLLECTION_TYPECODE", connection->error); + RETURN_FALSE; + } + + switch(coll->coll_typecode) { + case OCI_TYPECODE_TABLE: + case OCI_TYPECODE_VARRAY: + CALL_OCI_RETURN(connection->error, + OCIAttrGet( + (dvoid*) parmp1, + (ub4) OCI_DTYPE_PARAM, + (dvoid*) &parmp2, + (ub4 *) 0, + (ub4) OCI_ATTR_COLLECTION_ELEMENT, + connection->pError + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCI_ATTR_COLLECTION_ELEMENT", connection->error); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCIAttrGet( + (dvoid*) parmp2, + (ub4) OCI_DTYPE_PARAM, + (dvoid*) &(coll->elem_ref), + (ub4 *) 0, + (ub4) OCI_ATTR_REF_TDO, + connection->pError + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCI_ATTR_REF_TDO", connection->error); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCITypeByRef( + connection->session->pEnv, + connection->pError, + coll->elem_ref, + OCI_DURATION_SESSION, + OCI_TYPEGET_HEADER, + &(coll->element_type) + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCI_TYPEGET_HEADER", connection->error); + RETURN_FALSE; + } + + CALL_OCI_RETURN(connection->error, + OCIAttrGet( + (dvoid*) parmp2, + (ub4) OCI_DTYPE_PARAM, + (dvoid*) &(coll->element_typecode), + (ub4 *) 0, + (ub4) OCI_ATTR_TYPECODE, + connection->pError + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCI_ATTR_TYPECODE", connection->error); + RETURN_FALSE; + } + break; + default: + php_error_docref(NULL TSRMLS_CC, E_WARNING, "OCINewCollection - Unknown Type %d", coll->coll_typecode); + break; + } + + /* Create object to hold return table */ + CALL_OCI_RETURN(connection->error, + OCIObjectNew( + connection->session->pEnv, + connection->pError, + connection->pServiceContext, + OCI_TYPECODE_TABLE, + coll->tdo, + (dvoid *)0, + OCI_DURATION_DEFAULT, + TRUE, + (dvoid **) &(coll->coll) + ) + ); + + if (connection->error) { + oci_error(connection->pError, "OCIObjectNew", connection->error); + RETURN_FALSE; + } + + object_init_ex(return_value, oci_coll_class_entry_ptr); + add_property_resource(return_value, "collection",coll->id); +} +/* }}} */ +#endif + +#endif /* HAVE_OCI8 */ + +/* + * Local variables: + * tab-width: 4 + * c-basic-offset: 4 + * End: + * vim600: sw=4 ts=4 fdm=marker + * vim<600: sw=4 ts=4 + */ diff --git a/ext/oci8/oci8.dsp b/ext/oci8/oci8.dsp new file mode 100644 index 000000000..73c05346e --- /dev/null +++ b/ext/oci8/oci8.dsp @@ -0,0 +1,113 @@ +# Microsoft Developer Studio Project File - Name="oci8" - Package Owner=<4>
+# Microsoft Developer Studio Generated Build File, Format Version 6.00
+# ** DO NOT EDIT **
+
+# TARGTYPE "Win32 (x86) Dynamic-Link Library" 0x0102
+
+CFG=oci8 - Win32 Release_TS
+!MESSAGE This is not a valid makefile. To build this project using NMAKE,
+!MESSAGE use the Export Makefile command and run
+!MESSAGE
+!MESSAGE NMAKE /f "oci8.mak".
+!MESSAGE
+!MESSAGE You can specify a configuration when running NMAKE
+!MESSAGE by defining the macro CFG on the command line. For example:
+!MESSAGE
+!MESSAGE NMAKE /f "oci8.mak" CFG="oci8 - Win32 Release_TS"
+!MESSAGE
+!MESSAGE Possible choices for configuration are:
+!MESSAGE
+!MESSAGE "oci8 - Win32 Release_TS" (based on "Win32 (x86) Dynamic-Link Library")
+!MESSAGE "oci8 - Win32 Debug_TS" (based on "Win32 (x86) Dynamic-Link Library")
+!MESSAGE
+
+# Begin Project
+# PROP AllowPerConfigDependencies 0
+# PROP Scc_ProjName ""
+# PROP Scc_LocalPath ""
+CPP=cl.exe
+MTL=midl.exe
+RSC=rc.exe
+
+!IF "$(CFG)" == "oci8 - Win32 Release_TS"
+
+# PROP BASE Use_MFC 0
+# PROP BASE Use_Debug_Libraries 0
+# PROP BASE Output_Dir "Release_TS"
+# PROP BASE Intermediate_Dir "Release_TS"
+# PROP BASE Ignore_Export_Lib 0
+# PROP BASE Target_Dir ""
+# PROP Use_MFC 0
+# PROP Use_Debug_Libraries 0
+# PROP Output_Dir "Release_TS"
+# PROP Intermediate_Dir "Release_TS"
+# PROP Ignore_Export_Lib 0
+# PROP Target_Dir ""
+# ADD BASE CPP /nologo /MT /W3 /GX /O2 /I "..\.." /I "..\..\..\Zend" /I "..\..\..\bindlib_w32" /I "..\..\..\TSRM" /D "WIN32" /D "NDEBUG" /D "_WINDOWS" /D "_MBCS" /D "_USRDLL" /D ZTS=1 /YX /FD /c
+# ADD CPP /nologo /MD /W3 /GX /O2 /I "..\..\\" /I "..\..\main" /I "..\..\Zend" /I "..\..\..\bindlib_w32" /I "..\..\TSRM" /I "..\..\..\php_build\oci805\include" /D ZEND_DEBUG=0 /D "WIN32" /D "NDEBUG" /D "_WINDOWS" /D "_MBCS" /D "_USRDLL" /D ZTS=1 /D "ZEND_WIN32" /D "PHP_WIN32" /D HAVE_OCI8=1 /D HAVE_OCI8_TEMP_LOB=1 /D HAVE_OCI8_ATTR_STATEMENT=1 /D COMPILE_DL_OCI8=1 /YX /FD /c
+# ADD BASE MTL /nologo /D "NDEBUG" /mktyplib203 /win32
+# ADD MTL /nologo /D "NDEBUG" /mktyplib203 /win32
+# ADD BASE RSC /l 0x406 /d "NDEBUG"
+# ADD RSC /l 0x406 /d "NDEBUG"
+BSC32=bscmake.exe
+# ADD BASE BSC32 /nologo
+# ADD BSC32 /nologo
+LINK32=link.exe
+# ADD BASE LINK32 kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib php5ts.lib /nologo /dll /machine:I386
+# ADD LINK32 php5ts.lib oci.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /dll /machine:I386 /out:"..\..\Release_TS/php_oci8.dll" /libpath:"..\..\Release_TS" /libpath:"..\..\..\php_build\oci805\lib" /libpath:"..\..\Release_TS_Inline"
+
+!ELSEIF "$(CFG)" == "oci8 - Win32 Debug_TS"
+
+# PROP BASE Use_MFC 0
+# PROP BASE Use_Debug_Libraries 0
+# PROP BASE Output_Dir "Debug_TS"
+# PROP BASE Intermediate_Dir "Debug_TS"
+# PROP BASE Ignore_Export_Lib 0
+# PROP BASE Target_Dir ""
+# PROP Use_MFC 0
+# PROP Use_Debug_Libraries 0
+# PROP Output_Dir "Debug_TS"
+# PROP Intermediate_Dir "Debug_TS"
+# PROP Ignore_Export_Lib 0
+# PROP Target_Dir ""
+# ADD BASE CPP /nologo /MT /W3 /GX /O2 /I "..\.." /I "..\..\Zend" /I "..\..\..\bindlib_w32" /I "..\..\TSRM" /D "WIN32" /D "NDEBUG" /D "_WINDOWS" /D "_MBCS" /D "_USRDLL" /D ZTS=1 /YX /FD /c
+# ADD CPP /nologo /MDd /W3 /GX /ZI /Od /I "..\..\\" /I "..\..\main" /I "..\..\Zend" /I "..\..\..\bindlib_w32" /I "..\..\TSRM" /I "..\..\..\php_build\include\oci805" /D ZEND_DEBUG=1 /D "WIN32" /D "NDEBUG" /D "_WINDOWS" /D "_MBCS" /D "_USRDLL" /D "FTP_EXPORTS" /D ZTS=1 /D "ZEND_WIN32" /D "PHP_WIN32" /D HAVE_OCI8=1 /D HAVE_OCI8_TEMP_LOB=1 /D HAVE_OCI8_ATTR_STATEMENT=1 /D COMPILE_DL_OCI8=1 /FR /YX /FD /c
+# ADD BASE MTL /nologo /D "NDEBUG" /mktyplib203 /win32
+# ADD MTL /nologo /D "NDEBUG" /mktyplib203 /win32
+# ADD BASE RSC /l 0x406 /d "NDEBUG"
+# ADD RSC /l 0x406 /d "NDEBUG"
+BSC32=bscmake.exe
+# ADD BASE BSC32 /nologo
+# ADD BSC32 /nologo
+LINK32=link.exe
+# ADD BASE LINK32 kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib php5ts.lib /nologo /dll /machine:I386
+# ADD LINK32 php5ts_debug.lib oci.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /dll /machine:I386 /out:"..\..\Debug_TS/php_oci8.dll" /libpath:"..\..\Debug_TS" /libpath:"..\..\..\php_build\lib\oci805"
+
+!ENDIF
+
+# Begin Target
+
+# Name "oci8 - Win32 Release_TS"
+# Name "oci8 - Win32 Debug_TS"
+# Begin Group "Source Files"
+
+# PROP Default_Filter "cpp;c;cxx;rc;def;r;odl;idl;hpj;bat"
+# Begin Source File
+
+SOURCE=.\oci8.c
+# End Source File
+# End Group
+# Begin Group "Header Files"
+
+# PROP Default_Filter "h;hpp;hxx;hm;inl"
+# Begin Source File
+
+SOURCE=.\php_oci8.h
+# End Source File
+# End Group
+# Begin Group "Resource Files"
+
+# PROP Default_Filter "ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe"
+# End Group
+# End Target
+# End Project
diff --git a/ext/oci8/package.xml b/ext/oci8/package.xml new file mode 100644 index 000000000..b0b168a02 --- /dev/null +++ b/ext/oci8/package.xml @@ -0,0 +1,85 @@ +<?xml version="1.0" encoding="ISO-8859-1" ?> +<!DOCTYPE package SYSTEM "../pear/package.dtd"> +<package> + <name>oci8</name> + <summary>Oracle 8 call interface functions</summary> + <maintainers> + <maintainer> + <user>tony2001</user> + <name>Antony Dovgal</name> + <email>tony2001@php.net</email> + <role>developer</role> + </maintainer> + <maintainer> + <user>ssb</user> + <name>Stig Bakken</name> + <email>ssb@php.net</email> + <role>developer</role> + </maintainer> + <maintainer> + <user>thies</user> + <name>Thies C. Arntzen</name> + <email>thies@php.net</email> + <role>developer</role> + </maintainer> + <maintainer> + <user>asautins</user> + <name>Andy Sautins</name> + <email>asautins@php.net</email> + <role>developer</role> + </maintainer> + <maintainer> + <user>dbenson</user> + <name>David Benson</name> + <email>dbenson@php.net</email> + <role>developer</role> + </maintainer> + <maintainer> + <user>maxim</user> + <name>Maxim Maletsky</name> + <email>maxim@php.net</email> + <role>developer</role> + </maintainer> + <maintainer> + <user>phanto</user> + <name>Harald Radi</name> + <email>phanto@php.net</email> + <role>developer</role> + </maintainer> + </maintainers> + <description> +These functions allow you to access Oracle database servers using +the Oracle 8 call interface. + </description> + <license>PHP</license> + <release> + <state>beta</state> + <version>5.0.0rc1</version> + <date>2004-03-19</date> + <notes> +package.xml added to support intallation using pear installer + </notes> + <configureoptions> + <configureoption name="with-oci8" default="autodetect" prompt="ORACLE_HOME directory?"/> + </configureoptions> + <filelist> + <file role="doc" name="CREDITS"/> + <file role="src" name="config.m4"/> + <file role="src" name="config.w32"/> + <file role="src" name="oci8.dsp"/> + <file role="src" name="oci8.c"/> + <file role="src" name="php_oci8.h"/> + <file role="test" name="tests/bug26133.phpt"/> + <file role="test" name="tests/connect.inc"/> + <file role="test" name="tests/create_table.inc"/> + <file role="test" name="tests/drop_table.inc"/> + <file role="test" name="tests/skipif.inc"/> + </filelist> + <deps> + <dep type="php" rel="ge" version="5" /> + </deps> + </release> +</package> +<!-- +vim:et:ts=1:sw=1 +--> diff --git a/ext/oci8/php_oci8.h b/ext/oci8/php_oci8.h new file mode 100644 index 000000000..7463c6afa --- /dev/null +++ b/ext/oci8/php_oci8.h @@ -0,0 +1,225 @@ +/* + +----------------------------------------------------------------------+ + | PHP Version 5 | + +----------------------------------------------------------------------+ + | Copyright (c) 1997-2004 The PHP Group | + +----------------------------------------------------------------------+ + | This source file is subject to version 3.0 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_0.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. | + +----------------------------------------------------------------------+ + | Authors: Stig Sæther Bakken <ssb@php.net> | + | Thies C. Arntzen <thies@thieso.net> | + +----------------------------------------------------------------------+ +*/ + +/* $Id: php_oci8.h,v 1.35 2004/04/29 12:59:22 iliaa Exp $ */ + +#if HAVE_OCI8 +# ifndef PHP_OCI8_H +# define PHP_OCI8_H +# endif + +# if (defined(__osf__) && defined(__alpha)) +# ifndef A_OSF +# define A_OSF +# endif +# ifndef OSF1 +# define OSF1 +# endif +# ifndef _INTRINSICS +# define _INTRINSICS +# endif +# endif /* osf alpha */ + +#ifdef PHP_WIN32 +#define PHP_OCI_API __declspec(dllexport) +#else +#define PHP_OCI_API +#endif + +#if defined(min) +#undef min +#endif +#if defined(max) +#undef max +#endif + +#include <oci.h> + +#define OCI_SEEK_SET 0 +#define OCI_SEEK_CUR 1 +#define OCI_SEEK_END 2 + +typedef struct { + int num; + int persistent; + int is_open; + char *dbname; + OCIServer *pServer; +#if 0 + OCIFocbkStruct failover; +#endif +} oci_server; + +typedef struct { + int num; + zend_bool persistent; + zend_bool is_open; + zend_bool exclusive; +#if ZTS + THREAD_T thread; +#else + zend_bool thread; +#endif + zend_llist *sessions_list; + oci_server *server; + OCISession *pSession; + OCIEnv *pEnv; /* sessions own environment */ + ub2 charsetId; /* sessions used character set (mostly this will be 0, so NLS_LANG will be used. */ +} oci_session; + +typedef struct { + int id; + int is_open; + oci_session *session; + OCISvcCtx *pServiceContext; + sword error; + OCIError *pError; + int needs_commit; + HashTable *descriptors; +} oci_connection; + +typedef struct { + int id; + oci_connection *conn; + dvoid *ocidescr; + ub4 type; + int lob_current_position; + int lob_size; /* -1 = Lob wasn't initialized yet */ + int buffering; /* 0 - off, 1 - on, 2 - on and buffer was used */ +} oci_descriptor; + +typedef struct { + int id; + oci_connection *conn; + OCIType *tdo; + OCITypeCode coll_typecode; + OCIRef *elem_ref; + OCIType *element_type; + OCITypeCode element_typecode; + OCIColl *coll; +} oci_collection; + +typedef struct { + zval *zval; + text *name; + ub4 name_len; + ub4 type; +} oci_define; + +typedef struct { + int id; + oci_connection *conn; + sword error; + OCIError *pError; + OCIStmt *pStmt; + char *last_query; + HashTable *columns; + HashTable *binds; + HashTable *defines; + int ncolumns; + int executed; + int has_data; + ub2 stmttype; +} oci_statement; + +typedef struct { + OCIBind *pBind; + zval *zval; + dvoid *descr; /* used for binding of LOBS etc */ + OCIStmt *pStmt; /* used for binding REFCURSORs */ + sb2 indicator; + ub2 retcode; +} oci_bind; + +typedef struct { + oci_statement *statement; + OCIDefine *pDefine; + char *name; + ub4 name_len; + ub2 data_type; + ub2 data_size; + ub4 storage_size4; + sb2 indicator; + ub2 retcode; + ub2 retlen; + ub4 retlen4; + ub2 is_descr; + ub2 is_cursor; + int descr; + oci_statement *pstmt; + int stmtid; + int descid; + void *data; + oci_define *define; + int piecewise; + ub4 cb_retlen; + ub2 scale; + ub2 precision; +} oci_out_column; + +typedef struct { + sword error; + OCIError *pError; + + /* + char *default_username; + char *default_password; + char *default_dbname; + */ + + long debug_mode; + + int shutdown; + + /* XXX NYI + long allow_persistent; + long max_persistent; + long max_links; + */ + + OCIEnv *pEnv; + + int in_call; +} php_oci_globals; + +extern zend_module_entry oci8_module_entry; +#define phpext_oci8_ptr &oci8_module_entry + +#define OCI_MAX_NAME_LEN 64 +#define OCI_MAX_DATA_SIZE INT_MAX +#define OCI_PIECE_SIZE (64*1024)-1 + +#ifdef ZTS +#define OCI(v) TSRMG(oci_globals_id, php_oci_globals *, v) +#else +#define OCI(v) (oci_globals.v) +#endif + +#else /* !HAVE_OCI8 */ + +# define oci8_module_ptr NULL + +#endif /* HAVE_OCI8 */ + +/* + * Local variables: + * tab-width: 4 + * c-basic-offset: 4 + * End: + */ diff --git a/ext/oci8/tests/bug26133.phpt b/ext/oci8/tests/bug26133.phpt new file mode 100644 index 000000000..d3f0ed9bf --- /dev/null +++ b/ext/oci8/tests/bug26133.phpt @@ -0,0 +1,36 @@ +--TEST-- +Bug #26133 (ocifreedesc() segfault) +--SKIPIF-- +<?php + require 'skipif.inc'; +?> +--FILE-- +<?php + require 'connect.inc'; + require 'create_table.inc'; + + if ($connection) { + $ora_sql = "INSERT INTO + ".$schema."php_test_table (id, value) + VALUES ('1','1') + RETURNING + ROWID + INTO :v_rowid "; + + $statement = OCIParse($connection,$ora_sql); + $rowid = OCINewDescriptor($connection,OCI_D_ROWID); + OCIBindByName($statement,":v_rowid", $rowid,-1,OCI_B_ROWID); + if (OCIExecute($statement)) { + OCICommit($connection); + } + OCIFreeStatement($statement); + $rowid->free(); + } + + require 'drop_table.inc'; + + echo "Done\n"; +?> +--EXPECTF-- +Done + diff --git a/ext/oci8/tests/connect.inc b/ext/oci8/tests/connect.inc new file mode 100644 index 000000000..a0013e20d --- /dev/null +++ b/ext/oci8/tests/connect.inc @@ -0,0 +1,39 @@ +<?php + +/* + * Please, change user, password and dbase to match your configuration. + * + * */ + +$user = "user"; +$password = "pass"; +$dbase = "base"; + +/* + * You should have privileges to create tables in this schema + * + * */ + +$schema = "system"; + +/* + * Remove the last line in skipif.inc to run tests + * + * */ + + + if (!empty($dbase)) { + $connection = ocilogon($user,$password,$dbase); + } + else { + $connection = ocilogon($user,$password); + } + + if (!empty($schema)) { + $schema = $schema."."; + } + else { + $schema = ''; + } + +?> diff --git a/ext/oci8/tests/create_table.inc b/ext/oci8/tests/create_table.inc new file mode 100644 index 000000000..c423ce577 --- /dev/null +++ b/ext/oci8/tests/create_table.inc @@ -0,0 +1,12 @@ +<?php + + if ($connection) { + $ora_sql = "CREATE TABLE + ".$schema."php_test_table (id NUMBER, value NUMBER) + "; + + $statement = OCIParse($connection,$ora_sql); + OCIExecute($statement); + } + +?> diff --git a/ext/oci8/tests/drop_table.inc b/ext/oci8/tests/drop_table.inc new file mode 100644 index 000000000..4e558f5e3 --- /dev/null +++ b/ext/oci8/tests/drop_table.inc @@ -0,0 +1,12 @@ +<?php + + if ($connection) { + $ora_sql = "DROP TABLE + ".$schema."php_test_table + "; + + $statement = OCIParse($connection,$ora_sql); + OCIExecute($statement); + } + +?> diff --git a/ext/oci8/tests/skipif.inc b/ext/oci8/tests/skipif.inc new file mode 100644 index 000000000..ed0992c8d --- /dev/null +++ b/ext/oci8/tests/skipif.inc @@ -0,0 +1,10 @@ +<?php + +if (!extension_loaded('oci8')) die("skip oci8 extension is not available\n"); + +/* + * Remove or comment this line to run tests + * + * */ +die("skip change default login/password\n"); +?> |