summaryrefslogtreecommitdiff
path: root/main
diff options
context:
space:
mode:
authorWez Furlong <wez@php.net>2002-03-15 21:03:08 +0000
committerWez Furlong <wez@php.net>2002-03-15 21:03:08 +0000
commit0f65280cb5118d8c1a85db6626f7be365f3d1b26 (patch)
tree931b09acc5041eb771017e3ebf9ecb9aa833d722 /main
parent3a1ebd4f519facbd7ec769304857aad40e49cf1c (diff)
downloadphp-git-0f65280cb5118d8c1a85db6626f7be365f3d1b26.tar.gz
New PHP streams...
Diffstat (limited to 'main')
-rw-r--r--main/fopen_wrappers.c137
-rw-r--r--main/fopen_wrappers.h20
-rw-r--r--main/internal_functions.c.in6
-rw-r--r--main/main.c42
-rw-r--r--main/network.c403
-rw-r--r--main/php_network.h54
-rwxr-xr-xmain/php_streams.h55
-rwxr-xr-xmain/streams.c595
8 files changed, 1099 insertions, 213 deletions
diff --git a/main/fopen_wrappers.c b/main/fopen_wrappers.c
index 7cf55651d4..9bbd126571 100644
--- a/main/fopen_wrappers.c
+++ b/main/fopen_wrappers.c
@@ -82,55 +82,6 @@
#endif
/* }}} */
-static FILE *php_fopen_url_wrapper(const char *, char *, int, int *, int *, char ** TSRMLS_DC);
-static HashTable fopen_url_wrappers_hash;
-
-/* {{{ php_register_url_wrapper
- */
-PHPAPI int php_register_url_wrapper(const char *protocol, php_fopen_url_wrapper_t wrapper TSRMLS_DC)
-{
- if(PG(allow_url_fopen)) {
- return zend_hash_add(&fopen_url_wrappers_hash, (char *) protocol, strlen(protocol), &wrapper, sizeof(wrapper), NULL);
- } else {
- return FAILURE;
- }
-}
-/* }}} */
-
-/* {{{ php_unregister_url_wrapper
- */
-PHPAPI int php_unregister_url_wrapper(char *protocol TSRMLS_DC)
-{
- if(PG(allow_url_fopen)) {
- return zend_hash_del(&fopen_url_wrappers_hash, protocol, strlen(protocol));
- } else {
- return SUCCESS;
- }
-}
-/* }}} */
-
-/* {{{ php_init_fopen_wrappers
- */
-int php_init_fopen_wrappers(TSRMLS_D)
-{
- if(PG(allow_url_fopen)) {
- return zend_hash_init(&fopen_url_wrappers_hash, 0, NULL, NULL, 1);
- }
- return SUCCESS;
-}
-/* }}} */
-
-/* {{{ php_shutdown_fopen_wrappers
- */
-int php_shutdown_fopen_wrappers(TSRMLS_D)
-{
- if(PG(allow_url_fopen)) {
- zend_hash_destroy(&fopen_url_wrappers_hash);
- }
- return SUCCESS;
-}
-/* }}} */
-
/* {{{ php_check_specific_open_basedir
When open_basedir is not NULL, check if the given filename is located in
open_basedir. Returns -1 if error or not in the open_basedir, else 0
@@ -289,36 +240,6 @@ static FILE *php_fopen_and_set_opened_path(const char *path, char *mode, char **
}
/* }}} */
-/* {{{ php_fopen_wrapper
- */
-PHPAPI FILE *php_fopen_wrapper(char *path, char *mode, int options, int *issock, int *socketd, char **opened_path TSRMLS_DC)
-{
- if (opened_path) {
- *opened_path = NULL;
- }
-
- if(!path || !*path) {
- return NULL;
- }
-
-
- if(PG(allow_url_fopen)) {
- if (!(options & IGNORE_URL)) {
- return php_fopen_url_wrapper(path, mode, options, issock, socketd, opened_path TSRMLS_CC);
- }
- }
-
- if (options & USE_PATH && PG(include_path) != NULL) {
- return php_fopen_with_path(path, mode, PG(include_path), opened_path TSRMLS_CC);
- } else {
- if (options & ENFORCE_SAFE_MODE && PG(safe_mode) && (!php_checkuid(path, mode, CHECKUID_CHECK_MODE_PARAM))) {
- return NULL;
- }
- return php_fopen_and_set_opened_path(path, mode, opened_path TSRMLS_CC);
- }
-}
-/* }}} */
-
/* {{{ php_fopen_primary_script
*/
PHPAPI int php_fopen_primary_script(zend_file_handle *file_handle TSRMLS_DC)
@@ -535,64 +456,6 @@ PHPAPI FILE *php_fopen_with_path(char *filename, char *mode, char *path, char **
}
/* }}} */
-/* {{{ php_fopen_url_wrapper
- */
-static FILE *php_fopen_url_wrapper(const char *path, char *mode, int options, int *issock, int *socketd, char **opened_path TSRMLS_DC)
-{
- FILE *fp = NULL;
- const char *p;
- const char *protocol=NULL;
- int n=0;
-
- for (p=path; isalnum((int)*p); p++) {
- n++;
- }
- if ((*p==':')&&(n>1)) {
- protocol=path;
- }
-
- if (protocol) {
- php_fopen_url_wrapper_t *wrapper=NULL;
-
- if (FAILURE==zend_hash_find(&fopen_url_wrappers_hash, (char *) protocol, n, (void **)&wrapper)) {
- wrapper=NULL;
- protocol=NULL;
- }
- if (wrapper) {
- return (*wrapper)(path, mode, options, issock, socketd, opened_path TSRMLS_CC);
- }
- }
-
- if (!protocol || !strncasecmp(protocol, "file", n)){
- *issock = 0;
-
- if(protocol) {
- if(path[n+1]=='/') {
- if(path[n+2]=='/') {
- php_error(E_WARNING, "remote host file access not supported, %s", path);
- return NULL;
- }
- }
- path+= n+1;
- }
-
- if (options & USE_PATH) {
- fp = php_fopen_with_path((char *) path, mode, PG(include_path), opened_path TSRMLS_CC);
- } else {
- if (options & ENFORCE_SAFE_MODE && PG(safe_mode) && (!php_checkuid(path, mode, CHECKUID_CHECK_MODE_PARAM))) {
- fp = NULL;
- } else {
- fp = php_fopen_and_set_opened_path(path, mode, opened_path TSRMLS_CC);
- }
- }
- return (fp);
- }
-
- php_error(E_WARNING, "Invalid URL specified, %s", path);
- return NULL;
-}
-/* }}} */
-
/* {{{ php_strip_url_passwd
*/
PHPAPI char *php_strip_url_passwd(char *url)
diff --git a/main/fopen_wrappers.h b/main/fopen_wrappers.h
index dc28af9f47..24e60dcc00 100644
--- a/main/fopen_wrappers.h
+++ b/main/fopen_wrappers.h
@@ -27,12 +27,14 @@
#define IGNORE_URL 2
/* There's no USE_URL. */
#ifdef PHP_WIN32
-# define IGNORE_URL_WIN 2
+# define IGNORE_URL_WIN IGNORE_URL
#else
# define IGNORE_URL_WIN 0
#endif
#define ENFORCE_SAFE_MODE 4
+#define REPORT_ERRORS 8
+
#ifdef PHP_WIN32
# define SOCK_ERR INVALID_SOCKET
# define SOCK_CONN_ERR SOCKET_ERROR
@@ -42,22 +44,6 @@
# define SOCK_CONN_ERR -1
# define SOCK_RECV_ERR -1
#endif
-#define SOCK_WRITE(d, s) send(s, d, strlen(d), 0)
-#define SOCK_WRITEL(d, l, s) send(s, d, l, 0)
-#define SOCK_FGETC(s) php_sock_fgetc((s))
-#define SOCK_FGETS(b, l, s) php_sock_fgets((b), (l), (s))
-#define SOCK_FEOF(sock) php_sock_feof((sock))
-#define SOCK_FREAD(ptr, size, sock) php_sock_fread((ptr), (size), (sock))
-#define SOCK_FCLOSE(s) php_sock_close(s)
-
-#define FP_FGETS(buf, len, sock, fp, issock) \
- ((issock)?SOCK_FGETS(buf, len, sock):fgets(buf, len, fp))
-#define FP_FREAD(buf, len, sock, fp, issock) \
- ((issock)?SOCK_FREAD(buf, len, sock):fread(buf, 1, len, fp))
-#define FP_FEOF(sock, fp, issock) \
- ((issock)?SOCK_FEOF(sock):feof(fp))
-#define FP_FGETC(sock, fp, issock) \
- ((issock)?SOCK_FGETC(sock):fgetc(fp))
/* values for issock */
#define IS_NOT_SOCKET 0
diff --git a/main/internal_functions.c.in b/main/internal_functions.c.in
index f8d840d755..4903d6c4bb 100644
--- a/main/internal_functions.c.in
+++ b/main/internal_functions.c.in
@@ -29,6 +29,12 @@
#include <stdlib.h>
#include <stdio.h>
+#if HAVE_OPENSSL_EXT
+/* zlib typedefs free_func which causes problems if the SSL includes happen
+ * after zlib.h is included */
+# include <openssl/ssl.h>
+#endif
+
@EXT_INCLUDE_CODE@
zend_module_entry *php_builtin_extensions[] = {
diff --git a/main/main.c b/main/main.c
index 4d7d0b7cf3..e6d968dedb 100644
--- a/main/main.c
+++ b/main/main.c
@@ -73,6 +73,7 @@
#include "php_content_types.h"
#include "php_ticks.h"
#include "php_logos.h"
+#include "php_streams.h"
#include "SAPI.h"
/* }}} */
@@ -562,17 +563,24 @@ PHP_FUNCTION(set_time_limit)
*/
static FILE *php_fopen_wrapper_for_zend(const char *filename, char **opened_path)
{
- int issock=0, socketd=0;
- int old_chunk_size;
- FILE *retval;
+ FILE *retval = NULL;
+ php_stream * stream;
TSRMLS_FETCH();
-
- old_chunk_size = php_sock_set_def_chunk_size(1);
- retval=php_fopen_wrapper((char *) filename, "rb", USE_PATH|IGNORE_URL_WIN, &issock, &socketd, opened_path TSRMLS_CC);
- php_sock_set_def_chunk_size(old_chunk_size);
-
- if (issock) {
- retval = fdopen(socketd, "rb");
+
+ stream = php_stream_open_wrapper((char *)filename, "rb", USE_PATH|IGNORE_URL_WIN|REPORT_ERRORS, opened_path TSRMLS_CC);
+ if (stream) {
+ /* no need for us to check the stream type here */
+ php_stream_sock_set_chunk_size(stream, 1);
+
+ if (php_stream_cast(stream, PHP_STREAM_AS_STDIO | PHP_STREAM_CAST_TRY_HARD, (void**)&retval, 1) == SUCCESS) {
+ ZEND_REGISTER_RESOURCE(NULL, stream, php_file_le_stream());
+ }
+ else {
+ php_stream_close(stream);
+ if (opened_path && *opened_path)
+ efree(*opened_path);
+ retval = NULL;
+ }
}
return retval;
}
@@ -941,14 +949,14 @@ int php_module_startup(sapi_module_struct *sf)
REGISTER_INI_ENTRIES();
- /* initialize fopen wrappers registry
- (this uses configuration parameters from php.ini)
+ /* initialize stream wrappers registry
+ * (this uses configuration parameters from php.ini)
*/
- if (php_init_fopen_wrappers(TSRMLS_C) == FAILURE) {
- php_printf("PHP: Unable to initialize fopen url wrappers.\n");
+ if (php_init_stream_wrappers(TSRMLS_C) == FAILURE) {
+ php_printf("PHP: Unable to initialize stream url wrappers.\n");
return FAILURE;
}
-
+
/* initialize registry for images to be used in phpinfo()
(this uses configuration parameters from php.ini)
*/
@@ -1048,7 +1056,9 @@ void php_module_shutdown(TSRMLS_D)
sapi_flush(TSRMLS_C);
zend_shutdown(TSRMLS_C);
- php_shutdown_fopen_wrappers(TSRMLS_C);
+
+ php_shutdown_stream_wrappers(TSRMLS_C);
+
php_shutdown_info_logos();
UNREGISTER_INI_ENTRIES();
diff --git a/main/network.c b/main/network.c
index 6be650875a..1aedbe416a 100644
--- a/main/network.c
+++ b/main/network.c
@@ -17,6 +17,9 @@
*/
/* $Id$ */
+#define PHP_SOCK_CHUNK_SIZE 8192
+#define MAX_CHUNKS_PER_READ 10
+
#include "php.h"
#ifdef PHP_WIN32
@@ -408,6 +411,406 @@ int php_sockaddr_size(php_sockaddr_storage *addr)
}
/* }}} */
+PHPAPI php_stream * php_stream_sock_open_from_socket(int socket, int persistent)
+{
+ php_stream * stream;
+ php_netstream_data_t * sock;
+
+ sock = pemalloc(sizeof(php_netstream_data_t), persistent);
+ memset(sock, 0, sizeof(php_netstream_data_t));
+
+ sock->is_blocked = 1;
+ sock->chunk_size = PHP_SOCK_CHUNK_SIZE;
+ sock->timeout.tv_sec = -1;
+ sock->socket = socket;
+
+ stream = php_stream_alloc(&php_stream_socket_ops, sock, persistent, "r+");
+
+ if (stream == NULL)
+ pefree(sock, persistent);
+
+ return stream;
+}
+
+PHPAPI php_stream * php_stream_sock_open_host(const char * host, unsigned short port,
+ int socktype, int timeout, int persistent)
+{
+ int socket;
+
+ socket = php_hostconnect(host, port, socktype, timeout);
+
+ if (socket == -1)
+ return NULL;
+
+ return php_stream_sock_open_from_socket(socket, persistent);
+}
+
+PHPAPI php_stream * php_stream_sock_open_unix(const char * path, int persistent, struct timeval * timeout)
+{
+#if defined(AF_UNIX)
+ int socketd;
+ struct sockaddr_un unix_addr;
+
+ socketd = socket(PF_UNIX, SOCK_STREAM, 0);
+ if (socketd == SOCK_ERR)
+ return NULL;
+
+ memset(&unix_addr, 0, sizeof(unix_addr));
+ unix_addr.sun_family = AF_UNIX;
+ strlcpy(unix_addr.sun_path, path, sizeof(unix_addr.sun_path));
+
+ if (php_connect_nonb(socketd, (struct sockaddr *) &unix_addr, sizeof(unix_addr), timeout) == SOCK_CONN_ERR)
+ return NULL;
+
+ return php_stream_sock_open_from_socket(socketd, persistent);
+#else
+ return NULL;
+#endif
+}
+
+#if HAVE_OPENSSL_EXT
+PHPAPI int php_stream_sock_ssl_activate_with_method(php_stream * stream, int activate, SSL_METHOD * method)
+{
+ php_netstream_data_t * sock = (php_netstream_data_t*)stream->abstract;
+ SSL_CTX * ctx = NULL;
+
+ if (!php_stream_is(stream, PHP_STREAM_IS_SOCKET))
+ return FAILURE;
+
+ if (activate == sock->ssl_active)
+ return SUCCESS; /* already in desired mode */
+
+ if (activate && sock->ssl_handle == NULL) {
+ ctx = SSL_CTX_new(method);
+ if (ctx == NULL)
+ return FAILURE;
+
+ sock->ssl_handle = SSL_new(ctx);
+ if (sock->ssl_handle == NULL) {
+ SSL_CTX_free(ctx);
+ return FAILURE;
+ }
+
+ SSL_set_fd(sock->ssl_handle, sock->socket);
+ }
+
+ if (activate) {
+ if (SSL_connect(sock->ssl_handle) <= 0) {
+ SSL_shutdown(sock->ssl_handle);
+ return FAILURE;
+ }
+ sock->ssl_active = activate;
+ }
+ else {
+ SSL_shutdown(sock->ssl_handle);
+ sock->ssl_active = 0;
+ }
+ return SUCCESS;
+}
+#endif
+
+PHPAPI void php_stream_sock_set_timeout(php_stream * stream, struct timeval *timeout)
+{
+ php_netstream_data_t * sock = (php_netstream_data_t*)stream->abstract;
+
+ if (!php_stream_is(stream, PHP_STREAM_IS_SOCKET))
+ return;
+
+ sock->timeout = *timeout;
+ sock->timeout_event = 0;
+}
+
+PHPAPI int php_stream_sock_set_blocking(php_stream * stream, int mode)
+{
+ int oldmode;
+ php_netstream_data_t * sock = (php_netstream_data_t*)stream->abstract;
+
+ if (!php_stream_is(stream, PHP_STREAM_IS_SOCKET))
+ return 0;
+
+ oldmode = sock->is_blocked;
+ sock->is_blocked = mode;
+
+ return oldmode;
+}
+
+PHPAPI size_t php_stream_sock_set_chunk_size(php_stream * stream, size_t size)
+{
+ size_t oldsize;
+ php_netstream_data_t * sock = (php_netstream_data_t*)stream->abstract;
+
+ if (!php_stream_is(stream, PHP_STREAM_IS_SOCKET))
+ return 0;
+
+ oldsize = sock->chunk_size;
+ sock->chunk_size = size;
+
+ return oldsize;
+}
+
+#define TOREAD(sock) ((sock)->writepos - (sock)->readpos)
+#define READPTR(sock) ((sock)->readbuf + (sock)->readpos)
+#define WRITEPTR(sock) ((sock)->readbuf + (sock)->writepos)
+
+static size_t php_sockop_write(php_stream * stream, const char * buf, size_t count)
+{
+ php_netstream_data_t * sock = (php_netstream_data_t*)stream->abstract;
+#if HAVE_OPENSSL_EXT
+ if (sock->ssl_active)
+ return SSL_write(sock->ssl_handle, buf, count);
+#endif
+ return send(sock->socket, buf, count, 0);
+}
+static void php_sock_stream_wait_for_data(php_stream * stream, php_netstream_data_t * sock)
+{
+ fd_set fdr, tfdr;
+ int retval;
+ struct timeval timeout, *ptimeout;
+
+ FD_ZERO(&fdr);
+ FD_SET(sock->socket, &fdr);
+ sock->timeout_event = 0;
+
+ if (sock->timeout.tv_sec == -1)
+ ptimeout = NULL;
+ else
+ ptimeout = &timeout;
+
+ while(1) {
+ tfdr = fdr;
+ timeout = sock->timeout;
+
+ retval = select(sock->socket + 1, &tfdr, NULL, NULL, ptimeout);
+
+ if (retval == 0)
+ sock->timeout_event = 1;
+
+ if (retval >= 0)
+ break;
+ }
+}
+
+static size_t php_sock_stream_read_internal(php_stream * stream, php_netstream_data_t * sock)
+{
+ char buf[PHP_SOCK_CHUNK_SIZE];
+ int nr_bytes;
+ size_t nr_read = 0;
+
+ /* For blocking sockets, we wait until there is some
+ data to read (real data or EOF)
+
+ Otherwise, recv() may time out and return 0 and
+ therefore sock->eof would be set errornously.
+ */
+
+
+ if(sock->is_blocked) {
+ php_sock_stream_wait_for_data(stream, sock);
+ if (sock->timeout_event)
+ return 0;
+ }
+
+ /* read at a maximum sock->chunk_size */
+#if HAVE_OPENSSL_EXT
+ if (sock->ssl_active)
+ nr_bytes = SSL_read(sock->ssl_handle, buf, sock->chunk_size);
+ else
+#endif
+ nr_bytes = recv(sock->socket, buf, sock->chunk_size, 0);
+ if(nr_bytes > 0) {
+ if(sock->writepos + nr_bytes > sock->readbuflen) {
+ sock->readbuflen += sock->chunk_size;
+ sock->readbuf = perealloc(sock->readbuf, sock->readbuflen,
+ php_stream_is_persistent(stream));
+ }
+ memcpy(WRITEPTR(sock), buf, nr_bytes);
+ sock->writepos += nr_bytes;
+ nr_read = nr_bytes;
+ } else if(nr_bytes == 0 || (nr_bytes < 0 && errno != EWOULDBLOCK)) {
+ sock->eof = 1;
+ }
+
+ return nr_read;
+
+}
+
+static size_t php_sock_stream_read(php_stream * stream, php_netstream_data_t * sock)
+{
+ size_t nr_bytes;
+ size_t nr_read = 0;
+ int i;
+
+ for(i = 0; !sock->eof && i < MAX_CHUNKS_PER_READ; i++) {
+ nr_bytes = php_sock_stream_read_internal(stream, sock);
+ if(nr_bytes == 0) break;
+ nr_read += nr_bytes;
+ }
+
+ return nr_read;
+}
+
+static size_t php_sockop_read(php_stream * stream, char * buf, size_t count)
+{
+ php_netstream_data_t * sock = (php_netstream_data_t*)stream->abstract;
+ size_t ret = 0;
+
+ if (sock->is_blocked) {
+ while(!sock->eof && TOREAD(sock) < count && !sock->timeout_event)
+ php_sock_stream_read_internal(stream, sock);
+ }
+ else
+ php_sock_stream_read(stream, sock);
+
+ if(count < 0)
+ return ret;
+
+ ret = MIN(TOREAD(sock), count);
+ if (ret) {
+ memcpy(buf, READPTR(sock), ret);
+ sock->readpos += ret;
+ }
+
+ return ret;
+}
+
+static int php_sockop_close(php_stream * stream)
+{
+ php_netstream_data_t * sock = (php_netstream_data_t*)stream->abstract;
+
+#if HAVE_OPENSSL_EXT
+ if (sock->ssl_active) {
+ SSL_shutdown(sock->ssl_handle);
+ sock->ssl_active = 0;
+ SSL_free(sock->ssl_handle);
+ sock->ssl_handle = NULL;
+ }
+#endif
+
+ shutdown(sock->socket, 0);
+ closesocket(sock->socket);
+
+ if (sock->readbuf)
+ pefree(sock->readbuf, php_stream_is_persistent(stream));
+
+ pefree(sock, php_stream_is_persistent(stream));
+
+ return 0;
+}
+
+static int php_sockop_flush(php_stream * stream)
+{
+ php_netstream_data_t * sock = (php_netstream_data_t*)stream->abstract;
+ return fsync(sock->socket);
+}
+
+static int php_sockop_cast(php_stream * stream, int castas, void ** ret)
+{
+ php_netstream_data_t * sock = (php_netstream_data_t*)stream->abstract;
+ TSRMLS_FETCH();
+
+ switch(castas) {
+ case PHP_STREAM_AS_STDIO:
+#if HAVE_OPENSSL_EXT
+ if (sock->ssl_active)
+ return FAILURE;
+#endif
+ if (ret) {
+ /* DANGER!: data buffered in stream->readbuf will be forgotten! */
+ if (TOREAD(sock) > 0)
+ zend_error(E_WARNING, "%s(): buffered data lost during conversion to FILE*!", get_active_function_name(TSRMLS_C));
+ *ret = fdopen(sock->socket, stream->mode);
+ if (*ret)
+ return SUCCESS;
+ return FAILURE;
+ }
+ return SUCCESS;
+ case PHP_STREAM_AS_FD:
+ case PHP_STREAM_AS_SOCKETD:
+#if HAVE_OPENSSL_EXT
+ if (sock->ssl_active)
+ return FAILURE;
+#endif
+ if (ret)
+ *ret = (void*)sock->socket;
+ return SUCCESS;
+ default:
+ return FAILURE;
+ }
+}
+
+#define SEARCHCR() do { \
+ if (TOREAD(sock)) { \
+ for (p = READPTR(sock), pe = p + MIN(TOREAD(sock), maxlen); \
+ *p != '\n'; ) \
+ if (++p >= pe) { \
+ p = NULL; \
+ break; \
+ } \
+ } else \
+ p = NULL; \
+} while (0)
+
+
+static char * php_sockop_gets(php_stream * stream, char *buf, size_t maxlen)
+{
+ php_netstream_data_t * sock = (php_netstream_data_t*)stream->abstract;
+ char *p = NULL, *pe;
+ char *ret = NULL;
+ size_t amount = 0;
+
+ if (maxlen==0) {
+ buf[0] = 0;
+ return buf;
+ }
+
+ SEARCHCR();
+
+ if(!p) {
+ if(sock->is_blocked) {
+ while(!p && !sock->eof && !sock->timeout_event && TOREAD(sock) < maxlen) {
+ php_sock_stream_read_internal(stream, sock);
+ SEARCHCR();
+ }
+ } else {
+ php_sock_stream_read(stream, sock);
+ SEARCHCR();
+ }
+ }
+
+ if(p) {
+ amount = (ptrdiff_t) p - (ptrdiff_t) READPTR(sock) + 1;
+ } else {
+ amount = TOREAD(sock);
+ }
+
+ amount = MIN(amount, maxlen);
+
+ if(amount > 0) {
+ memcpy(buf, READPTR(sock), amount);
+ sock->readpos += amount;
+ }
+ buf[amount] = '\0';
+
+ /* signal error only, if we don't return data from this call and
+ if there is no data to read and if the eof flag is set */
+ if(amount || TOREAD(sock) || !sock->eof) {
+ ret = buf;
+ }
+
+ return ret;
+}
+
+php_stream_ops php_stream_socket_ops = {
+ php_sockop_write, php_sockop_read,
+ php_sockop_close, php_sockop_flush,
+ NULL, php_sockop_gets,
+ php_sockop_cast,
+ "socket"
+};
+
+
+
+
/*
* Local variables:
* tab-width: 8
diff --git a/main/php_network.h b/main/php_network.h
index b71737265f..b87e709fff 100644
--- a/main/php_network.h
+++ b/main/php_network.h
@@ -27,8 +27,17 @@
# undef FD_SETSIZE
# include "arpa/inet.h"
# define socklen_t unsigned int
+#else
+# undef closesocket
+# define closesocket close
+#endif
+
+#ifndef HAVE_SHUTDOWN
+#undef shutdown
+#define shutdown(s,n) /* nothing */
#endif
+
#ifdef HAVE_NETINET_IN_H
# include <netinet/in.h>
#endif
@@ -41,6 +50,10 @@
#include <sys/time.h>
#endif
+#if HAVE_OPENSSL_EXT
+#include <openssl/ssl.h>
+#endif
+
#ifdef HAVE_SOCKADDR_STORAGE
typedef struct sockaddr_storage php_sockaddr_storage;
#else
@@ -56,6 +69,47 @@ PHPAPI int php_connect_nonb(int sockfd, const struct sockaddr *addr, socklen_t a
void php_any_addr(int family, php_sockaddr_storage *addr, unsigned short port);
int php_sockaddr_size(php_sockaddr_storage *addr);
+struct _php_netstream_data_t {
+ int socket;
+ unsigned char *readbuf;
+ size_t readbuflen;
+ size_t readpos;
+ size_t writepos;
+ char eof;
+ char is_blocked;
+ size_t chunk_size;
+ struct timeval timeout;
+ char timeout_event;
+#if HAVE_OPENSSL_EXT
+ /* openssl specific bits here */
+ SSL * ssl_handle;
+ int ssl_active;
+#endif
+};
+typedef struct _php_netstream_data_t php_netstream_data_t;
+
+#define PHP_NETSTREAM_DATA_FROM_STREAM(stream) (php_netstream_data_t*)(stream)->abstract
+
+extern php_stream_ops php_stream_socket_ops;
+#define PHP_STREAM_IS_SOCKET (&php_stream_socket_ops)
+
+PHPAPI php_stream * php_stream_sock_open_from_socket(int socket, int persistent);
+/* open a connection to a host using php_hostconnect and return a stream */
+PHPAPI php_stream * php_stream_sock_open_host(const char * host, unsigned short port,
+ int socktype, int timeout, int persistent);
+PHPAPI php_stream * php_stream_sock_open_unix(const char * path, int persistent, struct timeval * timeout);
+
+PHPAPI void php_stream_sock_set_timeout(php_stream * stream, struct timeval *timeout);
+PHPAPI int php_stream_sock_set_blocking(php_stream * stream, int mode);
+/* set the chunk size for the stream; return the old chunk size */
+PHPAPI size_t php_stream_sock_set_chunk_size(php_stream * stream, size_t size);
+
+#if HAVE_OPENSSL_EXT
+PHPAPI int php_stream_sock_ssl_activate_with_method(php_stream * stream, int activate, SSL_METHOD * method);
+#define php_stream_sock_ssl_activate(stream, activate) php_stream_sock_ssl_activate_with_method((stream), (activate), SSLv23_client_method())
+
+#endif
+
#endif /* _PHP_NETWORK_H */
/*
diff --git a/main/php_streams.h b/main/php_streams.h
index 6acaafb3fc..39b8e70a46 100755
--- a/main/php_streams.h
+++ b/main/php_streams.h
@@ -19,8 +19,6 @@
#ifndef PHP_STREAMS_H
#define PHP_STREAMS_H
-#if HAVE_PHP_STREAM
-
#ifdef HAVE_SYS_TIME_H
#include <sys/time.h>
#endif
@@ -42,9 +40,23 @@ typedef struct _php_stream_ops {
const char * label; /* label for this ops structure */
} php_stream_ops;
+/* options uses the IGNORE_URL family of defines from fopen_wrappers.h */
+typedef php_stream * (*php_stream_factory_func_t)(char * filename, char * mode, int options, char ** opened_path TSRMLS_DC);
+typedef void (*php_stream_wrapper_dtor_func_t)(php_stream * stream);
+
+typedef struct _php_stream_wrapper {
+ php_stream_factory_func_t create;
+ php_stream_wrapper_dtor_func_t destroy;
+} php_stream_wrapper;
+
struct _php_stream {
php_stream_ops * ops;
void * abstract; /* convenience pointer for abstraction */
+
+ php_stream_wrapper * wrapper; /* which wrapper was used to open the stream */
+ void * wrapperthis; /* convenience pointer for a instance of a wrapper */
+ zval * wrapperdata; /* fgetwrapperdata retrieves this */
+
int is_persistent;
char mode[16]; /* "rwb" etc. ala stdio */
/* so we know how to clean it up correctly. This should be set to
@@ -68,15 +80,34 @@ PHPAPI int php_stream_seek(php_stream * stream, off_t offset, int whence);
PHPAPI off_t php_stream_tell(php_stream * stream);
PHPAPI size_t php_stream_read(php_stream * stream, char * buf, size_t count);
PHPAPI size_t php_stream_write(php_stream * stream, const char * buf, size_t count);
+#define php_stream_write_string(stream, str) php_stream_write(stream, str, strlen(str))
PHPAPI int php_stream_eof(php_stream * stream);
PHPAPI int php_stream_getc(php_stream * stream);
+PHPAPI int php_stream_putc(php_stream * stream, int c);
PHPAPI int php_stream_flush(php_stream * stream);
PHPAPI char *php_stream_gets(php_stream * stream, char *buf, size_t maxlen);
+PHPAPI int php_stream_puts(php_stream * stream, char * buf);
-/* operations for a stdio FILE; the FILE * must be placed in stream->abstract */
+/* copy up to maxlen bytes from src to dest. If maxlen is 0, copy until eof(src).
+ * Uses mmap if the src is a plain file and at offset 0 */
+PHPAPI size_t php_stream_copy_to_stream(php_stream * src, php_stream * dest, size_t maxlen);
+/* read all data from stream and put into a buffer. Caller must free buffer when done,
+ * according to allocopts.
+ * The copy will use mmap if available. */
+PHPAPI size_t php_stream_read_all(php_stream * src, char ** buf, int persistent);
+
+/* maybe implement someday */
+#define php_stream_error(stream) (0)
+
+/* operations for a stdio FILE; use the php_stream_fopen_XXX funcs below */
extern php_stream_ops php_stream_stdio_ops;
/* like fopen, but returns a stream */
-PHPAPI php_stream * php_stream_fopen(const char * filename, const char * mode);
+PHPAPI php_stream * php_stream_fopen(const char * filename, const char * mode, char **opened_path TSRMLS_DC);
+PHPAPI php_stream * php_stream_fopen_with_path(char * filename, char * mode, char * path, char **opened_path TSRMLS_DC);
+PHPAPI php_stream * php_stream_fopen_from_file(FILE * file, const char * mode);
+PHPAPI php_stream * php_stream_fopen_from_pipe(FILE * file, const char * mode);
+PHPAPI php_stream * php_stream_fopen_tmpfile(void);
+PHPAPI php_stream * php_stream_fopen_temporary_file(const char * dir, const char * pfx, char **opened_path TSRMLS_DC);
/* coerce the stream into some other form */
/* cast as a stdio FILE * */
@@ -86,15 +117,27 @@ PHPAPI php_stream * php_stream_fopen(const char * filename, const char * mode);
/* cast as a socketd */
#define PHP_STREAM_AS_SOCKETD 2
+/* try really, really hard to make sure the cast happens (socketpair) */
+#define PHP_STREAM_CAST_TRY_HARD 0x80000000
+
PHPAPI int php_stream_cast(php_stream * stream, int castas, void ** ret, int show_err);
/* use this to check if a stream can be cast into another form */
#define php_stream_can_cast(stream, as) php_stream_cast(stream, as, NULL, 0)
/* use this to check if a stream is of a particular type:
* PHPAPI int php_stream_is(php_stream * stream, php_stream_ops * ops); */
-#define php_stream_is(stream, anops) (stream->ops == anops)
+#define php_stream_is(stream, anops) ((stream)->ops == anops)
+#define PHP_STREAM_IS_STDIO &php_stream_stdio_ops
+
+#define php_stream_is_persistent(stream) (stream)->is_persistent
+
+/* Wrappers support */
+int php_init_stream_wrappers(TSRMLS_D);
+int php_shutdown_stream_wrappers(TSRMLS_D);
+PHPAPI int php_register_url_stream_wrapper(char * protocol, php_stream_wrapper * wrapper TSRMLS_DC);
+PHPAPI int php_unregister_url_stream_wrapper(char * protocol TSRMLS_DC);
-#endif /* HAVE_PHP_STREAM */
+PHPAPI php_stream * php_stream_open_wrapper(char * path, char * mode, int options, char ** opened_path TSRMLS_DC);
#endif
diff --git a/main/streams.c b/main/streams.c
index b79d04a6d4..58e6ba44bb 100755
--- a/main/streams.c
+++ b/main/streams.c
@@ -12,14 +12,28 @@
| obtain it through the world-wide-web, please send a note to |
| license@php.net so we can mail you a copy immediately. |
+----------------------------------------------------------------------+
- | Author: Wez Furlong (wez@thebrainroom.com) |
+ | Authors: |
+ | Wez Furlong (wez@thebrainroom.com) |
+ | Borrowed code from: |
+ | Rasmus Lerdorf <rasmus@lerdorf.on.ca> |
+ | Jim Winstead <jimw@php.net> |
+----------------------------------------------------------------------+
*/
#define _GNU_SOURCE
#include "php.h"
+#include "php_globals.h"
+#include "php_network.h"
+#include "php_open_temporary_file.h"
+
+#ifdef HAVE_SYS_MMAN_H
+#include <sys/mman.h>
+#endif
+
+#ifndef MAP_FAILED
+#define MAP_FAILED ((void *) -1)
+#endif
-#if HAVE_PHP_STREAM
#ifdef PHP_WIN32
#define EWOULDBLOCK WSAEWOULDBLOCK
@@ -27,18 +41,10 @@
#include "build-defs.h"
#endif
-#define MAX_CHUNK_SIZE 8192
-
-#define TOREAD(stream) ((stream)->readbuf.writepos - (stream)->readbuf.readpos)
-#define TOWRITE(stream) ((stream)->readbuf.writepos - (stream)->readbuf.readpos)
-
-#define READPTR(stream) ((stream)->readbuf.buffer + (stream)->readbuf.readpos)
-#define WRITEPTR(stream) ((stream)->readbuf.buffer + (stream)->readbuf.writepos)
-
-#define READ_MAX(stream, max) if (stream->is_blocked) stream_read_total(sock, max); else stream_readahead(sock)
+static HashTable url_stream_wrappers_hash;
/* allocate a new stream for a particular ops */
-PHPAPI php_stream * php_stream_alloc(php_stream_ops * ops, void * abstract, int persistent, const char * mode)
+PHPAPI php_stream * php_stream_alloc(php_stream_ops * ops, void * abstract, int persistent, const char * mode) /* {{{ */
{
php_stream * ret;
@@ -54,11 +60,18 @@ PHPAPI php_stream * php_stream_alloc(php_stream_ops * ops, void * abstract, int
return ret;
}
+/* }}} */
-PHPAPI int php_stream_free(php_stream * stream, int call_dtor)
+PHPAPI int php_stream_free(php_stream * stream, int call_dtor) /* {{{ */
{
int ret = 1;
+ php_stream_flush(stream);
+
+ if (stream->wrapper && stream->wrapper->destroy) {
+ stream->wrapper->destroy(stream);
+ }
+
if (call_dtor) {
if (stream->fclose_stdiocast == PHP_STREAM_FCLOSE_FOPENCOOKIE) {
@@ -82,11 +95,17 @@ PHPAPI int php_stream_free(php_stream * stream, int call_dtor)
stream->stdiocast = NULL;
}
}
+
+ if (stream->wrapperdata) {
+ FREE_ZVAL(stream->wrapperdata);
+ }
pefree(stream, stream->is_persistent);
return ret;
}
+/* }}} */
+/* {{{ generic stream operations */
PHPAPI size_t php_stream_read(php_stream * stream, char * buf, size_t size)
{
return stream->ops->read(stream, buf, size);
@@ -100,6 +119,14 @@ PHPAPI int php_stream_eof(php_stream * stream)
return stream->ops->read(stream, NULL, 0) == EOF ? 1 : 0;
}
+PHPAPI int php_stream_putc(php_stream * stream, int c)
+{
+ unsigned char buf = c;
+ if (php_stream_write(stream, &buf, 1) > 0)
+ return 1;
+ return EOF;
+}
+
PHPAPI int php_stream_getc(php_stream * stream)
{
char buf;
@@ -109,6 +136,17 @@ PHPAPI int php_stream_getc(php_stream * stream)
return EOF;
}
+PHPAPI int php_stream_puts(php_stream * stream, char * buf)
+{
+ int len;
+ char newline[2] = "\n"; /* is this OK for Win? */
+ len = strlen(buf);
+
+ if (len > 0 && php_stream_write(stream, buf, len) && php_stream_write(stream, newline, 1))
+ return 1;
+ return 0;
+}
+
PHPAPI char *php_stream_gets(php_stream * stream, char *buf, size_t maxlen)
{
@@ -142,18 +180,13 @@ PHPAPI char *php_stream_gets(php_stream * stream, char *buf, size_t maxlen)
PHPAPI int php_stream_flush(php_stream * stream)
{
- return stream->ops->flush(stream);
+ if (stream->ops->flush)
+ return stream->ops->flush(stream);
+ return 0;
}
PHPAPI size_t php_stream_write(php_stream * stream, const char * buf, size_t count)
{
- if (strchr(stream->mode, 'w') == NULL) {
- TSRMLS_FETCH();
-
- zend_error(E_WARNING, "%s(): stream was not opened for writing", get_active_function_name(TSRMLS_C));
- return 0;
- }
-
return stream->ops->write(stream, buf, count);
}
@@ -171,60 +204,310 @@ PHPAPI int php_stream_seek(php_stream * stream, off_t offset, int whence)
if (stream->ops->seek)
return stream->ops->seek(stream, offset, whence);
+ /* emulate forward moving seeks with reads */
+ if (whence == SEEK_CUR && offset > 0) {
+ while(offset-- > 0)
+ if (php_stream_getc(stream) == EOF)
+ return -1;
+ return 0;
+ }
+
zend_error(E_WARNING, "streams of type %s do not support seeking", stream->ops->label);
return -1;
}
-/*------- STDIO stream implementation -------*/
+#define CHUNK_SIZE 8192
+
+PHPAPI size_t php_stream_read_all(php_stream * src, char ** buf, int persistent)
+{
+ size_t ret = 0;
+ char * ptr;
+ size_t len = 0, max_len;
+ int step = CHUNK_SIZE;
+ int min_room = CHUNK_SIZE / 4;
+#if HAVE_MMAP
+ int srcfd;
+#endif
+
+#if HAVE_MMAP
+ /* try and optimize the case where we are copying from the start of a plain file.
+ * We could probably make this work in more situations, but I don't trust the stdio
+ * buffering layer.
+ * */
+ if ( php_stream_is(src, PHP_STREAM_IS_STDIO) &&
+ php_stream_tell(src) == 0 &&
+ SUCCESS == php_stream_cast(src, PHP_STREAM_AS_FD, (void**)&srcfd, 0))
+ {
+ struct stat sbuf;
+
+ if (fstat(srcfd, &sbuf) == 0) {
+ void * srcfile;
+
+ srcfile = mmap(NULL, sbuf.st_size, PROT_READ, MAP_SHARED, srcfd, 0);
+ if (srcfile != (void*)MAP_FAILED) {
+
+ *buf = pemalloc(persistent, sbuf.st_size);
+
+ if (*buf) {
+ memcpy(*buf, srcfile, sbuf.st_size);
+ ret = sbuf.st_size;
+ }
+
+ munmap(srcfile, sbuf.st_size);
+ return ret;
+ }
+ }
+ /* fall through - we might be able to copy in smaller chunks */
+ }
+#endif
+
+ ptr = *buf = pemalloc(persistent, step);
+ max_len = step;
+
+ while((ret = php_stream_read(src, ptr, max_len - len))) {
+ len += ret;
+ if (len + min_room >= max_len) {
+ *buf = perealloc(*buf, max_len + step, persistent);
+ max_len += step;
+ ptr = *buf + len;
+ }
+ }
+ if (len) {
+ *buf = perealloc(*buf, len, persistent);
+ }
+ else {
+ pefree(*buf, persistent);
+ *buf = NULL;
+ }
+ return len;
+}
+
+PHPAPI size_t php_stream_copy_to_stream(php_stream * src, php_stream * dest, size_t maxlen)
+{
+ char buf[CHUNK_SIZE];
+ size_t readchunk;
+ size_t haveread = 0;
+ size_t didread;
+#if HAVE_MMAP
+ int srcfd;
+#endif
+
+#if HAVE_MMAP
+ /* try and optimize the case where we are copying from the start of a plain file.
+ * We could probably make this work in more situations, but I don't trust the stdio
+ * buffering layer.
+ * */
+ if ( php_stream_is(src, PHP_STREAM_IS_STDIO) &&
+ php_stream_tell(src) == 0 &&
+ SUCCESS == php_stream_cast(src, PHP_STREAM_AS_FD, (void**)&srcfd, 0))
+ {
+ struct stat sbuf;
+
+ if (fstat(srcfd, &sbuf) == 0) {
+ void * srcfile;
+
+ srcfile = mmap(NULL, sbuf.st_size, PROT_READ, MAP_SHARED, srcfd, 0);
+ if (srcfile != (void*)MAP_FAILED) {
+ haveread = php_stream_write(dest, srcfile, sbuf.st_size);
+ munmap(srcfile, sbuf.st_size);
+ return haveread;
+ }
+ }
+ /* fall through - we might be able to copy in smaller chunks */
+ }
+#endif
+
+ while(1) {
+ readchunk = sizeof(buf);
+
+ if (maxlen && (maxlen - haveread) < readchunk)
+ readchunk = maxlen - haveread;
+
+ didread = php_stream_read(src, buf, readchunk);
+ if (didread) {
+ /* extra paranoid */
+ size_t didwrite, towrite;
+ char * writeptr;
+
+ towrite = didread;
+ writeptr = buf;
+ haveread += didread;
+
+ while(towrite) {
+ didwrite = php_stream_write(dest, writeptr, towrite);
+
+ if (didwrite == 0)
+ return 0; /* error */
+
+ towrite -= didwrite;
+ writeptr += didwrite;
+ }
+ }
+ else
+ return 0; /* error */
+
+ if (maxlen - haveread == 0)
+ break;
+ }
+
+ return haveread;
+
+}
+/* }}} */
+
+
+
+/* {{{ ------- STDIO stream implementation -------*/
+
+typedef struct {
+ FILE * file;
+ int is_pipe; /* use pclose */
+#if HAVE_FLUSHIO
+ char last_op;
+#endif
+} php_stdio_stream_data;
+
+PHPAPI php_stream * php_stream_fopen_temporary_file(const char * dir, const char * pfx, char **opened_path TSRMLS_DC)
+{
+ FILE * fp = php_open_temporary_file(dir, pfx, opened_path);
+
+ if (fp) {
+ php_stream * stream = php_stream_fopen_from_file(fp, "wb");
+ if (stream)
+ return stream;
+ fclose(fp);
+
+ zend_error(E_WARNING, "%s(): unable to allocate stream", get_active_function_name(TSRMLS_C));
+
+ return NULL;
+ }
+ return NULL;
+}
+
+PHPAPI php_stream * php_stream_fopen_tmpfile(void)
+{
+ FILE * fp;
+ php_stream * stream;
+
+ fp = tmpfile();
+ if (fp == NULL) {
+ zend_error(E_WARNING, "tmpfile(): %s", strerror(errno));
+ return NULL;
+ }
+ stream = php_stream_fopen_from_file(fp, "r+");
+ if (stream == NULL) {
+ zend_error(E_WARNING, "tmpfile(): %s", strerror(errno));
+ fclose(fp);
+ return NULL;
+ }
+ return stream;
+}
+
+
+PHPAPI php_stream * php_stream_fopen_from_file(FILE * file, const char * mode)
+{
+ php_stdio_stream_data * self;
+
+ self = emalloc(sizeof(*self));
+ self->file = file;
+ self->is_pipe = 0;
+ return php_stream_alloc(&php_stream_stdio_ops, self, 0, mode);
+}
+
+PHPAPI php_stream * php_stream_fopen_from_pipe(FILE * file, const char * mode)
+{
+ php_stdio_stream_data * self;
+
+ self = emalloc(sizeof(*self));
+ self->file = file;
+ self->is_pipe = 1;
+ return php_stream_alloc(&php_stream_stdio_ops, self, 0, mode);
+}
static size_t php_stdiop_write(php_stream * stream, const char * buf, size_t count)
{
- return fwrite(buf, 1, count, (FILE*)stream->abstract);
+ php_stdio_stream_data * data = (php_stdio_stream_data*)stream->abstract;
+
+#if HAVE_FLUSHIO
+ if (data->last_op == 'r')
+ fseek(data->file, 0, SEEK_CUR);
+ data->last_op = 'w';
+#endif
+
+ return fwrite(buf, 1, count, data->file);
}
static size_t php_stdiop_read(php_stream * stream, char * buf, size_t count)
{
+ php_stdio_stream_data * data = (php_stdio_stream_data*)stream->abstract;
+
if (buf == NULL && count == 0) {
/* check for EOF condition */
- if (feof((FILE*)stream->abstract)) {
+ if (feof(data->file)) {
return EOF;
}
return 0;
}
- return fread(buf, 1, count, (FILE*)stream->abstract);
+
+#if HAVE_FLUSHIO
+ if (data->last_op == 'w')
+ fseek(data->file, 0, SEEK_CUR);
+ data->last_op = 'r';
+#endif
+
+ return fread(buf, 1, count, data->file);
}
static int php_stdiop_close(php_stream * stream)
{
- return fclose((FILE*)stream->abstract);
+ int ret;
+ php_stdio_stream_data * data = (php_stdio_stream_data*)stream->abstract;
+
+ if (data->is_pipe)
+ ret = pclose(data->file);
+ else
+ ret = fclose(data->file);
+
+ efree(data);
+
+ return ret;
}
static int php_stdiop_flush(php_stream * stream)
{
- return fflush((FILE*)stream->abstract);
+ return fflush(((php_stdio_stream_data*)stream->abstract)->file);
}
static int php_stdiop_seek(php_stream * stream, off_t offset, int whence)
{
- return fseek((FILE*)stream->abstract, offset, whence);
+ return fseek(((php_stdio_stream_data*)stream->abstract)->file, offset, whence);
}
static char * php_stdiop_gets(php_stream * stream, char * buf, size_t size)
{
- return fgets(buf, size, (FILE*)stream->abstract);
+ php_stdio_stream_data * data = (php_stdio_stream_data*)stream->abstract;
+
+#if HAVE_FLUSHIO
+ if (data->last_op == 'w')
+ fseek(data->file, 0, SEEK_CUR);
+ data->last_op = 'r';
+#endif
+
+ return fgets(buf, size, data->file);
}
static int php_stdiop_cast(php_stream * stream, int castas, void ** ret)
{
int fd;
+ php_stdio_stream_data * data = (php_stdio_stream_data*)stream->abstract;
switch (castas) {
case PHP_STREAM_AS_STDIO:
if (ret)
- *ret = stream->abstract;
+ *ret = data->file;
return SUCCESS;
case PHP_STREAM_AS_FD:
- fd = fileno((FILE*)stream->abstract);
+ fd = fileno(data->file);
if (fd < 0)
return FAILURE;
if (ret)
@@ -242,21 +525,153 @@ php_stream_ops php_stream_stdio_ops = {
"STDIO"
};
-PHPAPI php_stream * php_stream_fopen(const char * filename, const char * mode)
+PHPAPI php_stream * php_stream_fopen_with_path(char * filename, char * mode, char * path, char **opened_path TSRMLS_DC) /* {{{ */
+{
+ /* code ripped off from fopen_wrappers.c */
+ char *pathbuf, *ptr, *end;
+ char *exec_fname;
+ char trypath[MAXPATHLEN];
+ struct stat sb;
+ php_stream *stream;
+ int path_length;
+ int filename_length;
+ int exec_fname_length;
+
+ if (opened_path) {
+ *opened_path = NULL;
+ }
+
+ if(!filename) {
+ return NULL;
+ }
+
+ filename_length = strlen(filename);
+
+ /* Relative path open */
+ if (*filename == '.') {
+ if (PG(safe_mode) && (!php_checkuid(filename, mode, CHECKUID_CHECK_MODE_PARAM))) {
+ return NULL;
+ }
+ return php_stream_fopen(filename, mode, opened_path TSRMLS_CC);
+ }
+
+ /*
+ * files in safe_mode_include_dir (or subdir) are excluded from
+ * safe mode GID/UID checks
+ */
+
+ /* Absolute path open */
+ if (IS_ABSOLUTE_PATH(filename, filename_length)) {
+ if ((php_check_safe_mode_include_dir(filename TSRMLS_CC)) == 0)
+ /* filename is in safe_mode_include_dir (or subdir) */
+ return php_stream_fopen(filename, mode, opened_path TSRMLS_CC);
+
+ if (PG(safe_mode) && (!php_checkuid(filename, mode, CHECKUID_CHECK_MODE_PARAM)))
+ return NULL;
+
+ return php_stream_fopen(filename, mode, opened_path TSRMLS_CC);
+ }
+
+ if (!path || (path && !*path)) {
+ if (PG(safe_mode) && (!php_checkuid(filename, mode, CHECKUID_CHECK_MODE_PARAM))) {
+ return NULL;
+ }
+ return php_stream_fopen(filename, mode, opened_path TSRMLS_CC);
+ }
+
+ /* check in provided path */
+ /* append the calling scripts' current working directory
+ * as a fall back case
+ */
+ if (zend_is_executing(TSRMLS_C)) {
+ exec_fname = zend_get_executed_filename(TSRMLS_C);
+ exec_fname_length = strlen(exec_fname);
+ path_length = strlen(path);
+
+ while ((--exec_fname_length >= 0) && !IS_SLASH(exec_fname[exec_fname_length]));
+ if ((exec_fname && exec_fname[0] == '[')
+ || exec_fname_length<=0) {
+ /* [no active file] or no path */
+ pathbuf = estrdup(path);
+ } else {
+ pathbuf = (char *) emalloc(exec_fname_length + path_length +1 +1);
+ memcpy(pathbuf, path, path_length);
+ pathbuf[path_length] = DEFAULT_DIR_SEPARATOR;
+ memcpy(pathbuf+path_length+1, exec_fname, exec_fname_length);
+ pathbuf[path_length + exec_fname_length +1] = '\0';
+ }
+ } else {
+ pathbuf = estrdup(path);
+ }
+
+ ptr = pathbuf;
+
+ while (ptr && *ptr) {
+ end = strchr(ptr, DEFAULT_DIR_SEPARATOR);
+ if (end != NULL) {
+ *end = '\0';
+ end++;
+ }
+ snprintf(trypath, MAXPATHLEN, "%s/%s", ptr, filename);
+ if (PG(safe_mode)) {
+ if (VCWD_STAT(trypath, &sb) == 0) {
+ /* file exists ... check permission */
+ if ((php_check_safe_mode_include_dir(trypath TSRMLS_CC) == 0) ||
+ php_checkuid(trypath, mode, CHECKUID_CHECK_MODE_PARAM))
+ /* UID ok, or trypath is in safe_mode_include_dir */
+ stream = php_stream_fopen(trypath, mode, opened_path TSRMLS_CC);
+ else
+ stream = NULL;
+
+ efree(pathbuf);
+ return stream;
+ }
+ }
+ stream = php_stream_fopen(trypath, mode, opened_path TSRMLS_CC);
+ if (stream) {
+ efree(pathbuf);
+ return stream;
+ }
+ ptr = end;
+ } /* end provided path */
+
+ efree(pathbuf);
+ return NULL;
+
+}
+/* }}} */
+
+PHPAPI php_stream * php_stream_fopen(const char * filename, const char * mode, char **opened_path TSRMLS_DC)
{
- FILE * fp = fopen(filename, mode);
+ FILE * fp;
+ char * realpath;
+
+ realpath = expand_filepath(filename, NULL TSRMLS_C);
+
+ fp = fopen(realpath, mode);
if (fp) {
- php_stream * ret = php_stream_alloc(&php_stream_stdio_ops, fp, 0, mode);
+ php_stream * ret = php_stream_fopen_from_file(fp, mode);
+
+ if (ret) {
+ if (opened_path) {
+ *opened_path = realpath;
+ realpath = NULL;
+ }
+ if (realpath)
+ efree(realpath);
- if (ret)
return ret;
+ }
fclose(fp);
}
+ efree(realpath);
return NULL;
}
+/* }}} */
+/* {{{ STDIO with fopencookie */
#if HAVE_FOPENCOOKIE
static ssize_t stream_cookie_reader(void *cookie, char *buffer, size_t size)
{
@@ -286,9 +701,14 @@ static COOKIE_IO_FUNCTIONS_T stream_cookie_functions =
#else
/* TODO: use socketpair() to emulate fopencookie, as suggested by Hartmut ? */
#endif
+/* }}} */
-PHPAPI int php_stream_cast(php_stream * stream, int castas, void ** ret, int show_err)
+PHPAPI int php_stream_cast(php_stream * stream, int castas, void ** ret, int show_err) /* {{{ */
{
+
+ /* trying hard is not yet implemented */
+ castas &= ~PHP_STREAM_CAST_TRY_HARD;
+
if (castas == PHP_STREAM_AS_STDIO) {
if (stream->stdiocast) {
if (ret)
@@ -354,9 +774,110 @@ exit_success:
return SUCCESS;
+} /* }}} */
+
+int php_init_stream_wrappers(TSRMLS_D)
+{
+ if (PG(allow_url_fopen))
+ return zend_hash_init(&url_stream_wrappers_hash, 0, NULL, NULL, 1);
+ return SUCCESS;
+}
+
+int php_shutdown_stream_wrappers(TSRMLS_D)
+{
+ if (PG(allow_url_fopen))
+ zend_hash_destroy(&url_stream_wrappers_hash);
+ return SUCCESS;
}
-#endif
+PHPAPI int php_register_url_stream_wrapper(char * protocol, php_stream_wrapper * wrapper TSRMLS_DC)
+{
+ if (PG(allow_url_fopen))
+ return zend_hash_add(&url_stream_wrappers_hash, protocol, strlen(protocol), wrapper, sizeof(wrapper), NULL);
+ return FAILURE;
+}
+PHPAPI int php_unregister_url_stream_wrapper(char * protocol TSRMLS_DC)
+{
+ if (PG(allow_url_fopen))
+ return zend_hash_del(&url_stream_wrappers_hash, protocol, strlen(protocol));
+ return SUCCESS;
+}
+
+static php_stream * php_stream_open_url(char * path, char * mode, int options, char ** opened_path TSRMLS_DC)
+{
+ php_stream_wrapper * wrapper;
+ const char * p, *protocol = NULL;
+ int n = 0;
+
+ for (p = path; isalnum((int)*p); p++)
+ n++;
+
+ if ((*p == ':') && (n > 1))
+ protocol = path;
+
+ if (protocol) {
+ if (FAILURE == zend_hash_find(&url_stream_wrappers_hash, (char*)protocol, n, (void**)&wrapper)) {
+ wrapper = NULL;
+ protocol = NULL;
+ }
+ if (wrapper) {
+ php_stream * stream = wrapper->create(path, mode, options, opened_path TSRMLS_CC);
+ if (stream)
+ stream->wrapper = wrapper;
+ return stream;
+ }
+ }
+
+ if (!protocol || !strncasecmp(protocol, "file", n)) {
+ if (protocol && path[n+1] == '/' && path[n+2] == '/') {
+ zend_error(E_WARNING, "remote host file access not supported, %s", path);
+ return NULL;
+ }
+ if (protocol)
+ path += n + 1;
+
+ /* fall back on regular file access */
+ return php_stream_open_wrapper(path, mode, (options & ~REPORT_ERRORS) | IGNORE_URL,
+ opened_path TSRMLS_CC);
+ }
+ return NULL;
+}
+
+PHPAPI php_stream * php_stream_open_wrapper(char * path, char * mode, int options, char ** opened_path TSRMLS_DC)
+{
+ php_stream * stream = NULL;
+
+ if (opened_path)
+ *opened_path = NULL;
+
+ if (!path || !*path)
+ return NULL;
+
+ if (PG(allow_url_fopen) && !(options & IGNORE_URL)) {
+ stream = php_stream_open_url(path, mode, options, opened_path TSRMLS_CC);
+ goto out;
+ }
+
+ if ((options & USE_PATH) && PG(include_path) != NULL) {
+ stream = php_stream_fopen_with_path(path, mode, PG(include_path), opened_path TSRMLS_CC);
+ goto out;
+ }
+
+ if ((options & ENFORCE_SAFE_MODE) && PG(safe_mode) && (!php_checkuid(path, mode, CHECKUID_CHECK_MODE_PARAM)))
+ return NULL;
+
+ stream = php_stream_fopen(path, mode, opened_path TSRMLS_CC);
+out:
+ if (stream == NULL && (options & REPORT_ERRORS)) {
+ char * tmp = estrdup(path);
+ php_strip_url_passwd(tmp);
+ zend_error(E_WARNING, "%s(\"%s\") - %s", get_active_function_name(TSRMLS_CC), tmp, strerror(errno));
+ efree(tmp);
+ }
+ return stream;
+}
+
+
/*
* Local variables:
* tab-width: 4