root/ext/oci8/oci8_lob.c

/* [<][>][^][v][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. php_oci_lob_create
  2. php_oci_lob_get_length
  3. php_oci_lob_callback
  4. php_oci_lob_calculate_buffer
  5. php_oci_lob_read
  6. php_oci_lob_write
  7. php_oci_lob_set_buffering
  8. php_oci_lob_get_buffering
  9. php_oci_lob_copy
  10. php_oci_lob_close
  11. php_oci_temp_lob_close
  12. php_oci_lob_flush
  13. php_oci_lob_free
  14. php_oci_lob_import
  15. php_oci_lob_append
  16. php_oci_lob_truncate
  17. php_oci_lob_erase
  18. php_oci_lob_is_equal
  19. php_oci_lob_write_tmp

/*
   +----------------------------------------------------------------------+
   | PHP Version 5                                                        |
   +----------------------------------------------------------------------+
   | Copyright (c) 1997-2015 The PHP Group                                |
   +----------------------------------------------------------------------+
   | This source file is subject to version 3.01 of the PHP license,      |
   | that is bundled with this package in the file LICENSE, and is        |
   | available through the world-wide-web at the following url:           |
   | http://www.php.net/license/3_01.txt                                  |
   | If you did not receive a copy of the PHP license and are unable to   |
   | obtain it through the world-wide-web, please send a note to          |
   | license@php.net so we can mail you a copy immediately.               |
   +----------------------------------------------------------------------+
   | Authors: Stig Sæther Bakken <ssb@php.net>                            |
   |          Thies C. Arntzen <thies@thieso.net>                         |
   |                                                                      |
   | 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>        |
   |                                                                      |
   | Redesigned by: Antony Dovgal <antony@zend.com>                       |
   |                Andi Gutmans <andi@zend.com>                          |
   |                Wez Furlong <wez@omniti.com>                          |
   +----------------------------------------------------------------------+
*/

/* $Id$ */



#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"
#include "php_oci8_int.h"

/* for import/export functions */
#include <fcntl.h>

#ifndef O_BINARY
#define O_BINARY 0
#endif

/* {{{ php_oci_lob_create()
 Create LOB descriptor and allocate all the resources needed */
php_oci_descriptor *php_oci_lob_create (php_oci_connection *connection, long type TSRMLS_DC)
{
        php_oci_descriptor *descriptor;
        sword errstatus;

        switch (type) {
                case OCI_DTYPE_FILE:
                case OCI_DTYPE_LOB:
                case OCI_DTYPE_ROWID:
                        /* these three are allowed */
                        break;
                default:
                        php_error_docref(NULL TSRMLS_CC, E_WARNING, "Unknown descriptor type %ld", type);
                        return NULL;
                        break;
        }

        descriptor = ecalloc(1, sizeof(php_oci_descriptor));
        descriptor->type = type;
        descriptor->connection = connection;
        zend_list_addref(descriptor->connection->id);

        PHP_OCI_CALL_RETURN(errstatus, OCIDescriptorAlloc, (connection->env, (dvoid*)&(descriptor->descriptor), descriptor->type, (size_t) 0, (dvoid **) 0));

        if (errstatus != OCI_SUCCESS) {
                OCI_G(errcode) = php_oci_error(OCI_G(err), errstatus TSRMLS_CC);
                PHP_OCI_HANDLE_ERROR(connection, OCI_G(errcode));
                efree(descriptor);
                return NULL;
        } else {
                OCI_G(errcode) = 0; /* retain backwards compat with OCI8 1.4 */
        }

        PHP_OCI_REGISTER_RESOURCE(descriptor, le_descriptor);
        
        descriptor->lob_current_position = 0;
        descriptor->lob_size = -1;                              /* we should set it to -1 to know, that it's just not initialized */
        descriptor->buffering = PHP_OCI_LOB_BUFFER_DISABLED;                            /* buffering is off by default */
        descriptor->charset_form = SQLCS_IMPLICIT;      /* default value */
        descriptor->charset_id = connection->charset;
        descriptor->is_open = 0;
        descriptor->chunk_size = 0;

        if (descriptor->type == OCI_DTYPE_LOB || descriptor->type == OCI_DTYPE_FILE) {
                /* add Lobs & Files to hash. we'll flush them at the end */
                if (!connection->descriptors) {
                        ALLOC_HASHTABLE(connection->descriptors);
                        zend_hash_init(connection->descriptors, 0, NULL, php_oci_descriptor_flush_hash_dtor, 0);
                        connection->descriptor_count = 0;
                }
                
                descriptor->index = (connection->descriptor_count)++;
                if (connection->descriptor_count == LONG_MAX) {
                        php_error_docref(NULL TSRMLS_CC, E_WARNING, "Internal descriptor counter has reached limit");
                        php_oci_connection_descriptors_free(connection TSRMLS_CC);
                        return NULL;
                }

                zend_hash_index_update(connection->descriptors,descriptor->index,&descriptor,sizeof(php_oci_descriptor *),NULL);
        }
        return descriptor;

}
/* }}} */

/* {{{ php_oci_lob_get_length()
 Get length of the LOB. The length is cached so we don't need to ask Oracle every time */
int php_oci_lob_get_length (php_oci_descriptor *descriptor, ub4 *length TSRMLS_DC)
{
        php_oci_connection *connection = descriptor->connection;
        sword errstatus;

        *length = 0;
        
        if (descriptor->lob_size >= 0) {
                *length = descriptor->lob_size;
                return 0;
        } else {
                if (descriptor->type == OCI_DTYPE_FILE) {
                        PHP_OCI_CALL_RETURN(errstatus, OCILobFileOpen, (connection->svc, connection->err, descriptor->descriptor, OCI_FILE_READONLY));
                        if (errstatus != OCI_SUCCESS) {
                                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                                return 1;
                        }
                }
                
                PHP_OCI_CALL_RETURN(errstatus, OCILobGetLength, (connection->svc, connection->err, descriptor->descriptor, (ub4 *)length));

                if (errstatus != OCI_SUCCESS) {
                        connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                        PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                        return 1;
                }

                descriptor->lob_size = *length;

                if (descriptor->type == OCI_DTYPE_FILE) {
                        PHP_OCI_CALL_RETURN(errstatus, OCILobFileClose, (connection->svc, connection->err, descriptor->descriptor));

                        if (errstatus != OCI_SUCCESS) {
                                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                                return 1;
                        }
                }

                connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */
        }
        return 0;       
}
/* }}} */

/* {{{ php_oci_lob_callback()
   Append LOB portion to a memory buffer */
sb4 php_oci_lob_callback (dvoid *ctxp, CONST dvoid *bufxp, oraub8 len, ub1 piece, dvoid **changed_bufpp, oraub8 *changed_lenp)
{
        ub4 lenp = (ub4) len;
        php_oci_lob_ctx *ctx = (php_oci_lob_ctx *)ctxp;

        switch (piece)
        {
                case OCI_LAST_PIECE:
                        if ((*(ctx->lob_len) + lenp) > (ctx->alloc_len)) {
                                /* this should not happen ever */
                                *(ctx->lob_data) = NULL;
                                *(ctx->lob_len) = 0;
                                return OCI_ERROR;
                        }
                        memcpy(*(ctx->lob_data) + *(ctx->lob_len), bufxp, (size_t) lenp);
                        *(ctx->lob_len) += lenp;
                        *(*(ctx->lob_data) + *(ctx->lob_len)) = 0x00;
                        return OCI_CONTINUE;

                case OCI_FIRST_PIECE:
                case OCI_NEXT_PIECE:
                        if ((*(ctx->lob_len) + lenp) > ctx->alloc_len) {
                                /* this should not happen ever */
                                *(ctx->lob_data) = NULL;
                                *(ctx->lob_len) = 0;
                                return OCI_ERROR;
                        }
                        memcpy(*(ctx->lob_data) + *(ctx->lob_len), bufxp, (size_t) lenp);
                        *(ctx->lob_len) += lenp;
                        return OCI_CONTINUE;

                default: {
                        TSRMLS_FETCH();
                        php_error_docref(NULL TSRMLS_CC, E_WARNING, "Unexpected LOB piece id received (value:%d)", piece);
                        *(ctx->lob_data) = NULL;
                        *(ctx->lob_len) = 0;
                        return OCI_ERROR;
                }
        }
}
/* }}} */

/* {{{ php_oci_lob_calculate_buffer() 
   Work out the size for LOB buffering */
static inline int php_oci_lob_calculate_buffer(php_oci_descriptor *descriptor, long read_length TSRMLS_DC)
{
        php_oci_connection *connection = descriptor->connection;
        ub4 chunk_size;
        sword errstatus;

        if (descriptor->type == OCI_DTYPE_FILE) {
                return read_length;
        }

        if (!descriptor->chunk_size) {
                PHP_OCI_CALL_RETURN(errstatus, OCILobGetChunkSize, (connection->svc, connection->err, descriptor->descriptor, &chunk_size));

                if (errstatus != OCI_SUCCESS) {
                        connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                        PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                        return read_length; /* we have to return original length here */
                }
                descriptor->chunk_size = chunk_size;
                connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */
        }
        
        if ((read_length % descriptor->chunk_size) != 0) {
                return descriptor->chunk_size * ((read_length / descriptor->chunk_size) + 1);
        }
        return read_length;
}
/* }}} */

/* {{{ php_oci_lob_read()
 Read specified portion of the LOB into the buffer */
int php_oci_lob_read (php_oci_descriptor *descriptor, long read_length, long initial_offset, char **data, ub4 *data_len TSRMLS_DC)
{
        php_oci_connection *connection = descriptor->connection;
        ub4 length = 0;
        int buffer_size = PHP_OCI_LOB_BUFFER_SIZE;
        php_oci_lob_ctx ctx;
        ub1 *bufp;
        oraub8 bytes_read, offset = 0;
        oraub8 requested_len = read_length; /* this is by default */
        oraub8 chars_read = 0;
        int is_clob = 0;
        sb4 bytes_per_char = 1;
        sword errstatus;

        *data_len = 0;
        *data = NULL;

        ctx.lob_len = data_len;
        ctx.lob_data = data;
        ctx.alloc_len = 0;

        if (php_oci_lob_get_length(descriptor, &length TSRMLS_CC)) {
                return 1;
        }

        if (length <= 0) {
                return 0;
        }
        
        if (initial_offset > length) {
                php_error_docref(NULL TSRMLS_CC, E_WARNING, "Offset must be less than size of the LOB");
                return 1;
        }
                
        if (read_length == -1) {
                requested_len = length;
        }
        
        if (requested_len > (length - initial_offset)) {
                requested_len = length - initial_offset;
        }
        
        if (requested_len <= 0) {
                return 0;
        }
        
        offset = initial_offset;

        if (descriptor->type == OCI_DTYPE_FILE) {
                PHP_OCI_CALL_RETURN(errstatus, OCILobFileOpen, (connection->svc, connection->err, descriptor->descriptor, OCI_FILE_READONLY));

                if (errstatus != OCI_SUCCESS) {
                        connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                        PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                        return 1;
                }
        } else {
                ub2 charset_id = 0;

                PHP_OCI_CALL_RETURN(errstatus, OCILobCharSetId, (connection->env, connection->err, descriptor->descriptor, &charset_id));

                if (errstatus != OCI_SUCCESS) {
                        connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                        PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                        return 1;
                }

                if (charset_id > 0) { /* charset_id is always > 0 for [N]CLOBs */
                        is_clob = 1;
                }
        }

        if (is_clob) {
                PHP_OCI_CALL_RETURN(errstatus, OCINlsNumericInfoGet, (connection->env, connection->err, &bytes_per_char, OCI_NLS_CHARSET_MAXBYTESZ));

                if (errstatus != OCI_SUCCESS) {
                        connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                        PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                        return 1;
                }
        } else {
                /* BLOBs don't have encoding, so bytes_per_char == 1 */
        }

        ctx.alloc_len = (requested_len + 1) * bytes_per_char;
        *data = ecalloc(bytes_per_char, requested_len + 1);

        if (is_clob) {
                chars_read = requested_len;
                bytes_read = 0;
        } else {
                chars_read = 0;
                bytes_read = requested_len;
        }

        buffer_size = (requested_len < buffer_size ) ? requested_len : buffer_size;             /* optimize buffer size */
        buffer_size = php_oci_lob_calculate_buffer(descriptor, buffer_size TSRMLS_CC);  /* use chunk size */

        bufp = (ub1 *) ecalloc(1, buffer_size);
        PHP_OCI_CALL_RETURN(errstatus, OCILobRead2,
                (
                        connection->svc,
                        connection->err,
                        descriptor->descriptor,
                        (oraub8 *)&bytes_read,                                                  /* IN/OUT bytes toread/read */
                        (oraub8 *)&chars_read,                                                  /* IN/OUT chars toread/read */
                        (oraub8) offset + 1,                                                    /* offset (starts with 1) */
                        (dvoid *) bufp,
                        (oraub8) buffer_size,                                                   /* size of buffer */
                        OCI_FIRST_PIECE,
                        (dvoid *)&ctx,
                        (OCICallbackLobRead2) php_oci_lob_callback,                             /* callback... */
                        (ub2) descriptor->charset_id,                      /* The character set ID of the buffer data. */
                        (ub1) descriptor->charset_form                                    /* The character set form of the buffer data. */
                )
        );
        
        efree(bufp);
        
        if (is_clob) {
                offset = descriptor->lob_current_position + chars_read;
        } else {
                offset = descriptor->lob_current_position + bytes_read;
        }
        
        if (errstatus != OCI_SUCCESS) {
                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                if (*data) {
                        efree(*data);
                        *data = NULL;
                }
                *data_len = 0;
                return 1;
        }
        
        descriptor->lob_current_position = (int)offset;

        if (descriptor->type == OCI_DTYPE_FILE) {
                PHP_OCI_CALL_RETURN(errstatus, OCILobFileClose, (connection->svc, connection->err, descriptor->descriptor));

                if (errstatus != OCI_SUCCESS) {
                        connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                        PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                        if (*data) {
                                efree(*data);
                                *data = NULL;
                        }
                        *data_len = 0;
                        return 1;
                }
        }

        connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */
        return 0;
}
/* }}} */

/* {{{ php_oci_lob_write()
 Write data to the LOB */
int php_oci_lob_write (php_oci_descriptor *descriptor, ub4 offset, char *data, int data_len, ub4 *bytes_written TSRMLS_DC)
{
        OCILobLocator *lob                 = (OCILobLocator *) descriptor->descriptor;
        php_oci_connection *connection = (php_oci_connection *) descriptor->connection;
        ub4 lob_length;
        sword errstatus;
        
        *bytes_written = 0;
        if (php_oci_lob_get_length(descriptor, &lob_length TSRMLS_CC)) {
                return 1;
        }
        
        if (!data || data_len <= 0) {
                return 0;
        }
        
        if (offset < 0) {
                offset = 0;
        }
        
        if (offset > descriptor->lob_current_position) {
                offset = descriptor->lob_current_position;
        }
        
        PHP_OCI_CALL_RETURN(errstatus, OCILobWrite,
                        (
                                connection->svc,
                                connection->err,
                                lob,
                                (ub4 *)&data_len,
                                (ub4) offset + 1,
                                (dvoid *) data,
                                (ub4) data_len,
                                OCI_ONE_PIECE,
                                (dvoid *)0,
                                (OCICallbackLobWrite) 0,
                                (ub2) descriptor->charset_id,
                                (ub1) descriptor->charset_form
                        )
                );

        if (errstatus) {
                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                *bytes_written = 0;
                return 1;
        }
        *bytes_written = data_len;
        descriptor->lob_current_position += data_len;
        
        if (descriptor->lob_current_position > descriptor->lob_size) {
                descriptor->lob_size = descriptor->lob_current_position;
        }
        
        /* marking buffer as used */
        if (descriptor->buffering == PHP_OCI_LOB_BUFFER_ENABLED) {
                descriptor->buffering = PHP_OCI_LOB_BUFFER_USED;
        }
        
        connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */
        return 0;
}
/* }}} */

/* {{{ php_oci_lob_set_buffering()
 Turn buffering off/onn for this particular LOB */
int php_oci_lob_set_buffering (php_oci_descriptor *descriptor, int on_off TSRMLS_DC)
{
        php_oci_connection *connection = descriptor->connection;
        sword errstatus;

        if (!on_off && descriptor->buffering == PHP_OCI_LOB_BUFFER_DISABLED) {
                /* disabling when it's already off */
                return 0;
        }
        
        if (on_off && descriptor->buffering != PHP_OCI_LOB_BUFFER_DISABLED) {
                /* enabling when it's already on */
                return 0;
        }
        
        if (on_off) {
                PHP_OCI_CALL_RETURN(errstatus, OCILobEnableBuffering, (connection->svc, connection->err, descriptor->descriptor));
        } else {
                PHP_OCI_CALL_RETURN(errstatus, OCILobDisableBuffering, (connection->svc, connection->err, descriptor->descriptor));
        }

        if (errstatus != OCI_SUCCESS) {
                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                return 1;
        }
        descriptor->buffering = on_off ? PHP_OCI_LOB_BUFFER_ENABLED : PHP_OCI_LOB_BUFFER_DISABLED;
        connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */
        return 0;
}
/* }}} */

/* {{{ php_oci_lob_get_buffering()
 Return current buffering state for the LOB */
int php_oci_lob_get_buffering (php_oci_descriptor *descriptor)
{
        if (descriptor->buffering != PHP_OCI_LOB_BUFFER_DISABLED) {
                return 1;
        } else {
                return 0;
        }
}
/* }}} */

/* {{{ php_oci_lob_copy()
 Copy one LOB (or its part) to another one */
int php_oci_lob_copy (php_oci_descriptor *descriptor_dest, php_oci_descriptor *descriptor_from, long length TSRMLS_DC)
{
        php_oci_connection *connection = descriptor_dest->connection;
        ub4 length_dest, length_from, copy_len;
        sword errstatus;
        
        if (php_oci_lob_get_length(descriptor_dest, &length_dest TSRMLS_CC)) {
                return 1;
        }
        
        if (php_oci_lob_get_length(descriptor_from, &length_from TSRMLS_CC)) {
                return 1;
        }

        if (length == -1) {
                copy_len = length_from - descriptor_from->lob_current_position;
        } else {
                copy_len = length;
        }

        if ((int)copy_len <= 0) {
                /* silently fail, there is nothing to copy */
                return 1;
        }

        PHP_OCI_CALL_RETURN(errstatus, OCILobCopy,
                        (
                         connection->svc,
                         connection->err,
                         descriptor_dest->descriptor,
                         descriptor_from->descriptor,
                         copy_len,
                         descriptor_dest->lob_current_position+1,
                         descriptor_from->lob_current_position+1
                        )
        );

        if (errstatus != OCI_SUCCESS) {
                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                return 1;
        }
        
        connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */
        return 0;
}
/* }}} */

/* {{{ php_oci_lob_close()
 Close LOB */
int php_oci_lob_close (php_oci_descriptor *descriptor TSRMLS_DC)
{
        php_oci_connection *connection = descriptor->connection;
        sword errstatus;

        if (descriptor->is_open) {
                PHP_OCI_CALL_RETURN(errstatus, OCILobClose, (connection->svc, connection->err, descriptor->descriptor));

                if (errstatus != OCI_SUCCESS) {
                        connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                        PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                        return 1;
                }
                connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */
        }

        if (php_oci_temp_lob_close(descriptor TSRMLS_CC)) {
                return 1;
        }
        
        return 0;
}
/* }}} */

/* {{{ php_oci_temp_lob_close()
   Close Temporary LOB */
int php_oci_temp_lob_close (php_oci_descriptor *descriptor TSRMLS_DC)
{
        php_oci_connection *connection = descriptor->connection;
        int is_temporary;
        sword errstatus;

        PHP_OCI_CALL_RETURN(errstatus, OCILobIsTemporary, (connection->env,connection->err, descriptor->descriptor, &is_temporary));
        
        if (errstatus != OCI_SUCCESS) {
                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                return 1;
        }
        
        if (is_temporary) {
                PHP_OCI_CALL_RETURN(errstatus, OCILobFreeTemporary, (connection->svc, connection->err, descriptor->descriptor));
                
                if (errstatus != OCI_SUCCESS) {
                        connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                        PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                        return 1;
                }
        }
        connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */
        return 0;
}
/* }}} */

/* {{{ php_oci_lob_flush()
 Flush buffers for the LOB (only if they have been used) */
int php_oci_lob_flush(php_oci_descriptor *descriptor, long flush_flag TSRMLS_DC)
{
        OCILobLocator *lob = descriptor->descriptor;
        php_oci_connection *connection = descriptor->connection;
        sword errstatus;

        if (!lob) {
                return 1;
        }

        switch (flush_flag) {
                case 0:
                case OCI_LOB_BUFFER_FREE:
                        /* only these two are allowed */
                        break;
                default:
                        php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid flag value: %ld", flush_flag);
                        return 1;
                        break;
        }
        
        /* do not really flush buffer, but report success
         * to suppress OCI error when flushing not used buffer
         * */
        if (descriptor->buffering != PHP_OCI_LOB_BUFFER_USED) {
                return 0;
        }

        PHP_OCI_CALL_RETURN(errstatus, OCILobFlushBuffer, (connection->svc, connection->err, lob, flush_flag));

        if (errstatus != OCI_SUCCESS) {
                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                return 1;
        }

        /* marking buffer as enabled and not used */
        descriptor->buffering = PHP_OCI_LOB_BUFFER_ENABLED;
        connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */
        return 0;
}
/* }}} */

/* {{{ php_oci_lob_free()
 Close LOB descriptor and free associated resources */
void php_oci_lob_free (php_oci_descriptor *descriptor TSRMLS_DC)
{
        if (!descriptor || !descriptor->connection) {
                return;
        }

        if (descriptor->connection->descriptors) {
                /* delete descriptor from the hash */
                zend_hash_index_del(descriptor->connection->descriptors, descriptor->index);
                if (zend_hash_num_elements(descriptor->connection->descriptors) == 0) {
                        descriptor->connection->descriptor_count = 0;
                } else {
                        if (descriptor->index + 1 == descriptor->connection->descriptor_count) {
                                /* If the descriptor being freed is the end-most one
                                 * allocated, then the descriptor_count is reduced so
                                 * a future descriptor can reuse the hash table index.
                                 * This can prevent the hash index range increasing in
                                 * the common case that each descriptor is
                                 * allocated/used/freed before another descriptor is
                                 * needed.  However it is possible that a script frees
                                 * descriptors in arbitrary order which would prevent
                                 * descriptor_count ever being reduced to zero until
                                 * zend_hash_num_elements() returns 0.
                                 */
                                descriptor->connection->descriptor_count--;
                        }
                }
        }
        
        /* flushing Lobs & Files with buffering enabled */
        if ((descriptor->type == OCI_DTYPE_FILE || descriptor->type == OCI_DTYPE_LOB) && descriptor->buffering == PHP_OCI_LOB_BUFFER_USED) {
                php_oci_lob_flush(descriptor, OCI_LOB_BUFFER_FREE TSRMLS_CC);
        }

        if (descriptor->type == OCI_DTYPE_LOB) {
                php_oci_temp_lob_close(descriptor TSRMLS_CC);
        }

        PHP_OCI_CALL(OCIDescriptorFree, (descriptor->descriptor, descriptor->type));

        zend_list_delete(descriptor->connection->id);
        efree(descriptor);
}
/* }}} */

/* {{{ php_oci_lob_import()
 Import LOB contents from the given file */
int php_oci_lob_import (php_oci_descriptor *descriptor, char *filename TSRMLS_DC)
{
        int fp;
        ub4 loblen;
        OCILobLocator *lob = (OCILobLocator *)descriptor->descriptor;
        php_oci_connection *connection = descriptor->connection;
        char buf[8192];
        ub4 offset = 1;
        sword errstatus;
        
#if (PHP_MAJOR_VERSION == 5 && PHP_MINOR_VERSION > 3) || (PHP_MAJOR_VERSION > 5)
        /* Safe mode has been removed in PHP 5.4 */
        if (php_check_open_basedir(filename TSRMLS_CC)) {
#else
        if ((PG(safe_mode) && (!php_checkuid(filename, NULL, CHECKUID_CHECK_FILE_AND_DIR))) || php_check_open_basedir(filename TSRMLS_CC)) {
#endif
                return 1;
        }
        
        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 1;
        }

        while ((loblen = read(fp, &buf, sizeof(buf))) > 0) {    
                PHP_OCI_CALL_RETURN(errstatus,
                                OCILobWrite,
                                (
                                        connection->svc,
                                        connection->err,
                                        lob,
                                        &loblen,
                                        offset,
                                        (dvoid *) &buf,
                                        loblen,
                                        OCI_ONE_PIECE,
                                        (dvoid *)0,
                                        (OCICallbackLobWrite) 0,
                                        (ub2) descriptor->charset_id,
                                        (ub1) descriptor->charset_form
                                )
                );

                if (errstatus != OCI_SUCCESS) {
                        connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                        PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                        close(fp);
                        return 1;
                } else {
                        connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */
                }
                offset += loblen;
        }
        close(fp);
        
        return 0;
}
        /* }}} */

/* {{{ php_oci_lob_append()
 Append data to the end of the LOB */
int php_oci_lob_append (php_oci_descriptor *descriptor_dest, php_oci_descriptor *descriptor_from TSRMLS_DC)
{
        php_oci_connection *connection = descriptor_dest->connection;
        OCILobLocator *lob_dest = descriptor_dest->descriptor;
        OCILobLocator *lob_from = descriptor_from->descriptor;
        ub4 dest_len, from_len;
        sword errstatus;

        if (php_oci_lob_get_length(descriptor_dest, &dest_len TSRMLS_CC)) {
                return 1;
        }
        
        if (php_oci_lob_get_length(descriptor_from, &from_len TSRMLS_CC)) {
                return 1;
        }

        if (from_len <= 0) {
                return 0;
        }

        PHP_OCI_CALL_RETURN(errstatus, OCILobAppend, (connection->svc, connection->err, lob_dest, lob_from));

        if (errstatus != OCI_SUCCESS) {
                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                return 1;
        }
        connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */
        return 0;
}
/* }}} */

/* {{{ php_oci_lob_truncate()
 Truncate LOB to the given length */
int php_oci_lob_truncate (php_oci_descriptor *descriptor, long new_lob_length TSRMLS_DC)
{
        php_oci_connection *connection = descriptor->connection;
        OCILobLocator *lob = descriptor->descriptor;
        ub4 lob_length;
        sword errstatus;
        
        if (php_oci_lob_get_length(descriptor, &lob_length TSRMLS_CC)) {
                return 1;
        }
        
        if (lob_length <= 0) {
                return 0;
        }

        if (new_lob_length < 0) {
                php_error_docref(NULL TSRMLS_CC, E_WARNING, "Size must be greater than or equal to 0");
                return 1;
        }

        if (new_lob_length > lob_length) {
                php_error_docref(NULL TSRMLS_CC, E_WARNING, "Size must be less than or equal to the current LOB size");
                return 1;
        }
        
        PHP_OCI_CALL_RETURN(errstatus, OCILobTrim, (connection->svc, connection->err, lob, new_lob_length));

        if (errstatus != OCI_SUCCESS) {
                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                return 1;
        }
        
        descriptor->lob_size = new_lob_length;
        connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */

        return 0;
}
/* }}} */

/* {{{ php_oci_lob_erase()
 Erase (or fill with whitespaces, depending on LOB type) the LOB (or its part) */
int php_oci_lob_erase (php_oci_descriptor *descriptor, long offset, ub4 length, ub4 *bytes_erased TSRMLS_DC)
{
        php_oci_connection *connection = descriptor->connection;
        OCILobLocator *lob = descriptor->descriptor;
        ub4 lob_length;
        sword errstatus;

        *bytes_erased = 0;
        
        if (php_oci_lob_get_length(descriptor, &lob_length TSRMLS_CC)) {
                return 1;
        }
        
        if (offset == -1) {
                offset = descriptor->lob_current_position;
        }

        if (length == -1) {
                length = lob_length;
        }
        
        PHP_OCI_CALL_RETURN(errstatus, OCILobErase, (connection->svc, connection->err, lob, (ub4 *)&length, offset+1));

        if (errstatus != OCI_SUCCESS) {
                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                return 1;
        }
        
        *bytes_erased = length;
        connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */
        return 0;
}
/* }}} */

/* {{{ php_oci_lob_is_equal()
 Compare two LOB descriptors and figure out if they are pointing to the same LOB */
int php_oci_lob_is_equal (php_oci_descriptor *descriptor_first, php_oci_descriptor *descriptor_second, boolean *result TSRMLS_DC)
{
        php_oci_connection *connection = descriptor_first->connection;
        OCILobLocator *first_lob   = descriptor_first->descriptor;
        OCILobLocator *second_lob  = descriptor_second->descriptor;
        sword errstatus;

        PHP_OCI_CALL_RETURN(errstatus, OCILobIsEqual, (connection->env, first_lob, second_lob, result));

        if (errstatus) {
                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                return 1;
        }
        connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */
        return 0;
}
/* }}} */

/* {{{ php_oci_lob_write_tmp()
 Create temporary LOB and write data to it */
int php_oci_lob_write_tmp (php_oci_descriptor *descriptor, long type, char *data, int data_len TSRMLS_DC)
{
        php_oci_connection *connection = descriptor->connection;
        OCILobLocator *lob                 = descriptor->descriptor;
        ub4 bytes_written = 0;
        sword errstatus;
        
        switch (type) {
                case OCI_TEMP_BLOB:
                case OCI_TEMP_CLOB:
                        /* only these two are allowed */
                        break;
                default:
                        php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid temporary lob type: %ld", type);
                        return 1;
                        break;
        }

        if (data_len < 0) {
                return 1;
        }

        PHP_OCI_CALL_RETURN(errstatus, OCILobCreateTemporary,
                        (
                         connection->svc,
                         connection->err,
                         lob,
                         OCI_DEFAULT,
                         OCI_DEFAULT,
                         (ub1)type,
                         OCI_ATTR_NOCACHE,
                         OCI_DURATION_SESSION
                        )
        );

        if (errstatus) {
                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                return 1;
        }

        PHP_OCI_CALL_RETURN(errstatus, OCILobOpen, (connection->svc, connection->err, lob, OCI_LOB_READWRITE));

        if (errstatus) {
                connection->errcode = php_oci_error(connection->err, errstatus TSRMLS_CC);
                PHP_OCI_HANDLE_ERROR(connection, connection->errcode);
                return 1;
        }

        descriptor->is_open = 1;
        connection->errcode = 0; /* retain backwards compat with OCI8 1.4 */

        return php_oci_lob_write(descriptor, 0, data, data_len, &bytes_written TSRMLS_CC);
}
/* }}} */

#endif /* HAVE_OCI8 */

/*
 * Local variables:
 * tab-width: 4
 * c-basic-offset: 4
 * End:
 * vim600: noet sw=4 ts=4 fdm=marker
 * vim<600: noet sw=4 ts=4
 */

/* [<][>][^][v][top][bottom][index][help] */