mirror of
https://github.com/php/php-src.git
synced 2025-08-16 14:08:47 +02:00
@- Added support for an optional output handler function for output
@ buffering. This enables transparent rendering of XML through XSL, @ transparent compression, etc. (Zeev)
This commit is contained in:
parent
77b55ad07d
commit
cf8e389799
5 changed files with 155 additions and 45 deletions
|
@ -29,13 +29,12 @@ static int php_ub_body_write(const char *str, uint str_length);
|
||||||
static int php_ub_body_write_no_header(const char *str, uint str_length);
|
static int php_ub_body_write_no_header(const char *str, uint str_length);
|
||||||
static int php_b_body_write(const char *str, uint str_length);
|
static int php_b_body_write(const char *str, uint str_length);
|
||||||
|
|
||||||
static void php_ob_init(uint initial_size, uint block_size);
|
static void php_ob_init(uint initial_size, uint block_size, zval *output_handler);
|
||||||
static void php_ob_destroy(void);
|
static void php_ob_destroy(void);
|
||||||
static void php_ob_append(const char *text, uint text_length);
|
static void php_ob_append(const char *text, uint text_length);
|
||||||
#if 0
|
#if 0
|
||||||
static void php_ob_prepend(const char *text, uint text_length);
|
static void php_ob_prepend(const char *text, uint text_length);
|
||||||
#endif
|
#endif
|
||||||
static inline void php_ob_send(void);
|
|
||||||
|
|
||||||
|
|
||||||
#ifdef ZTS
|
#ifdef ZTS
|
||||||
|
@ -51,6 +50,7 @@ static void php_output_init_globals(OLS_D)
|
||||||
OG(implicit_flush) = 0;
|
OG(implicit_flush) = 0;
|
||||||
OG(output_start_filename) = NULL;
|
OG(output_start_filename) = NULL;
|
||||||
OG(output_start_lineno) = 0;
|
OG(output_start_lineno) = 0;
|
||||||
|
OG(lock) = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -88,37 +88,75 @@ PHPAPI int php_header_write(const char *str, uint str_length)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Start output buffering */
|
/* Start output buffering */
|
||||||
PHPAPI void php_start_ob_buffer()
|
PHPAPI int php_start_ob_buffer(zval *output_handler)
|
||||||
{
|
{
|
||||||
OLS_FETCH();
|
OLS_FETCH();
|
||||||
|
|
||||||
php_ob_init(4096, 1024);
|
if (OG(lock)) {
|
||||||
|
return FAILURE;
|
||||||
|
}
|
||||||
|
php_ob_init(4096, 1024, output_handler);
|
||||||
OG(php_body_write) = php_b_body_write;
|
OG(php_body_write) = php_b_body_write;
|
||||||
|
return SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* End output buffering (one level) */
|
/* End output buffering (one level) */
|
||||||
PHPAPI void php_end_ob_buffer(int send_buffer)
|
PHPAPI void php_end_ob_buffer(int send_buffer)
|
||||||
{
|
{
|
||||||
|
char *final_buffer=NULL;
|
||||||
|
int final_buffer_length;
|
||||||
|
zval *alternate_buffer=NULL;
|
||||||
SLS_FETCH();
|
SLS_FETCH();
|
||||||
OLS_FETCH();
|
OLS_FETCH();
|
||||||
|
|
||||||
if (OG(nesting_level)==0) {
|
if (OG(nesting_level)==0) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (OG(active_ob_buffer).output_handler) {
|
||||||
|
zval **params[1];
|
||||||
|
zval *orig_buffer;
|
||||||
|
CLS_FETCH();
|
||||||
|
|
||||||
|
ALLOC_INIT_ZVAL(orig_buffer);
|
||||||
|
orig_buffer->value.str.val = OG(active_ob_buffer).buffer;
|
||||||
|
orig_buffer->value.str.len = OG(active_ob_buffer).text_length;
|
||||||
|
orig_buffer->type = IS_STRING;
|
||||||
|
orig_buffer->refcount=2; /* don't let call_user_function() destroy our buffer */
|
||||||
|
|
||||||
|
params[0] = &orig_buffer;
|
||||||
|
OG(lock) = 1;
|
||||||
|
if (call_user_function_ex(CG(function_table), NULL, OG(active_ob_buffer).output_handler, &alternate_buffer, 1, params, 1, NULL)==SUCCESS) {
|
||||||
|
convert_to_string_ex(&alternate_buffer);
|
||||||
|
final_buffer = alternate_buffer->value.str.val;
|
||||||
|
final_buffer_length = alternate_buffer->value.str.len;
|
||||||
|
}
|
||||||
|
OG(lock) = 0;
|
||||||
|
zval_ptr_dtor(&OG(active_ob_buffer).output_handler);
|
||||||
|
if (orig_buffer->refcount==2) { /* free the zval */
|
||||||
|
FREE_ZVAL(orig_buffer);
|
||||||
|
} else {
|
||||||
|
orig_buffer->refcount--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!final_buffer) {
|
||||||
|
final_buffer = OG(active_ob_buffer).buffer;
|
||||||
|
final_buffer_length = OG(active_ob_buffer).text_length;
|
||||||
|
}
|
||||||
if (OG(nesting_level)==1) { /* end buffering */
|
if (OG(nesting_level)==1) { /* end buffering */
|
||||||
if (SG(headers_sent) && !SG(request_info).headers_only) {
|
if (SG(headers_sent) && !SG(request_info).headers_only) {
|
||||||
OG(php_body_write) = php_ub_body_write_no_header;
|
OG(php_body_write) = php_ub_body_write_no_header;
|
||||||
} else {
|
} else {
|
||||||
OG(php_body_write) = php_ub_body_write;
|
OG(php_body_write) = php_ub_body_write;
|
||||||
}
|
}
|
||||||
if (send_buffer) {
|
}
|
||||||
php_ob_send();
|
if (send_buffer) {
|
||||||
}
|
OG(php_body_write)(final_buffer, final_buffer_length);
|
||||||
} else { /* only flush the buffer, if necessary */
|
}
|
||||||
if (send_buffer) {
|
if (alternate_buffer) {
|
||||||
OG(php_body_write)(OG(active_ob_buffer).buffer, OG(active_ob_buffer).text_length);
|
zval_ptr_dtor(&alternate_buffer);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
php_ob_destroy();
|
php_ob_destroy();
|
||||||
}
|
}
|
||||||
|
@ -170,7 +208,7 @@ static inline void php_ob_allocate(void)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void php_ob_init(uint initial_size, uint block_size)
|
static void php_ob_init(uint initial_size, uint block_size, zval *output_handler)
|
||||||
{
|
{
|
||||||
OLS_FETCH();
|
OLS_FETCH();
|
||||||
|
|
||||||
|
@ -185,6 +223,7 @@ static void php_ob_init(uint initial_size, uint block_size)
|
||||||
OG(active_ob_buffer).size = initial_size;
|
OG(active_ob_buffer).size = initial_size;
|
||||||
OG(active_ob_buffer).buffer = (char *) emalloc(initial_size+1);
|
OG(active_ob_buffer).buffer = (char *) emalloc(initial_size+1);
|
||||||
OG(active_ob_buffer).text_length = 0;
|
OG(active_ob_buffer).text_length = 0;
|
||||||
|
OG(active_ob_buffer).output_handler = output_handler;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -245,14 +284,6 @@ static void php_ob_prepend(const char *text, uint text_length)
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static inline void php_ob_send()
|
|
||||||
{
|
|
||||||
OLS_FETCH();
|
|
||||||
|
|
||||||
/* header_write is a simple, unbuffered output function */
|
|
||||||
OG(php_body_write)(OG(active_ob_buffer).buffer, OG(active_ob_buffer).text_length);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Return the current output buffer */
|
/* Return the current output buffer */
|
||||||
int php_ob_get_buffer(pval *p)
|
int php_ob_get_buffer(pval *p)
|
||||||
|
@ -362,7 +393,29 @@ static int php_ub_body_write(const char *str, uint str_length)
|
||||||
Turn on Output Buffering */
|
Turn on Output Buffering */
|
||||||
PHP_FUNCTION(ob_start)
|
PHP_FUNCTION(ob_start)
|
||||||
{
|
{
|
||||||
php_start_ob_buffer();
|
zval *output_handler;
|
||||||
|
|
||||||
|
switch (ZEND_NUM_ARGS()) {
|
||||||
|
case 0:
|
||||||
|
output_handler = NULL;
|
||||||
|
break;
|
||||||
|
case 1: {
|
||||||
|
zval **output_handler_p;
|
||||||
|
|
||||||
|
if (zend_get_parameters_ex(1, &output_handler_p)==FAILURE) {
|
||||||
|
RETURN_FALSE;
|
||||||
|
}
|
||||||
|
SEPARATE_ZVAL(output_handler_p);
|
||||||
|
output_handler = *output_handler_p;
|
||||||
|
output_handler->refcount++;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (php_start_ob_buffer(output_handler)==FAILURE) {
|
||||||
|
php_error(E_WARNING, "Cannot use output buffering in output buffering display handlers");
|
||||||
|
RETURN_FALSE;
|
||||||
|
}
|
||||||
|
RETURN_TRUE;
|
||||||
}
|
}
|
||||||
/* }}} */
|
/* }}} */
|
||||||
|
|
||||||
|
|
|
@ -26,7 +26,7 @@
|
||||||
PHPAPI void php_output_startup(void);
|
PHPAPI void php_output_startup(void);
|
||||||
PHPAPI int php_body_write(const char *str, uint str_length);
|
PHPAPI int php_body_write(const char *str, uint str_length);
|
||||||
PHPAPI int php_header_write(const char *str, uint str_length);
|
PHPAPI int php_header_write(const char *str, uint str_length);
|
||||||
PHPAPI void php_start_ob_buffer(void);
|
PHPAPI int php_start_ob_buffer(zval *output_handler);
|
||||||
PHPAPI void php_end_ob_buffer(int send_buffer);
|
PHPAPI void php_end_ob_buffer(int send_buffer);
|
||||||
PHPAPI void php_end_ob_buffers(int send_buffer);
|
PHPAPI void php_end_ob_buffers(int send_buffer);
|
||||||
PHPAPI int php_ob_get_buffer(pval *p);
|
PHPAPI int php_ob_get_buffer(pval *p);
|
||||||
|
@ -50,6 +50,7 @@ typedef struct _php_ob_buffer {
|
||||||
uint size;
|
uint size;
|
||||||
uint text_length;
|
uint text_length;
|
||||||
int block_size;
|
int block_size;
|
||||||
|
zval *output_handler;
|
||||||
} php_ob_buffer;
|
} php_ob_buffer;
|
||||||
|
|
||||||
typedef struct _php_output_globals {
|
typedef struct _php_output_globals {
|
||||||
|
@ -61,6 +62,7 @@ typedef struct _php_output_globals {
|
||||||
int output_start_lineno;
|
int output_start_lineno;
|
||||||
zend_stack ob_buffers;
|
zend_stack ob_buffers;
|
||||||
int nesting_level;
|
int nesting_level;
|
||||||
|
zend_bool lock;
|
||||||
} php_output_globals;
|
} php_output_globals;
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -607,7 +607,7 @@ int php_request_startup(CLS_D ELS_DC PLS_DC SLS_DC)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (PG(output_buffering)) {
|
if (PG(output_buffering)) {
|
||||||
php_start_ob_buffer();
|
php_start_ob_buffer(NULL);
|
||||||
} else if (PG(implicit_flush)) {
|
} else if (PG(implicit_flush)) {
|
||||||
php_start_implicit_flush();
|
php_start_implicit_flush();
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,13 +29,12 @@ static int php_ub_body_write(const char *str, uint str_length);
|
||||||
static int php_ub_body_write_no_header(const char *str, uint str_length);
|
static int php_ub_body_write_no_header(const char *str, uint str_length);
|
||||||
static int php_b_body_write(const char *str, uint str_length);
|
static int php_b_body_write(const char *str, uint str_length);
|
||||||
|
|
||||||
static void php_ob_init(uint initial_size, uint block_size);
|
static void php_ob_init(uint initial_size, uint block_size, zval *output_handler);
|
||||||
static void php_ob_destroy(void);
|
static void php_ob_destroy(void);
|
||||||
static void php_ob_append(const char *text, uint text_length);
|
static void php_ob_append(const char *text, uint text_length);
|
||||||
#if 0
|
#if 0
|
||||||
static void php_ob_prepend(const char *text, uint text_length);
|
static void php_ob_prepend(const char *text, uint text_length);
|
||||||
#endif
|
#endif
|
||||||
static inline void php_ob_send(void);
|
|
||||||
|
|
||||||
|
|
||||||
#ifdef ZTS
|
#ifdef ZTS
|
||||||
|
@ -51,6 +50,7 @@ static void php_output_init_globals(OLS_D)
|
||||||
OG(implicit_flush) = 0;
|
OG(implicit_flush) = 0;
|
||||||
OG(output_start_filename) = NULL;
|
OG(output_start_filename) = NULL;
|
||||||
OG(output_start_lineno) = 0;
|
OG(output_start_lineno) = 0;
|
||||||
|
OG(lock) = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -88,37 +88,75 @@ PHPAPI int php_header_write(const char *str, uint str_length)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Start output buffering */
|
/* Start output buffering */
|
||||||
PHPAPI void php_start_ob_buffer()
|
PHPAPI int php_start_ob_buffer(zval *output_handler)
|
||||||
{
|
{
|
||||||
OLS_FETCH();
|
OLS_FETCH();
|
||||||
|
|
||||||
php_ob_init(4096, 1024);
|
if (OG(lock)) {
|
||||||
|
return FAILURE;
|
||||||
|
}
|
||||||
|
php_ob_init(4096, 1024, output_handler);
|
||||||
OG(php_body_write) = php_b_body_write;
|
OG(php_body_write) = php_b_body_write;
|
||||||
|
return SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* End output buffering (one level) */
|
/* End output buffering (one level) */
|
||||||
PHPAPI void php_end_ob_buffer(int send_buffer)
|
PHPAPI void php_end_ob_buffer(int send_buffer)
|
||||||
{
|
{
|
||||||
|
char *final_buffer=NULL;
|
||||||
|
int final_buffer_length;
|
||||||
|
zval *alternate_buffer=NULL;
|
||||||
SLS_FETCH();
|
SLS_FETCH();
|
||||||
OLS_FETCH();
|
OLS_FETCH();
|
||||||
|
|
||||||
if (OG(nesting_level)==0) {
|
if (OG(nesting_level)==0) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (OG(active_ob_buffer).output_handler) {
|
||||||
|
zval **params[1];
|
||||||
|
zval *orig_buffer;
|
||||||
|
CLS_FETCH();
|
||||||
|
|
||||||
|
ALLOC_INIT_ZVAL(orig_buffer);
|
||||||
|
orig_buffer->value.str.val = OG(active_ob_buffer).buffer;
|
||||||
|
orig_buffer->value.str.len = OG(active_ob_buffer).text_length;
|
||||||
|
orig_buffer->type = IS_STRING;
|
||||||
|
orig_buffer->refcount=2; /* don't let call_user_function() destroy our buffer */
|
||||||
|
|
||||||
|
params[0] = &orig_buffer;
|
||||||
|
OG(lock) = 1;
|
||||||
|
if (call_user_function_ex(CG(function_table), NULL, OG(active_ob_buffer).output_handler, &alternate_buffer, 1, params, 1, NULL)==SUCCESS) {
|
||||||
|
convert_to_string_ex(&alternate_buffer);
|
||||||
|
final_buffer = alternate_buffer->value.str.val;
|
||||||
|
final_buffer_length = alternate_buffer->value.str.len;
|
||||||
|
}
|
||||||
|
OG(lock) = 0;
|
||||||
|
zval_ptr_dtor(&OG(active_ob_buffer).output_handler);
|
||||||
|
if (orig_buffer->refcount==2) { /* free the zval */
|
||||||
|
FREE_ZVAL(orig_buffer);
|
||||||
|
} else {
|
||||||
|
orig_buffer->refcount--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!final_buffer) {
|
||||||
|
final_buffer = OG(active_ob_buffer).buffer;
|
||||||
|
final_buffer_length = OG(active_ob_buffer).text_length;
|
||||||
|
}
|
||||||
if (OG(nesting_level)==1) { /* end buffering */
|
if (OG(nesting_level)==1) { /* end buffering */
|
||||||
if (SG(headers_sent) && !SG(request_info).headers_only) {
|
if (SG(headers_sent) && !SG(request_info).headers_only) {
|
||||||
OG(php_body_write) = php_ub_body_write_no_header;
|
OG(php_body_write) = php_ub_body_write_no_header;
|
||||||
} else {
|
} else {
|
||||||
OG(php_body_write) = php_ub_body_write;
|
OG(php_body_write) = php_ub_body_write;
|
||||||
}
|
}
|
||||||
if (send_buffer) {
|
}
|
||||||
php_ob_send();
|
if (send_buffer) {
|
||||||
}
|
OG(php_body_write)(final_buffer, final_buffer_length);
|
||||||
} else { /* only flush the buffer, if necessary */
|
}
|
||||||
if (send_buffer) {
|
if (alternate_buffer) {
|
||||||
OG(php_body_write)(OG(active_ob_buffer).buffer, OG(active_ob_buffer).text_length);
|
zval_ptr_dtor(&alternate_buffer);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
php_ob_destroy();
|
php_ob_destroy();
|
||||||
}
|
}
|
||||||
|
@ -170,7 +208,7 @@ static inline void php_ob_allocate(void)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void php_ob_init(uint initial_size, uint block_size)
|
static void php_ob_init(uint initial_size, uint block_size, zval *output_handler)
|
||||||
{
|
{
|
||||||
OLS_FETCH();
|
OLS_FETCH();
|
||||||
|
|
||||||
|
@ -185,6 +223,7 @@ static void php_ob_init(uint initial_size, uint block_size)
|
||||||
OG(active_ob_buffer).size = initial_size;
|
OG(active_ob_buffer).size = initial_size;
|
||||||
OG(active_ob_buffer).buffer = (char *) emalloc(initial_size+1);
|
OG(active_ob_buffer).buffer = (char *) emalloc(initial_size+1);
|
||||||
OG(active_ob_buffer).text_length = 0;
|
OG(active_ob_buffer).text_length = 0;
|
||||||
|
OG(active_ob_buffer).output_handler = output_handler;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -245,14 +284,6 @@ static void php_ob_prepend(const char *text, uint text_length)
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static inline void php_ob_send()
|
|
||||||
{
|
|
||||||
OLS_FETCH();
|
|
||||||
|
|
||||||
/* header_write is a simple, unbuffered output function */
|
|
||||||
OG(php_body_write)(OG(active_ob_buffer).buffer, OG(active_ob_buffer).text_length);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Return the current output buffer */
|
/* Return the current output buffer */
|
||||||
int php_ob_get_buffer(pval *p)
|
int php_ob_get_buffer(pval *p)
|
||||||
|
@ -362,7 +393,29 @@ static int php_ub_body_write(const char *str, uint str_length)
|
||||||
Turn on Output Buffering */
|
Turn on Output Buffering */
|
||||||
PHP_FUNCTION(ob_start)
|
PHP_FUNCTION(ob_start)
|
||||||
{
|
{
|
||||||
php_start_ob_buffer();
|
zval *output_handler;
|
||||||
|
|
||||||
|
switch (ZEND_NUM_ARGS()) {
|
||||||
|
case 0:
|
||||||
|
output_handler = NULL;
|
||||||
|
break;
|
||||||
|
case 1: {
|
||||||
|
zval **output_handler_p;
|
||||||
|
|
||||||
|
if (zend_get_parameters_ex(1, &output_handler_p)==FAILURE) {
|
||||||
|
RETURN_FALSE;
|
||||||
|
}
|
||||||
|
SEPARATE_ZVAL(output_handler_p);
|
||||||
|
output_handler = *output_handler_p;
|
||||||
|
output_handler->refcount++;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (php_start_ob_buffer(output_handler)==FAILURE) {
|
||||||
|
php_error(E_WARNING, "Cannot use output buffering in output buffering display handlers");
|
||||||
|
RETURN_FALSE;
|
||||||
|
}
|
||||||
|
RETURN_TRUE;
|
||||||
}
|
}
|
||||||
/* }}} */
|
/* }}} */
|
||||||
|
|
||||||
|
|
|
@ -26,7 +26,7 @@
|
||||||
PHPAPI void php_output_startup(void);
|
PHPAPI void php_output_startup(void);
|
||||||
PHPAPI int php_body_write(const char *str, uint str_length);
|
PHPAPI int php_body_write(const char *str, uint str_length);
|
||||||
PHPAPI int php_header_write(const char *str, uint str_length);
|
PHPAPI int php_header_write(const char *str, uint str_length);
|
||||||
PHPAPI void php_start_ob_buffer(void);
|
PHPAPI int php_start_ob_buffer(zval *output_handler);
|
||||||
PHPAPI void php_end_ob_buffer(int send_buffer);
|
PHPAPI void php_end_ob_buffer(int send_buffer);
|
||||||
PHPAPI void php_end_ob_buffers(int send_buffer);
|
PHPAPI void php_end_ob_buffers(int send_buffer);
|
||||||
PHPAPI int php_ob_get_buffer(pval *p);
|
PHPAPI int php_ob_get_buffer(pval *p);
|
||||||
|
@ -50,6 +50,7 @@ typedef struct _php_ob_buffer {
|
||||||
uint size;
|
uint size;
|
||||||
uint text_length;
|
uint text_length;
|
||||||
int block_size;
|
int block_size;
|
||||||
|
zval *output_handler;
|
||||||
} php_ob_buffer;
|
} php_ob_buffer;
|
||||||
|
|
||||||
typedef struct _php_output_globals {
|
typedef struct _php_output_globals {
|
||||||
|
@ -61,6 +62,7 @@ typedef struct _php_output_globals {
|
||||||
int output_start_lineno;
|
int output_start_lineno;
|
||||||
zend_stack ob_buffers;
|
zend_stack ob_buffers;
|
||||||
int nesting_level;
|
int nesting_level;
|
||||||
|
zend_bool lock;
|
||||||
} php_output_globals;
|
} php_output_globals;
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue