2000-10-13 08:09:31 +08:00
|
|
|
/*
|
|
|
|
+----------------------------------------------------------------------+
|
2004-01-08 16:18:22 +08:00
|
|
|
| PHP Version 5 |
|
2000-10-13 08:09:31 +08:00
|
|
|
+----------------------------------------------------------------------+
|
2007-01-01 17:29:37 +08:00
|
|
|
| Copyright (c) 1997-2007 The PHP Group |
|
2000-10-13 08:09:31 +08:00
|
|
|
+----------------------------------------------------------------------+
|
2006-01-01 21:10:10 +08:00
|
|
|
| This source file is subject to version 3.01 of the PHP license, |
|
2000-10-13 08:09:31 +08:00
|
|
|
| that is bundled with this package in the file LICENSE, and is |
|
2003-06-11 04:04:29 +08:00
|
|
|
| available through the world-wide-web at the following url: |
|
2006-01-01 21:10:10 +08:00
|
|
|
| http://www.php.net/license/3_01.txt |
|
2000-10-13 08:09:31 +08:00
|
|
|
| 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. |
|
|
|
|
+----------------------------------------------------------------------+
|
2002-02-28 16:29:35 +08:00
|
|
|
| Authors: Rasmus Lerdorf <rasmus@php.net> |
|
2000-10-13 08:09:31 +08:00
|
|
|
| Jim Winstead <jimw@php.net> |
|
|
|
|
| Hartmut Holzgraefe <hholzgra@php.net> |
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
*/
|
|
|
|
/* $Id$ */
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2001-06-26 21:33:24 +08:00
|
|
|
#if HAVE_UNISTD_H
|
2001-06-22 15:07:48 +08:00
|
|
|
#include <unistd.h>
|
2001-06-26 21:33:24 +08:00
|
|
|
#endif
|
2000-10-13 08:09:31 +08:00
|
|
|
|
|
|
|
#include "php.h"
|
|
|
|
#include "php_globals.h"
|
|
|
|
#include "php_standard.h"
|
|
|
|
#include "php_fopen_wrappers.h"
|
2002-09-08 04:54:14 +08:00
|
|
|
#include "SAPI.h"
|
2000-10-13 08:09:31 +08:00
|
|
|
|
2002-09-08 03:00:46 +08:00
|
|
|
static size_t php_stream_output_write(php_stream *stream, const char *buf, size_t count TSRMLS_DC)
|
|
|
|
{
|
|
|
|
PHPWRITE(buf, count);
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t php_stream_output_read(php_stream *stream, char *buf, size_t count TSRMLS_DC)
|
|
|
|
{
|
2002-10-05 18:35:13 +08:00
|
|
|
stream->eof = 1;
|
2002-09-08 03:00:46 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int php_stream_output_close(php_stream *stream, int close_handle TSRMLS_DC)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int php_stream_output_flush(php_stream *stream TSRMLS_DC)
|
|
|
|
{
|
|
|
|
sapi_flush(TSRMLS_C);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
php_stream_ops php_stream_output_ops = {
|
|
|
|
php_stream_output_write,
|
|
|
|
php_stream_output_read,
|
|
|
|
php_stream_output_close,
|
|
|
|
php_stream_output_flush,
|
|
|
|
"Output",
|
2002-09-23 09:47:04 +08:00
|
|
|
NULL, /* seek */
|
|
|
|
NULL, /* cast */
|
|
|
|
NULL, /* stat */
|
|
|
|
NULL /* set_option */
|
2002-09-08 03:00:46 +08:00
|
|
|
};
|
|
|
|
|
2002-10-22 00:41:06 +08:00
|
|
|
static size_t php_stream_input_write(php_stream *stream, const char *buf, size_t count TSRMLS_DC)
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t php_stream_input_read(php_stream *stream, char *buf, size_t count TSRMLS_DC)
|
|
|
|
{
|
2005-11-18 03:39:39 +08:00
|
|
|
off_t *position = (off_t*)stream->abstract;
|
2002-10-22 03:08:18 +08:00
|
|
|
size_t read_bytes = 0;
|
2005-11-18 03:39:39 +08:00
|
|
|
|
2002-10-22 00:41:06 +08:00
|
|
|
if(!stream->eof) {
|
2002-11-13 02:29:11 +08:00
|
|
|
if(SG(request_info).raw_post_data) { /* data has already been read by a post handler */
|
2005-11-18 03:39:39 +08:00
|
|
|
read_bytes = SG(request_info).raw_post_data_length - *position;
|
2002-10-22 00:41:06 +08:00
|
|
|
if(read_bytes <= count) {
|
|
|
|
stream->eof = 1;
|
|
|
|
} else {
|
|
|
|
read_bytes = count;
|
|
|
|
}
|
|
|
|
if(read_bytes) {
|
2005-11-18 03:39:39 +08:00
|
|
|
memcpy(buf, SG(request_info).raw_post_data + *position, read_bytes);
|
2002-10-22 00:41:06 +08:00
|
|
|
}
|
2002-11-13 02:29:11 +08:00
|
|
|
} else if(sapi_module.read_post) {
|
2002-10-22 00:41:06 +08:00
|
|
|
read_bytes = sapi_module.read_post(buf, count TSRMLS_CC);
|
|
|
|
if(read_bytes <= 0){
|
|
|
|
stream->eof = 1;
|
|
|
|
read_bytes = 0;
|
|
|
|
}
|
2002-11-13 02:29:11 +08:00
|
|
|
} else {
|
|
|
|
stream->eof = 1;
|
2002-10-22 00:41:06 +08:00
|
|
|
}
|
|
|
|
}
|
2002-10-22 03:08:18 +08:00
|
|
|
|
2005-11-18 03:39:39 +08:00
|
|
|
*position += read_bytes;
|
2002-11-21 18:29:11 +08:00
|
|
|
SG(read_post_bytes) += read_bytes;
|
2002-10-22 03:08:18 +08:00
|
|
|
return read_bytes;
|
2002-10-22 00:41:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int php_stream_input_close(php_stream *stream, int close_handle TSRMLS_DC)
|
|
|
|
{
|
2005-11-18 03:39:39 +08:00
|
|
|
efree(stream->abstract);
|
|
|
|
|
2002-10-22 00:41:06 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int php_stream_input_flush(php_stream *stream TSRMLS_DC)
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
php_stream_ops php_stream_input_ops = {
|
|
|
|
php_stream_input_write,
|
|
|
|
php_stream_input_read,
|
|
|
|
php_stream_input_close,
|
|
|
|
php_stream_input_flush,
|
|
|
|
"Input",
|
|
|
|
NULL, /* seek */
|
|
|
|
NULL, /* cast */
|
|
|
|
NULL, /* stat */
|
|
|
|
NULL /* set_option */
|
|
|
|
};
|
|
|
|
|
2003-02-18 14:10:14 +08:00
|
|
|
static void php_stream_apply_filter_list(php_stream *stream, char *filterlist, int read_chain, int write_chain TSRMLS_DC) {
|
|
|
|
char *p, *token;
|
|
|
|
php_stream_filter *temp_filter;
|
|
|
|
|
|
|
|
p = php_strtok_r(filterlist, "|", &token);
|
|
|
|
while (p) {
|
|
|
|
if (read_chain) {
|
2003-04-16 22:30:25 +08:00
|
|
|
if ((temp_filter = php_stream_filter_create(p, NULL, php_stream_is_persistent(stream) TSRMLS_CC))) {
|
2003-02-18 14:10:14 +08:00
|
|
|
php_stream_filter_append(&stream->readfilters, temp_filter);
|
|
|
|
} else {
|
2005-08-18 21:34:04 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Unable to create filter (%s)", p);
|
2003-02-18 14:10:14 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (write_chain) {
|
2003-04-16 22:30:25 +08:00
|
|
|
if ((temp_filter = php_stream_filter_create(p, NULL, php_stream_is_persistent(stream) TSRMLS_CC))) {
|
2003-02-18 14:10:14 +08:00
|
|
|
php_stream_filter_append(&stream->writefilters, temp_filter);
|
|
|
|
} else {
|
2005-08-18 21:34:04 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Unable to create filter (%s)", p);
|
2003-02-18 14:10:14 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
p = php_strtok_r(NULL, "|", &token);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-04-11 20:13:17 +08:00
|
|
|
php_stream * php_stream_url_wrap_php(php_stream_wrapper *wrapper, char *path, char *mode, int options, char **opened_path, php_stream_context *context STREAMS_DC TSRMLS_DC)
|
2000-10-13 08:09:31 +08:00
|
|
|
{
|
2003-02-13 21:43:37 +08:00
|
|
|
int fd = -1;
|
2003-02-21 04:05:31 +08:00
|
|
|
int mode_rw = 0;
|
2002-03-16 05:03:08 +08:00
|
|
|
php_stream * stream = NULL;
|
2003-02-18 14:10:14 +08:00
|
|
|
char *p, *token, *pathdup;
|
2005-10-29 23:09:12 +08:00
|
|
|
long max_memory;
|
2002-03-22 01:02:50 +08:00
|
|
|
|
2006-05-01 23:59:56 +08:00
|
|
|
if (!strncasecmp(path, "php://", 6)) {
|
2002-03-22 01:02:50 +08:00
|
|
|
path += 6;
|
2006-05-01 23:59:56 +08:00
|
|
|
}
|
2000-10-13 08:09:31 +08:00
|
|
|
|
2005-10-29 23:09:12 +08:00
|
|
|
if (!strncasecmp(path, "temp", 4)) {
|
|
|
|
path += 4;
|
|
|
|
max_memory = PHP_STREAM_MAX_MEM;
|
|
|
|
if (!strncasecmp(path, "/maxmemory:", 11)) {
|
|
|
|
path += 11;
|
2005-10-29 23:49:28 +08:00
|
|
|
max_memory = strtol(path, NULL, 10);
|
2005-10-29 23:09:12 +08:00
|
|
|
if (max_memory < 0) {
|
2006-05-16 22:12:52 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_RECOVERABLE_ERROR, "Max memory must be >= 0");
|
2005-10-29 23:09:12 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
2005-11-12 00:51:24 +08:00
|
|
|
return php_stream_temp_create(TEMP_STREAM_DEFAULT, max_memory);
|
2005-10-29 22:29:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!strcasecmp(path, "memory")) {
|
2005-11-12 00:51:24 +08:00
|
|
|
return php_stream_memory_create(TEMP_STREAM_DEFAULT);
|
2005-10-29 22:29:59 +08:00
|
|
|
}
|
|
|
|
|
2002-09-08 03:00:46 +08:00
|
|
|
if (!strcasecmp(path, "output")) {
|
|
|
|
return php_stream_alloc(&php_stream_output_ops, NULL, 0, "wb");
|
|
|
|
}
|
2006-11-06 04:44:52 +08:00
|
|
|
|
2002-10-22 00:41:06 +08:00
|
|
|
if (!strcasecmp(path, "input")) {
|
2007-01-25 05:43:47 +08:00
|
|
|
/* Override default behavior for php://input when used as an include and allow_url_include is being used in BC (off) mode */
|
|
|
|
if ((options & STREAM_OPEN_FOR_INCLUDE) && !PG(allow_url_include_list) ) {
|
2006-11-06 04:44:52 +08:00
|
|
|
if (options & REPORT_ERRORS) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "URL file-access is disabled in the server configuration");
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
2005-11-18 03:39:39 +08:00
|
|
|
return php_stream_alloc(&php_stream_input_ops, ecalloc(1, sizeof(off_t)), 0, "rb");
|
2002-11-13 02:29:11 +08:00
|
|
|
}
|
2002-10-22 00:41:06 +08:00
|
|
|
|
2002-03-16 05:03:08 +08:00
|
|
|
if (!strcasecmp(path, "stdin")) {
|
2007-01-25 05:43:47 +08:00
|
|
|
/* Override default behavior for php://stdin when used as an include and allow_url_include is being used in BC (off) mode */
|
|
|
|
if ((options & STREAM_OPEN_FOR_INCLUDE) && !PG(allow_url_include_list) ) {
|
2006-11-06 04:44:52 +08:00
|
|
|
if (options & REPORT_ERRORS) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "URL file-access is disabled in the server configuration");
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
2006-11-03 21:34:39 +08:00
|
|
|
if (!strcmp(sapi_module.name, "cli")) {
|
|
|
|
static int cli_in = 0;
|
|
|
|
fd = STDIN_FILENO;
|
|
|
|
if (cli_in) {
|
|
|
|
fd = dup(fd);
|
|
|
|
} else {
|
|
|
|
cli_in = 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
fd = dup(STDIN_FILENO);
|
|
|
|
}
|
2002-03-16 05:03:08 +08:00
|
|
|
} else if (!strcasecmp(path, "stdout")) {
|
2006-11-03 21:34:39 +08:00
|
|
|
if (!strcmp(sapi_module.name, "cli")) {
|
|
|
|
static int cli_out = 0;
|
|
|
|
fd = STDOUT_FILENO;
|
|
|
|
if (cli_out++) {
|
|
|
|
fd = dup(fd);
|
|
|
|
} else {
|
|
|
|
cli_out = 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
fd = dup(STDOUT_FILENO);
|
|
|
|
}
|
2002-03-16 05:03:08 +08:00
|
|
|
} else if (!strcasecmp(path, "stderr")) {
|
2006-11-03 21:34:39 +08:00
|
|
|
if (!strcmp(sapi_module.name, "cli")) {
|
|
|
|
static int cli_err = 0;
|
|
|
|
fd = STDERR_FILENO;
|
|
|
|
if (cli_err++) {
|
|
|
|
fd = dup(fd);
|
|
|
|
} else {
|
|
|
|
cli_err = 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
fd = dup(STDERR_FILENO);
|
|
|
|
}
|
2003-03-19 00:40:29 +08:00
|
|
|
} else if (!strncasecmp(path, "filter/", 7)) {
|
2003-02-21 04:05:31 +08:00
|
|
|
/* Save time/memory when chain isn't specified */
|
|
|
|
if (strchr(mode, 'r') || strchr(mode, '+')) {
|
|
|
|
mode_rw |= PHP_STREAM_FILTER_READ;
|
|
|
|
}
|
|
|
|
if (strchr(mode, 'w') || strchr(mode, '+') || strchr(mode, 'a')) {
|
|
|
|
mode_rw |= PHP_STREAM_FILTER_WRITE;
|
|
|
|
}
|
2003-02-18 14:10:14 +08:00
|
|
|
pathdup = estrndup(path + 6, strlen(path + 6));
|
|
|
|
p = strstr(pathdup, "/resource=");
|
|
|
|
if (!p) {
|
2005-10-29 23:09:12 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_ERROR, "No URL resource specified");
|
2003-02-18 14:10:14 +08:00
|
|
|
efree(pathdup);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (!(stream = php_stream_open_wrapper(p + 10, mode, options, opened_path))) {
|
|
|
|
efree(pathdup);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
*p = '\0';
|
|
|
|
|
|
|
|
p = php_strtok_r(pathdup + 1, "/", &token);
|
|
|
|
while (p) {
|
|
|
|
if (!strncasecmp(p, "read=", 5)) {
|
|
|
|
php_stream_apply_filter_list(stream, p + 5, 1, 0 TSRMLS_CC);
|
|
|
|
} else if (!strncasecmp(p, "write=", 6)) {
|
|
|
|
php_stream_apply_filter_list(stream, p + 6, 0, 1 TSRMLS_CC);
|
|
|
|
} else {
|
2003-02-21 04:05:31 +08:00
|
|
|
php_stream_apply_filter_list(stream, p, mode_rw & PHP_STREAM_FILTER_READ, mode_rw & PHP_STREAM_FILTER_WRITE TSRMLS_CC);
|
2003-02-18 14:10:14 +08:00
|
|
|
}
|
|
|
|
p = php_strtok_r(NULL, "/", &token);
|
|
|
|
}
|
|
|
|
efree(pathdup);
|
2003-03-19 00:40:29 +08:00
|
|
|
|
|
|
|
return stream;
|
|
|
|
} else {
|
|
|
|
/* invalid php://thingy */
|
2006-12-18 22:55:23 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid php:// URL specified");
|
2003-03-19 00:40:29 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* must be stdin, stderr or stdout */
|
|
|
|
if (fd == -1) {
|
|
|
|
/* failed to dup */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2003-03-19 07:37:54 +08:00
|
|
|
stream = php_stream_fopen_from_fd(fd, mode, NULL);
|
2003-03-19 00:40:29 +08:00
|
|
|
if (stream == NULL) {
|
|
|
|
close(fd);
|
|
|
|
}
|
2003-02-18 14:10:14 +08:00
|
|
|
|
2002-03-16 05:03:08 +08:00
|
|
|
return stream;
|
2000-10-13 08:09:31 +08:00
|
|
|
}
|
2002-03-16 05:03:08 +08:00
|
|
|
|
2002-03-25 02:05:49 +08:00
|
|
|
static php_stream_wrapper_ops php_stdio_wops = {
|
2002-03-16 05:03:08 +08:00
|
|
|
php_stream_url_wrap_php,
|
2002-09-26 18:14:41 +08:00
|
|
|
NULL, /* close */
|
|
|
|
NULL, /* fstat */
|
|
|
|
NULL, /* stat */
|
|
|
|
NULL, /* opendir */
|
2003-05-14 14:10:04 +08:00
|
|
|
"PHP",
|
2003-12-13 07:06:42 +08:00
|
|
|
NULL, /* unlink */
|
2003-12-13 12:07:18 +08:00
|
|
|
NULL, /* rename */
|
|
|
|
NULL, /* mkdir */
|
|
|
|
NULL /* rmdir */
|
2002-03-25 02:05:49 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
php_stream_wrapper php_stream_php_wrapper = {
|
|
|
|
&php_stdio_wops,
|
2002-04-17 06:14:27 +08:00
|
|
|
NULL,
|
|
|
|
0, /* is_url */
|
2002-03-16 05:03:08 +08:00
|
|
|
};
|
|
|
|
|
2001-06-05 21:12:10 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Local variables:
|
|
|
|
* tab-width: 4
|
|
|
|
* c-basic-offset: 4
|
|
|
|
* End:
|
2001-09-09 21:29:31 +08:00
|
|
|
* vim600: sw=4 ts=4 fdm=marker
|
|
|
|
* vim<600: sw=4 ts=4
|
2001-06-05 21:12:10 +08:00
|
|
|
*/
|