Annotation of embedaddon/php/main/streams/plain_wrapper.c, revision 1.1.1.4

1.1       misho       1: /*
                      2:    +----------------------------------------------------------------------+
                      3:    | PHP Version 5                                                        |
                      4:    +----------------------------------------------------------------------+
1.1.1.4 ! misho       5:    | Copyright (c) 1997-2014 The PHP Group                                |
1.1       misho       6:    +----------------------------------------------------------------------+
                      7:    | This source file is subject to version 3.01 of the PHP license,      |
                      8:    | that is bundled with this package in the file LICENSE, and is        |
                      9:    | available through the world-wide-web at the following url:           |
                     10:    | http://www.php.net/license/3_01.txt                                  |
                     11:    | If you did not receive a copy of the PHP license and are unable to   |
                     12:    | obtain it through the world-wide-web, please send a note to          |
                     13:    | license@php.net so we can mail you a copy immediately.               |
                     14:    +----------------------------------------------------------------------+
                     15:    | Authors: Wez Furlong <wez@thebrainroom.com>                          |
                     16:    +----------------------------------------------------------------------+
                     17:  */
                     18: 
1.1.1.2   misho      19: /* $Id$ */
1.1       misho      20: 
                     21: #include "php.h"
                     22: #include "php_globals.h"
                     23: #include "php_network.h"
                     24: #include "php_open_temporary_file.h"
                     25: #include "ext/standard/file.h"
                     26: #include "ext/standard/flock_compat.h"
                     27: #include "ext/standard/php_filestat.h"
                     28: #include <stddef.h>
                     29: #include <fcntl.h>
                     30: #if HAVE_SYS_WAIT_H
                     31: #include <sys/wait.h>
                     32: #endif
                     33: #if HAVE_SYS_FILE_H
                     34: #include <sys/file.h>
                     35: #endif
                     36: #ifdef HAVE_SYS_MMAN_H
                     37: #include <sys/mman.h>
                     38: #endif
                     39: #include "SAPI.h"
                     40: 
                     41: #include "php_streams_int.h"
                     42: #ifdef PHP_WIN32
                     43: # include "win32/winutil.h"
                     44: #endif
                     45: 
                     46: #define php_stream_fopen_from_fd_int(fd, mode, persistent_id)  _php_stream_fopen_from_fd_int((fd), (mode), (persistent_id) STREAMS_CC TSRMLS_CC)
                     47: #define php_stream_fopen_from_fd_int_rel(fd, mode, persistent_id)       _php_stream_fopen_from_fd_int((fd), (mode), (persistent_id) STREAMS_REL_CC TSRMLS_CC)
                     48: #define php_stream_fopen_from_file_int(file, mode)     _php_stream_fopen_from_file_int((file), (mode) STREAMS_CC TSRMLS_CC)
                     49: #define php_stream_fopen_from_file_int_rel(file, mode)  _php_stream_fopen_from_file_int((file), (mode) STREAMS_REL_CC TSRMLS_CC)
                     50: 
1.1.1.2   misho      51: #if !defined(WINDOWS) && !defined(NETWARE)
                     52: extern int php_get_uid_by_name(const char *name, uid_t *uid TSRMLS_DC);
                     53: extern int php_get_gid_by_name(const char *name, gid_t *gid TSRMLS_DC);
                     54: #endif
                     55: 
1.1       misho      56: /* parse standard "fopen" modes into open() flags */
                     57: PHPAPI int php_stream_parse_fopen_modes(const char *mode, int *open_flags)
                     58: {
                     59:        int flags;
                     60: 
                     61:        switch (mode[0]) {
                     62:                case 'r':
                     63:                        flags = 0;
                     64:                        break;
                     65:                case 'w':
                     66:                        flags = O_TRUNC|O_CREAT;
                     67:                        break;
                     68:                case 'a':
                     69:                        flags = O_CREAT|O_APPEND;
                     70:                        break;
                     71:                case 'x':
                     72:                        flags = O_CREAT|O_EXCL;
                     73:                        break;
                     74:                case 'c':
                     75:                        flags = O_CREAT;
                     76:                        break;
                     77:                default:
                     78:                        /* unknown mode */
                     79:                        return FAILURE;
                     80:        }
1.1.1.4 ! misho      81: 
1.1       misho      82:        if (strchr(mode, '+')) {
                     83:                flags |= O_RDWR;
                     84:        } else if (flags) {
                     85:                flags |= O_WRONLY;
                     86:        } else {
                     87:                flags |= O_RDONLY;
                     88:        }
                     89: 
1.1.1.4 ! misho      90: #if defined(O_NONBLOCK)
        !            91:        if (strchr(mode, 'n')) {
        !            92:                flags |= O_NONBLOCK;
        !            93:        }
        !            94: #endif
        !            95: 
1.1       misho      96: #if defined(_O_TEXT) && defined(O_BINARY)
                     97:        if (strchr(mode, 't')) {
                     98:                flags |= _O_TEXT;
                     99:        } else {
                    100:                flags |= O_BINARY;
                    101:        }
                    102: #endif
                    103: 
                    104:        *open_flags = flags;
                    105:        return SUCCESS;
                    106: }
                    107: 
                    108: 
                    109: /* {{{ ------- STDIO stream implementation -------*/
                    110: 
                    111: typedef struct {
                    112:        FILE *file;
                    113:        int fd;                                 /* underlying file descriptor */
                    114:        unsigned is_process_pipe:1;     /* use pclose instead of fclose */
                    115:        unsigned is_pipe:1;                     /* don't try and seek */
                    116:        unsigned cached_fstat:1;        /* sb is valid */
                    117:        unsigned _reserved:29;
1.1.1.2   misho     118: 
1.1       misho     119:        int lock_flag;                  /* stores the lock state */
                    120:        char *temp_file_name;   /* if non-null, this is the path to a temporary file that
                    121:                                                         * is to be deleted when the stream is closed */
                    122: #if HAVE_FLUSHIO
                    123:        char last_op;
                    124: #endif
                    125: 
                    126: #if HAVE_MMAP
                    127:        char *last_mapped_addr;
                    128:        size_t last_mapped_len;
                    129: #endif
                    130: #ifdef PHP_WIN32
                    131:        char *last_mapped_addr;
                    132:        HANDLE file_mapping;
                    133: #endif
                    134: 
                    135:        struct stat sb;
                    136: } php_stdio_stream_data;
                    137: #define PHP_STDIOP_GET_FD(anfd, data)  anfd = (data)->file ? fileno((data)->file) : (data)->fd
                    138: 
                    139: static int do_fstat(php_stdio_stream_data *d, int force)
                    140: {
                    141:        if (!d->cached_fstat || force) {
                    142:                int fd;
                    143:                int r;
1.1.1.2   misho     144: 
1.1       misho     145:                PHP_STDIOP_GET_FD(fd, d);
                    146:                r = fstat(fd, &d->sb);
                    147:                d->cached_fstat = r == 0;
                    148: 
                    149:                return r;
                    150:        }
                    151:        return 0;
                    152: }
                    153: 
                    154: static php_stream *_php_stream_fopen_from_fd_int(int fd, const char *mode, const char *persistent_id STREAMS_DC TSRMLS_DC)
                    155: {
                    156:        php_stdio_stream_data *self;
1.1.1.2   misho     157: 
1.1       misho     158:        self = pemalloc_rel_orig(sizeof(*self), persistent_id);
                    159:        memset(self, 0, sizeof(*self));
                    160:        self->file = NULL;
                    161:        self->is_pipe = 0;
                    162:        self->lock_flag = LOCK_UN;
                    163:        self->is_process_pipe = 0;
                    164:        self->temp_file_name = NULL;
                    165:        self->fd = fd;
1.1.1.2   misho     166: 
1.1       misho     167:        return php_stream_alloc_rel(&php_stream_stdio_ops, self, persistent_id, mode);
                    168: }
                    169: 
                    170: static php_stream *_php_stream_fopen_from_file_int(FILE *file, const char *mode STREAMS_DC TSRMLS_DC)
                    171: {
                    172:        php_stdio_stream_data *self;
1.1.1.2   misho     173: 
1.1       misho     174:        self = emalloc_rel_orig(sizeof(*self));
                    175:        memset(self, 0, sizeof(*self));
                    176:        self->file = file;
                    177:        self->is_pipe = 0;
                    178:        self->lock_flag = LOCK_UN;
                    179:        self->is_process_pipe = 0;
                    180:        self->temp_file_name = NULL;
                    181:        self->fd = fileno(file);
                    182: 
                    183:        return php_stream_alloc_rel(&php_stream_stdio_ops, self, 0, mode);
                    184: }
                    185: 
                    186: PHPAPI php_stream *_php_stream_fopen_temporary_file(const char *dir, const char *pfx, char **opened_path STREAMS_DC TSRMLS_DC)
                    187: {
                    188:        int fd = php_open_temporary_fd(dir, pfx, opened_path TSRMLS_CC);
                    189: 
                    190:        if (fd != -1)   {
                    191:                php_stream *stream = php_stream_fopen_from_fd_int_rel(fd, "r+b", NULL);
                    192:                if (stream) {
                    193:                        return stream;
                    194:                }
                    195:                close(fd);
                    196: 
                    197:                php_error_docref(NULL TSRMLS_CC, E_WARNING, "unable to allocate stream");
                    198: 
                    199:                return NULL;
                    200:        }
                    201:        return NULL;
                    202: }
                    203: 
                    204: PHPAPI php_stream *_php_stream_fopen_tmpfile(int dummy STREAMS_DC TSRMLS_DC)
                    205: {
                    206:        char *opened_path = NULL;
                    207:        int fd = php_open_temporary_fd(NULL, "php", &opened_path TSRMLS_CC);
                    208: 
                    209:        if (fd != -1)   {
                    210:                php_stream *stream = php_stream_fopen_from_fd_int_rel(fd, "r+b", NULL);
                    211:                if (stream) {
                    212:                        php_stdio_stream_data *self = (php_stdio_stream_data*)stream->abstract;
                    213:                        stream->wrapper = &php_plain_files_wrapper;
                    214:                        stream->orig_path = estrdup(opened_path);
                    215: 
                    216:                        self->temp_file_name = opened_path;
                    217:                        self->lock_flag = LOCK_UN;
1.1.1.2   misho     218: 
1.1       misho     219:                        return stream;
                    220:                }
                    221:                close(fd);
                    222: 
                    223:                php_error_docref(NULL TSRMLS_CC, E_WARNING, "unable to allocate stream");
                    224: 
                    225:                return NULL;
                    226:        }
                    227:        return NULL;
                    228: }
                    229: 
                    230: PHPAPI php_stream *_php_stream_fopen_from_fd(int fd, const char *mode, const char *persistent_id STREAMS_DC TSRMLS_DC)
                    231: {
                    232:        php_stream *stream = php_stream_fopen_from_fd_int_rel(fd, mode, persistent_id);
                    233: 
                    234:        if (stream) {
                    235:                php_stdio_stream_data *self = (php_stdio_stream_data*)stream->abstract;
                    236: 
                    237: #ifdef S_ISFIFO
                    238:                /* detect if this is a pipe */
                    239:                if (self->fd >= 0) {
                    240:                        self->is_pipe = (do_fstat(self, 0) == 0 && S_ISFIFO(self->sb.st_mode)) ? 1 : 0;
                    241:                }
                    242: #elif defined(PHP_WIN32)
                    243:                {
                    244:                        zend_uintptr_t handle = _get_osfhandle(self->fd);
                    245: 
                    246:                        if (handle != (zend_uintptr_t)INVALID_HANDLE_VALUE) {
                    247:                                self->is_pipe = GetFileType((HANDLE)handle) == FILE_TYPE_PIPE;
                    248:                        }
                    249:                }
                    250: #endif
1.1.1.2   misho     251: 
1.1       misho     252:                if (self->is_pipe) {
                    253:                        stream->flags |= PHP_STREAM_FLAG_NO_SEEK;
                    254:                } else {
                    255:                        stream->position = lseek(self->fd, 0, SEEK_CUR);
                    256: #ifdef ESPIPE
                    257:                        if (stream->position == (off_t)-1 && errno == ESPIPE) {
                    258:                                stream->position = 0;
                    259:                                stream->flags |= PHP_STREAM_FLAG_NO_SEEK;
                    260:                                self->is_pipe = 1;
                    261:                        }
                    262: #endif
                    263:                }
                    264:        }
                    265: 
                    266:        return stream;
                    267: }
                    268: 
                    269: PHPAPI php_stream *_php_stream_fopen_from_file(FILE *file, const char *mode STREAMS_DC TSRMLS_DC)
                    270: {
                    271:        php_stream *stream = php_stream_fopen_from_file_int_rel(file, mode);
                    272: 
                    273:        if (stream) {
                    274:                php_stdio_stream_data *self = (php_stdio_stream_data*)stream->abstract;
                    275: 
                    276: #ifdef S_ISFIFO
                    277:                /* detect if this is a pipe */
                    278:                if (self->fd >= 0) {
                    279:                        self->is_pipe = (do_fstat(self, 0) == 0 && S_ISFIFO(self->sb.st_mode)) ? 1 : 0;
                    280:                }
                    281: #elif defined(PHP_WIN32)
                    282:                {
                    283:                        zend_uintptr_t handle = _get_osfhandle(self->fd);
                    284: 
                    285:                        if (handle != (zend_uintptr_t)INVALID_HANDLE_VALUE) {
                    286:                                self->is_pipe = GetFileType((HANDLE)handle) == FILE_TYPE_PIPE;
                    287:                        }
                    288:                }
                    289: #endif
1.1.1.2   misho     290: 
1.1       misho     291:                if (self->is_pipe) {
                    292:                        stream->flags |= PHP_STREAM_FLAG_NO_SEEK;
                    293:                } else {
                    294:                        stream->position = ftell(file);
                    295:                }
                    296:        }
                    297: 
                    298:        return stream;
                    299: }
                    300: 
                    301: PHPAPI php_stream *_php_stream_fopen_from_pipe(FILE *file, const char *mode STREAMS_DC TSRMLS_DC)
                    302: {
                    303:        php_stdio_stream_data *self;
                    304:        php_stream *stream;
                    305: 
                    306:        self = emalloc_rel_orig(sizeof(*self));
                    307:        memset(self, 0, sizeof(*self));
                    308:        self->file = file;
                    309:        self->is_pipe = 1;
                    310:        self->lock_flag = LOCK_UN;
                    311:        self->is_process_pipe = 1;
                    312:        self->fd = fileno(file);
                    313:        self->temp_file_name = NULL;
                    314: 
                    315:        stream = php_stream_alloc_rel(&php_stream_stdio_ops, self, 0, mode);
                    316:        stream->flags |= PHP_STREAM_FLAG_NO_SEEK;
                    317:        return stream;
                    318: }
                    319: 
                    320: static size_t php_stdiop_write(php_stream *stream, const char *buf, size_t count TSRMLS_DC)
                    321: {
                    322:        php_stdio_stream_data *data = (php_stdio_stream_data*)stream->abstract;
                    323: 
                    324:        assert(data != NULL);
                    325: 
                    326:        if (data->fd >= 0) {
                    327:                int bytes_written = write(data->fd, buf, count);
                    328:                if (bytes_written < 0) return 0;
                    329:                return (size_t) bytes_written;
                    330:        } else {
                    331: 
                    332: #if HAVE_FLUSHIO
                    333:                if (!data->is_pipe && data->last_op == 'r') {
                    334:                        fseek(data->file, 0, SEEK_CUR);
                    335:                }
                    336:                data->last_op = 'w';
                    337: #endif
                    338: 
                    339:                return fwrite(buf, 1, count, data->file);
                    340:        }
                    341: }
                    342: 
                    343: static size_t php_stdiop_read(php_stream *stream, char *buf, size_t count TSRMLS_DC)
                    344: {
                    345:        php_stdio_stream_data *data = (php_stdio_stream_data*)stream->abstract;
                    346:        size_t ret;
                    347: 
                    348:        assert(data != NULL);
                    349: 
                    350:        if (data->fd >= 0) {
                    351:                ret = read(data->fd, buf, count);
                    352: 
                    353:                if (ret == (size_t)-1 && errno == EINTR) {
                    354:                        /* Read was interrupted, retry once,
                    355:                           If read still fails, giveup with feof==0
                    356:                           so script can retry if desired */
                    357:                        ret = read(data->fd, buf, count);
                    358:                }
1.1.1.2   misho     359: 
1.1       misho     360:                stream->eof = (ret == 0 || (ret == (size_t)-1 && errno != EWOULDBLOCK && errno != EINTR && errno != EBADF));
1.1.1.2   misho     361: 
1.1       misho     362:        } else {
                    363: #if HAVE_FLUSHIO
                    364:                if (!data->is_pipe && data->last_op == 'w')
                    365:                        fseek(data->file, 0, SEEK_CUR);
                    366:                data->last_op = 'r';
                    367: #endif
                    368: 
                    369:                ret = fread(buf, 1, count, data->file);
                    370: 
                    371:                stream->eof = feof(data->file);
                    372:        }
                    373:        return ret;
                    374: }
                    375: 
                    376: static int php_stdiop_close(php_stream *stream, int close_handle TSRMLS_DC)
                    377: {
                    378:        int ret;
                    379:        php_stdio_stream_data *data = (php_stdio_stream_data*)stream->abstract;
                    380: 
                    381:        assert(data != NULL);
                    382: 
                    383: #if HAVE_MMAP
                    384:        if (data->last_mapped_addr) {
                    385:                munmap(data->last_mapped_addr, data->last_mapped_len);
                    386:                data->last_mapped_addr = NULL;
                    387:        }
                    388: #elif defined(PHP_WIN32)
                    389:        if (data->last_mapped_addr) {
                    390:                UnmapViewOfFile(data->last_mapped_addr);
                    391:                data->last_mapped_addr = NULL;
                    392:        }
                    393:        if (data->file_mapping) {
                    394:                CloseHandle(data->file_mapping);
                    395:                data->file_mapping = NULL;
                    396:        }
                    397: #endif
1.1.1.2   misho     398: 
1.1       misho     399:        if (close_handle) {
                    400:                if (data->file) {
                    401:                        if (data->is_process_pipe) {
                    402:                                errno = 0;
                    403:                                ret = pclose(data->file);
                    404: 
                    405: #if HAVE_SYS_WAIT_H
                    406:                                if (WIFEXITED(ret)) {
                    407:                                        ret = WEXITSTATUS(ret);
                    408:                                }
                    409: #endif
                    410:                        } else {
                    411:                                ret = fclose(data->file);
                    412:                                data->file = NULL;
                    413:                        }
                    414:                } else if (data->fd != -1) {
                    415:                        ret = close(data->fd);
                    416:                        data->fd = -1;
                    417:                } else {
                    418:                        return 0; /* everything should be closed already -> success */
                    419:                }
                    420:                if (data->temp_file_name) {
                    421:                        unlink(data->temp_file_name);
                    422:                        /* temporary streams are never persistent */
                    423:                        efree(data->temp_file_name);
                    424:                        data->temp_file_name = NULL;
                    425:                }
                    426:        } else {
                    427:                ret = 0;
                    428:                data->file = NULL;
                    429:                data->fd = -1;
                    430:        }
                    431: 
                    432:        pefree(data, stream->is_persistent);
                    433: 
                    434:        return ret;
                    435: }
                    436: 
                    437: static int php_stdiop_flush(php_stream *stream TSRMLS_DC)
                    438: {
                    439:        php_stdio_stream_data *data = (php_stdio_stream_data*)stream->abstract;
                    440: 
                    441:        assert(data != NULL);
                    442: 
                    443:        /*
                    444:         * stdio buffers data in user land. By calling fflush(3), this
                    445:         * data is send to the kernel using write(2). fsync'ing is
                    446:         * something completely different.
                    447:         */
                    448:        if (data->file) {
                    449:                return fflush(data->file);
                    450:        }
                    451:        return 0;
                    452: }
                    453: 
                    454: static int php_stdiop_seek(php_stream *stream, off_t offset, int whence, off_t *newoffset TSRMLS_DC)
                    455: {
                    456:        php_stdio_stream_data *data = (php_stdio_stream_data*)stream->abstract;
                    457:        int ret;
                    458: 
                    459:        assert(data != NULL);
                    460: 
                    461:        if (data->is_pipe) {
                    462:                php_error_docref(NULL TSRMLS_CC, E_WARNING, "cannot seek on a pipe");
                    463:                return -1;
                    464:        }
                    465: 
                    466:        if (data->fd >= 0) {
                    467:                off_t result;
1.1.1.2   misho     468: 
1.1       misho     469:                result = lseek(data->fd, offset, whence);
                    470:                if (result == (off_t)-1)
                    471:                        return -1;
                    472: 
                    473:                *newoffset = result;
                    474:                return 0;
1.1.1.2   misho     475: 
1.1       misho     476:        } else {
                    477:                ret = fseek(data->file, offset, whence);
                    478:                *newoffset = ftell(data->file);
                    479:                return ret;
                    480:        }
                    481: }
                    482: 
                    483: static int php_stdiop_cast(php_stream *stream, int castas, void **ret TSRMLS_DC)
                    484: {
                    485:        int fd;
                    486:        php_stdio_stream_data *data = (php_stdio_stream_data*) stream->abstract;
                    487: 
                    488:        assert(data != NULL);
1.1.1.2   misho     489: 
1.1       misho     490:        /* as soon as someone touches the stdio layer, buffering may ensue,
                    491:         * so we need to stop using the fd directly in that case */
                    492: 
                    493:        switch (castas) {
                    494:                case PHP_STREAM_AS_STDIO:
                    495:                        if (ret) {
                    496: 
                    497:                                if (data->file == NULL) {
                    498:                                        /* we were opened as a plain file descriptor, so we
                    499:                                         * need fdopen now */
                    500:                                        char fixed_mode[5];
                    501:                                        php_stream_mode_sanitize_fdopen_fopencookie(stream, fixed_mode);
                    502:                                        data->file = fdopen(data->fd, fixed_mode);
                    503:                                        if (data->file == NULL) {
                    504:                                                return FAILURE;
                    505:                                        }
                    506:                                }
1.1.1.2   misho     507: 
1.1       misho     508:                                *(FILE**)ret = data->file;
                    509:                                data->fd = -1;
                    510:                        }
                    511:                        return SUCCESS;
                    512: 
                    513:                case PHP_STREAM_AS_FD_FOR_SELECT:
                    514:                        PHP_STDIOP_GET_FD(fd, data);
                    515:                        if (fd < 0) {
                    516:                                return FAILURE;
                    517:                        }
                    518:                        if (ret) {
                    519:                                *(int*)ret = fd;
                    520:                        }
                    521:                        return SUCCESS;
                    522: 
                    523:                case PHP_STREAM_AS_FD:
                    524:                        PHP_STDIOP_GET_FD(fd, data);
                    525: 
                    526:                        if (fd < 0) {
                    527:                                return FAILURE;
                    528:                        }
                    529:                        if (data->file) {
                    530:                                fflush(data->file);
                    531:                        }
                    532:                        if (ret) {
                    533:                                *(int*)ret = fd;
                    534:                        }
                    535:                        return SUCCESS;
                    536:                default:
                    537:                        return FAILURE;
                    538:        }
                    539: }
                    540: 
                    541: static int php_stdiop_stat(php_stream *stream, php_stream_statbuf *ssb TSRMLS_DC)
                    542: {
                    543:        int ret;
                    544:        php_stdio_stream_data *data = (php_stdio_stream_data*) stream->abstract;
                    545: 
                    546:        assert(data != NULL);
                    547: 
                    548:        ret = do_fstat(data, 1);
                    549:        memcpy(&ssb->sb, &data->sb, sizeof(ssb->sb));
                    550:        return ret;
                    551: }
                    552: 
                    553: static int php_stdiop_set_option(php_stream *stream, int option, int value, void *ptrparam TSRMLS_DC)
                    554: {
                    555:        php_stdio_stream_data *data = (php_stdio_stream_data*) stream->abstract;
                    556:        size_t size;
                    557:        int fd;
                    558: #ifdef O_NONBLOCK
                    559:        /* FIXME: make this work for win32 */
                    560:        int flags;
                    561:        int oldval;
                    562: #endif
1.1.1.2   misho     563: 
1.1       misho     564:        PHP_STDIOP_GET_FD(fd, data);
1.1.1.2   misho     565: 
1.1       misho     566:        switch(option) {
                    567:                case PHP_STREAM_OPTION_BLOCKING:
                    568:                        if (fd == -1)
                    569:                                return -1;
                    570: #ifdef O_NONBLOCK
                    571:                        flags = fcntl(fd, F_GETFL, 0);
                    572:                        oldval = (flags & O_NONBLOCK) ? 0 : 1;
                    573:                        if (value)
                    574:                                flags &= ~O_NONBLOCK;
                    575:                        else
                    576:                                flags |= O_NONBLOCK;
1.1.1.2   misho     577: 
1.1       misho     578:                        if (-1 == fcntl(fd, F_SETFL, flags))
                    579:                                return -1;
                    580:                        return oldval;
                    581: #else
                    582:                        return -1; /* not yet implemented */
                    583: #endif
1.1.1.2   misho     584: 
1.1       misho     585:                case PHP_STREAM_OPTION_WRITE_BUFFER:
                    586: 
                    587:                        if (data->file == NULL) {
                    588:                                return -1;
                    589:                        }
1.1.1.2   misho     590: 
1.1       misho     591:                        if (ptrparam)
                    592:                                size = *(size_t *)ptrparam;
                    593:                        else
                    594:                                size = BUFSIZ;
                    595: 
                    596:                        switch(value) {
                    597:                                case PHP_STREAM_BUFFER_NONE:
                    598:                                        return setvbuf(data->file, NULL, _IONBF, 0);
1.1.1.2   misho     599: 
1.1       misho     600:                                case PHP_STREAM_BUFFER_LINE:
                    601:                                        return setvbuf(data->file, NULL, _IOLBF, size);
1.1.1.2   misho     602: 
1.1       misho     603:                                case PHP_STREAM_BUFFER_FULL:
                    604:                                        return setvbuf(data->file, NULL, _IOFBF, size);
                    605: 
                    606:                                default:
                    607:                                        return -1;
                    608:                        }
                    609:                        break;
1.1.1.2   misho     610: 
1.1       misho     611:                case PHP_STREAM_OPTION_LOCKING:
                    612:                        if (fd == -1) {
                    613:                                return -1;
                    614:                        }
                    615: 
                    616:                        if ((zend_uintptr_t) ptrparam == PHP_STREAM_LOCK_SUPPORTED) {
                    617:                                return 0;
                    618:                        }
                    619: 
                    620:                        if (!flock(fd, value)) {
                    621:                                data->lock_flag = value;
                    622:                                return 0;
                    623:                        } else {
                    624:                                return -1;
                    625:                        }
                    626:                        break;
                    627: 
                    628:                case PHP_STREAM_OPTION_MMAP_API:
                    629: #if HAVE_MMAP
                    630:                        {
                    631:                                php_stream_mmap_range *range = (php_stream_mmap_range*)ptrparam;
                    632:                                int prot, flags;
1.1.1.2   misho     633: 
1.1       misho     634:                                switch (value) {
                    635:                                        case PHP_STREAM_MMAP_SUPPORTED:
                    636:                                                return fd == -1 ? PHP_STREAM_OPTION_RETURN_ERR : PHP_STREAM_OPTION_RETURN_OK;
                    637: 
                    638:                                        case PHP_STREAM_MMAP_MAP_RANGE:
                    639:                                                do_fstat(data, 1);
                    640:                                                if (range->length == 0 && range->offset > 0 && range->offset < data->sb.st_size) {
                    641:                                                        range->length = data->sb.st_size - range->offset;
                    642:                                                }
                    643:                                                if (range->length == 0 || range->length > data->sb.st_size) {
                    644:                                                        range->length = data->sb.st_size;
                    645:                                                }
                    646:                                                if (range->offset >= data->sb.st_size) {
                    647:                                                        range->offset = data->sb.st_size;
                    648:                                                        range->length = 0;
                    649:                                                }
                    650:                                                switch (range->mode) {
                    651:                                                        case PHP_STREAM_MAP_MODE_READONLY:
                    652:                                                                prot = PROT_READ;
                    653:                                                                flags = MAP_PRIVATE;
                    654:                                                                break;
                    655:                                                        case PHP_STREAM_MAP_MODE_READWRITE:
                    656:                                                                prot = PROT_READ | PROT_WRITE;
                    657:                                                                flags = MAP_PRIVATE;
                    658:                                                                break;
                    659:                                                        case PHP_STREAM_MAP_MODE_SHARED_READONLY:
                    660:                                                                prot = PROT_READ;
                    661:                                                                flags = MAP_SHARED;
                    662:                                                                break;
                    663:                                                        case PHP_STREAM_MAP_MODE_SHARED_READWRITE:
                    664:                                                                prot = PROT_READ | PROT_WRITE;
                    665:                                                                flags = MAP_SHARED;
                    666:                                                                break;
                    667:                                                        default:
                    668:                                                                return PHP_STREAM_OPTION_RETURN_ERR;
                    669:                                                }
                    670:                                                range->mapped = (char*)mmap(NULL, range->length, prot, flags, fd, range->offset);
                    671:                                                if (range->mapped == (char*)MAP_FAILED) {
                    672:                                                        range->mapped = NULL;
                    673:                                                        return PHP_STREAM_OPTION_RETURN_ERR;
                    674:                                                }
                    675:                                                /* remember the mapping */
                    676:                                                data->last_mapped_addr = range->mapped;
                    677:                                                data->last_mapped_len = range->length;
                    678:                                                return PHP_STREAM_OPTION_RETURN_OK;
                    679: 
                    680:                                        case PHP_STREAM_MMAP_UNMAP:
                    681:                                                if (data->last_mapped_addr) {
                    682:                                                        munmap(data->last_mapped_addr, data->last_mapped_len);
                    683:                                                        data->last_mapped_addr = NULL;
                    684: 
                    685:                                                        return PHP_STREAM_OPTION_RETURN_OK;
                    686:                                                }
                    687:                                                return PHP_STREAM_OPTION_RETURN_ERR;
                    688:                                }
                    689:                        }
                    690: #elif defined(PHP_WIN32)
                    691:                        {
                    692:                                php_stream_mmap_range *range = (php_stream_mmap_range*)ptrparam;
                    693:                                HANDLE hfile = (HANDLE)_get_osfhandle(fd);
                    694:                                DWORD prot, acc, loffs = 0, delta = 0;
                    695: 
                    696:                                switch (value) {
                    697:                                        case PHP_STREAM_MMAP_SUPPORTED:
                    698:                                                return hfile == INVALID_HANDLE_VALUE ? PHP_STREAM_OPTION_RETURN_ERR : PHP_STREAM_OPTION_RETURN_OK;
                    699: 
                    700:                                        case PHP_STREAM_MMAP_MAP_RANGE:
                    701:                                                switch (range->mode) {
                    702:                                                        case PHP_STREAM_MAP_MODE_READONLY:
                    703:                                                                prot = PAGE_READONLY;
                    704:                                                                acc = FILE_MAP_READ;
                    705:                                                                break;
                    706:                                                        case PHP_STREAM_MAP_MODE_READWRITE:
                    707:                                                                prot = PAGE_READWRITE;
                    708:                                                                acc = FILE_MAP_READ | FILE_MAP_WRITE;
                    709:                                                                break;
                    710:                                                        case PHP_STREAM_MAP_MODE_SHARED_READONLY:
                    711:                                                                prot = PAGE_READONLY;
                    712:                                                                acc = FILE_MAP_READ;
                    713:                                                                /* TODO: we should assign a name for the mapping */
                    714:                                                                break;
                    715:                                                        case PHP_STREAM_MAP_MODE_SHARED_READWRITE:
                    716:                                                                prot = PAGE_READWRITE;
                    717:                                                                acc = FILE_MAP_READ | FILE_MAP_WRITE;
                    718:                                                                /* TODO: we should assign a name for the mapping */
                    719:                                                                break;
                    720:                                                        default:
                    721:                                                                return PHP_STREAM_OPTION_RETURN_ERR;
                    722:                                                }
                    723: 
                    724:                                                /* create a mapping capable of viewing the whole file (this costs no real resources) */
                    725:                                                data->file_mapping = CreateFileMapping(hfile, NULL, prot, 0, 0, NULL);
                    726: 
                    727:                                                if (data->file_mapping == NULL) {
                    728:                                                        return PHP_STREAM_OPTION_RETURN_ERR;
                    729:                                                }
                    730: 
                    731:                                                size = GetFileSize(hfile, NULL);
                    732:                                                if (range->length == 0 && range->offset > 0 && range->offset < size) {
                    733:                                                        range->length = size - range->offset;
                    734:                                                }
                    735:                                                if (range->length == 0 || range->length > size) {
                    736:                                                        range->length = size;
                    737:                                                }
                    738:                                                if (range->offset >= size) {
                    739:                                                        range->offset = size;
                    740:                                                        range->length = 0;
                    741:                                                }
                    742: 
                    743:                                                /* figure out how big a chunk to map to be able to view the part that we need */
                    744:                                                if (range->offset != 0) {
                    745:                                                        SYSTEM_INFO info;
                    746:                                                        DWORD gran;
                    747: 
                    748:                                                        GetSystemInfo(&info);
                    749:                                                        gran = info.dwAllocationGranularity;
                    750:                                                        loffs = (range->offset / gran) * gran;
                    751:                                                        delta = range->offset - loffs;
                    752:                                                }
                    753: 
                    754:                                                data->last_mapped_addr = MapViewOfFile(data->file_mapping, acc, 0, loffs, range->length + delta);
                    755: 
                    756:                                                if (data->last_mapped_addr) {
                    757:                                                        /* give them back the address of the start offset they requested */
                    758:                                                        range->mapped = data->last_mapped_addr + delta;
                    759:                                                        return PHP_STREAM_OPTION_RETURN_OK;
                    760:                                                }
                    761: 
                    762:                                                CloseHandle(data->file_mapping);
                    763:                                                data->file_mapping = NULL;
                    764: 
                    765:                                                return PHP_STREAM_OPTION_RETURN_ERR;
                    766: 
                    767:                                        case PHP_STREAM_MMAP_UNMAP:
                    768:                                                if (data->last_mapped_addr) {
                    769:                                                        UnmapViewOfFile(data->last_mapped_addr);
                    770:                                                        data->last_mapped_addr = NULL;
                    771:                                                        CloseHandle(data->file_mapping);
                    772:                                                        data->file_mapping = NULL;
                    773:                                                        return PHP_STREAM_OPTION_RETURN_OK;
                    774:                                                }
                    775:                                                return PHP_STREAM_OPTION_RETURN_ERR;
                    776: 
                    777:                                        default:
                    778:                                                return PHP_STREAM_OPTION_RETURN_ERR;
                    779:                                }
                    780:                        }
                    781: 
                    782: #endif
                    783:                        return PHP_STREAM_OPTION_RETURN_NOTIMPL;
                    784: 
                    785:                case PHP_STREAM_OPTION_TRUNCATE_API:
                    786:                        switch (value) {
                    787:                                case PHP_STREAM_TRUNCATE_SUPPORTED:
                    788:                                        return fd == -1 ? PHP_STREAM_OPTION_RETURN_ERR : PHP_STREAM_OPTION_RETURN_OK;
                    789: 
                    790:                                case PHP_STREAM_TRUNCATE_SET_SIZE: {
                    791:                                        ptrdiff_t new_size = *(ptrdiff_t*)ptrparam;
                    792:                                        if (new_size < 0) {
                    793:                                                return PHP_STREAM_OPTION_RETURN_ERR;
                    794:                                        }
                    795:                                        return ftruncate(fd, new_size) == 0 ? PHP_STREAM_OPTION_RETURN_OK : PHP_STREAM_OPTION_RETURN_ERR;
                    796:                                }
                    797:                        }
1.1.1.2   misho     798: 
1.1       misho     799:                default:
                    800:                        return PHP_STREAM_OPTION_RETURN_NOTIMPL;
                    801:        }
                    802: }
                    803: 
                    804: PHPAPI php_stream_ops  php_stream_stdio_ops = {
                    805:        php_stdiop_write, php_stdiop_read,
                    806:        php_stdiop_close, php_stdiop_flush,
                    807:        "STDIO",
                    808:        php_stdiop_seek,
                    809:        php_stdiop_cast,
                    810:        php_stdiop_stat,
                    811:        php_stdiop_set_option
                    812: };
                    813: /* }}} */
                    814: 
                    815: /* {{{ plain files opendir/readdir implementation */
                    816: static size_t php_plain_files_dirstream_read(php_stream *stream, char *buf, size_t count TSRMLS_DC)
                    817: {
                    818:        DIR *dir = (DIR*)stream->abstract;
                    819:        /* avoid libc5 readdir problems */
                    820:        char entry[sizeof(struct dirent)+MAXPATHLEN];
                    821:        struct dirent *result = (struct dirent *)&entry;
                    822:        php_stream_dirent *ent = (php_stream_dirent*)buf;
                    823: 
                    824:        /* avoid problems if someone mis-uses the stream */
                    825:        if (count != sizeof(php_stream_dirent))
                    826:                return 0;
                    827: 
                    828:        if (php_readdir_r(dir, (struct dirent *)entry, &result) == 0 && result) {
                    829:                PHP_STRLCPY(ent->d_name, result->d_name, sizeof(ent->d_name), strlen(result->d_name));
                    830:                return sizeof(php_stream_dirent);
                    831:        }
                    832:        return 0;
                    833: }
                    834: 
                    835: static int php_plain_files_dirstream_close(php_stream *stream, int close_handle TSRMLS_DC)
                    836: {
                    837:        return closedir((DIR *)stream->abstract);
                    838: }
                    839: 
                    840: static int php_plain_files_dirstream_rewind(php_stream *stream, off_t offset, int whence, off_t *newoffs TSRMLS_DC)
                    841: {
                    842:        rewinddir((DIR *)stream->abstract);
                    843:        return 0;
                    844: }
                    845: 
                    846: static php_stream_ops  php_plain_files_dirstream_ops = {
                    847:        NULL, php_plain_files_dirstream_read,
                    848:        php_plain_files_dirstream_close, NULL,
                    849:        "dir",
                    850:        php_plain_files_dirstream_rewind,
                    851:        NULL, /* cast */
                    852:        NULL, /* stat */
                    853:        NULL  /* set_option */
                    854: };
                    855: 
                    856: static php_stream *php_plain_files_dir_opener(php_stream_wrapper *wrapper, char *path, char *mode,
                    857:                int options, char **opened_path, php_stream_context *context STREAMS_DC TSRMLS_DC)
                    858: {
                    859:        DIR *dir = NULL;
                    860:        php_stream *stream = NULL;
                    861: 
                    862: #ifdef HAVE_GLOB
                    863:        if (options & STREAM_USE_GLOB_DIR_OPEN) {
                    864:                return php_glob_stream_wrapper.wops->dir_opener(&php_glob_stream_wrapper, path, mode, options, opened_path, context STREAMS_REL_CC TSRMLS_CC);
                    865:        }
                    866: #endif
                    867: 
                    868:        if (((options & STREAM_DISABLE_OPEN_BASEDIR) == 0) && php_check_open_basedir(path TSRMLS_CC)) {
                    869:                return NULL;
                    870:        }
1.1.1.2   misho     871: 
1.1       misho     872:        dir = VCWD_OPENDIR(path);
                    873: 
                    874: #ifdef PHP_WIN32
                    875:        if (!dir) {
                    876:                php_win32_docref2_from_error(GetLastError(), path, path TSRMLS_CC);
                    877:        }
                    878: 
                    879:        if (dir && dir->finished) {
                    880:                closedir(dir);
                    881:                dir = NULL;
                    882:        }
                    883: #endif
                    884:        if (dir) {
                    885:                stream = php_stream_alloc(&php_plain_files_dirstream_ops, dir, 0, mode);
                    886:                if (stream == NULL)
                    887:                        closedir(dir);
                    888:        }
1.1.1.2   misho     889: 
1.1       misho     890:        return stream;
                    891: }
                    892: /* }}} */
                    893: 
                    894: /* {{{ php_stream_fopen */
                    895: PHPAPI php_stream *_php_stream_fopen(const char *filename, const char *mode, char **opened_path, int options STREAMS_DC TSRMLS_DC)
                    896: {
                    897:        char *realpath = NULL;
                    898:        int open_flags;
                    899:        int fd;
                    900:        php_stream *ret;
                    901:        int persistent = options & STREAM_OPEN_PERSISTENT;
                    902:        char *persistent_id = NULL;
                    903: 
                    904:        if (FAILURE == php_stream_parse_fopen_modes(mode, &open_flags)) {
                    905:                if (options & REPORT_ERRORS) {
                    906:                        php_error_docref(NULL TSRMLS_CC, E_WARNING, "`%s' is not a valid mode for fopen", mode);
                    907:                }
                    908:                return NULL;
                    909:        }
                    910: 
                    911:        if (options & STREAM_ASSUME_REALPATH) {
                    912:                realpath = estrdup(filename);
                    913:        } else {
                    914:                if ((realpath = expand_filepath(filename, NULL TSRMLS_CC)) == NULL) {
                    915:                        return NULL;
                    916:                }
                    917:        }
                    918: 
                    919:        if (persistent) {
                    920:                spprintf(&persistent_id, 0, "streams_stdio_%d_%s", open_flags, realpath);
                    921:                switch (php_stream_from_persistent_id(persistent_id, &ret TSRMLS_CC)) {
                    922:                        case PHP_STREAM_PERSISTENT_SUCCESS:
                    923:                                if (opened_path) {
                    924:                                        *opened_path = realpath;
                    925:                                        realpath = NULL;
                    926:                                }
                    927:                                /* fall through */
                    928: 
                    929:                        case PHP_STREAM_PERSISTENT_FAILURE:
                    930:                                if (realpath) {
                    931:                                        efree(realpath);
                    932:                                }
                    933:                                efree(persistent_id);;
                    934:                                return ret;
                    935:                }
                    936:        }
1.1.1.2   misho     937: 
1.1       misho     938:        fd = open(realpath, open_flags, 0666);
                    939: 
                    940:        if (fd != -1)   {
                    941: 
                    942:                if (options & STREAM_OPEN_FOR_INCLUDE) {
                    943:                        ret = php_stream_fopen_from_fd_int_rel(fd, mode, persistent_id);
                    944:                } else {
                    945:                        ret = php_stream_fopen_from_fd_rel(fd, mode, persistent_id);
                    946:                }
                    947: 
                    948:                if (ret)        {
                    949:                        if (opened_path) {
                    950:                                *opened_path = realpath;
                    951:                                realpath = NULL;
                    952:                        }
                    953:                        if (realpath) {
                    954:                                efree(realpath);
                    955:                        }
                    956:                        if (persistent_id) {
                    957:                                efree(persistent_id);
                    958:                        }
                    959: 
                    960:                        /* WIN32 always set ISREG flag */
                    961: #ifndef PHP_WIN32
                    962:                        /* sanity checks for include/require.
                    963:                         * We check these after opening the stream, so that we save
                    964:                         * on fstat() syscalls */
                    965:                        if (options & STREAM_OPEN_FOR_INCLUDE) {
                    966:                                php_stdio_stream_data *self = (php_stdio_stream_data*)ret->abstract;
                    967:                                int r;
                    968: 
                    969:                                r = do_fstat(self, 0);
                    970:                                if ((r == 0 && !S_ISREG(self->sb.st_mode))) {
                    971:                                        if (opened_path) {
                    972:                                                efree(*opened_path);
                    973:                                                *opened_path = NULL;
                    974:                                        }
                    975:                                        php_stream_close(ret);
                    976:                                        return NULL;
                    977:                                }
                    978:                        }
                    979: #endif
                    980: 
                    981:                        return ret;
                    982:                }
                    983:                close(fd);
                    984:        }
                    985:        efree(realpath);
                    986:        if (persistent_id) {
                    987:                efree(persistent_id);
                    988:        }
                    989:        return NULL;
                    990: }
                    991: /* }}} */
                    992: 
                    993: 
                    994: static php_stream *php_plain_files_stream_opener(php_stream_wrapper *wrapper, char *path, char *mode,
                    995:                int options, char **opened_path, php_stream_context *context STREAMS_DC TSRMLS_DC)
                    996: {
                    997:        if (((options & STREAM_DISABLE_OPEN_BASEDIR) == 0) && php_check_open_basedir(path TSRMLS_CC)) {
                    998:                return NULL;
                    999:        }
                   1000: 
                   1001:        return php_stream_fopen_rel(path, mode, opened_path, options);
                   1002: }
                   1003: 
                   1004: static int php_plain_files_url_stater(php_stream_wrapper *wrapper, char *url, int flags, php_stream_statbuf *ssb, php_stream_context *context TSRMLS_DC)
                   1005: {
                   1006: 
                   1007:        if (strncmp(url, "file://", sizeof("file://") - 1) == 0) {
                   1008:                url += sizeof("file://") - 1;
                   1009:        }
                   1010: 
                   1011:        if (php_check_open_basedir_ex(url, (flags & PHP_STREAM_URL_STAT_QUIET) ? 0 : 1 TSRMLS_CC)) {
                   1012:                return -1;
                   1013:        }
                   1014: 
                   1015: #ifdef PHP_WIN32
                   1016:        if (EG(windows_version_info).dwMajorVersion >= 5) {
                   1017:                if (flags & PHP_STREAM_URL_STAT_LINK) {
                   1018:                        return VCWD_LSTAT(url, &ssb->sb);
                   1019:                }
                   1020:        }
                   1021: #else
                   1022: # ifdef HAVE_SYMLINK
                   1023:        if (flags & PHP_STREAM_URL_STAT_LINK) {
                   1024:                return VCWD_LSTAT(url, &ssb->sb);
                   1025:        } else
                   1026: # endif
                   1027: #endif
                   1028:                return VCWD_STAT(url, &ssb->sb);
                   1029: }
                   1030: 
                   1031: static int php_plain_files_unlink(php_stream_wrapper *wrapper, char *url, int options, php_stream_context *context TSRMLS_DC)
                   1032: {
                   1033:        char *p;
                   1034:        int ret;
                   1035: 
                   1036:        if ((p = strstr(url, "://")) != NULL) {
                   1037:                url = p + 3;
                   1038:        }
                   1039: 
1.1.1.2   misho    1040:        if (php_check_open_basedir(url TSRMLS_CC)) {
                   1041:                return 0;
1.1       misho    1042:        }
                   1043: 
                   1044:        ret = VCWD_UNLINK(url);
                   1045:        if (ret == -1) {
                   1046:                if (options & REPORT_ERRORS) {
                   1047:                        php_error_docref1(NULL TSRMLS_CC, url, E_WARNING, "%s", strerror(errno));
                   1048:                }
                   1049:                return 0;
                   1050:        }
                   1051: 
                   1052:        /* Clear stat cache (and realpath cache) */
                   1053:        php_clear_stat_cache(1, NULL, 0 TSRMLS_CC);
                   1054: 
                   1055:        return 1;
                   1056: }
                   1057: 
                   1058: static int php_plain_files_rename(php_stream_wrapper *wrapper, char *url_from, char *url_to, int options, php_stream_context *context TSRMLS_DC)
                   1059: {
                   1060:        char *p;
                   1061:        int ret;
                   1062: 
                   1063:        if (!url_from || !url_to) {
                   1064:                return 0;
                   1065:        }
                   1066: 
                   1067: #ifdef PHP_WIN32
                   1068:        if (!php_win32_check_trailing_space(url_from, strlen(url_from))) {
                   1069:                php_win32_docref2_from_error(ERROR_INVALID_NAME, url_from, url_to TSRMLS_CC);
                   1070:                return 0;
                   1071:        }
                   1072:        if (!php_win32_check_trailing_space(url_to, strlen(url_to))) {
                   1073:                php_win32_docref2_from_error(ERROR_INVALID_NAME, url_from, url_to TSRMLS_CC);
                   1074:                return 0;
                   1075:        }
                   1076: #endif
                   1077: 
                   1078:        if ((p = strstr(url_from, "://")) != NULL) {
                   1079:                url_from = p + 3;
                   1080:        }
                   1081: 
                   1082:        if ((p = strstr(url_to, "://")) != NULL) {
                   1083:                url_to = p + 3;
                   1084:        }
                   1085: 
                   1086:        if (php_check_open_basedir(url_from TSRMLS_CC) || php_check_open_basedir(url_to TSRMLS_CC)) {
                   1087:                return 0;
                   1088:        }
                   1089: 
                   1090:        ret = VCWD_RENAME(url_from, url_to);
                   1091: 
                   1092:        if (ret == -1) {
                   1093: #ifndef PHP_WIN32
                   1094: # ifdef EXDEV
                   1095:                if (errno == EXDEV) {
                   1096:                        struct stat sb;
                   1097:                        if (php_copy_file(url_from, url_to TSRMLS_CC) == SUCCESS) {
                   1098:                                if (VCWD_STAT(url_from, &sb) == 0) {
                   1099: #  if !defined(TSRM_WIN32) && !defined(NETWARE)
                   1100:                                        if (VCWD_CHMOD(url_to, sb.st_mode)) {
                   1101:                                                if (errno == EPERM) {
                   1102:                                                        php_error_docref2(NULL TSRMLS_CC, url_from, url_to, E_WARNING, "%s", strerror(errno));
                   1103:                                                        VCWD_UNLINK(url_from);
                   1104:                                                        return 1;
                   1105:                                                }
                   1106:                                                php_error_docref2(NULL TSRMLS_CC, url_from, url_to, E_WARNING, "%s", strerror(errno));
                   1107:                                                return 0;
                   1108:                                        }
                   1109:                                        if (VCWD_CHOWN(url_to, sb.st_uid, sb.st_gid)) {
                   1110:                                                if (errno == EPERM) {
                   1111:                                                        php_error_docref2(NULL TSRMLS_CC, url_from, url_to, E_WARNING, "%s", strerror(errno));
                   1112:                                                        VCWD_UNLINK(url_from);
                   1113:                                                        return 1;
                   1114:                                                }
                   1115:                                                php_error_docref2(NULL TSRMLS_CC, url_from, url_to, E_WARNING, "%s", strerror(errno));
                   1116:                                                return 0;
                   1117:                                        }
                   1118: #  endif
                   1119:                                        VCWD_UNLINK(url_from);
                   1120:                                        return 1;
                   1121:                                }
                   1122:                        }
                   1123:                        php_error_docref2(NULL TSRMLS_CC, url_from, url_to, E_WARNING, "%s", strerror(errno));
                   1124:                        return 0;
                   1125:                }
                   1126: # endif
                   1127: #endif
                   1128: 
                   1129: #ifdef PHP_WIN32
                   1130:                php_win32_docref2_from_error(GetLastError(), url_from, url_to TSRMLS_CC);
                   1131: #else
                   1132:                php_error_docref2(NULL TSRMLS_CC, url_from, url_to, E_WARNING, "%s", strerror(errno));
                   1133: #endif
                   1134:                return 0;
                   1135:        }
                   1136: 
                   1137:        /* Clear stat cache (and realpath cache) */
                   1138:        php_clear_stat_cache(1, NULL, 0 TSRMLS_CC);
                   1139: 
                   1140:        return 1;
                   1141: }
                   1142: 
                   1143: static int php_plain_files_mkdir(php_stream_wrapper *wrapper, char *dir, int mode, int options, php_stream_context *context TSRMLS_DC)
                   1144: {
                   1145:        int ret, recursive = options & PHP_STREAM_MKDIR_RECURSIVE;
                   1146:        char *p;
                   1147: 
                   1148:        if ((p = strstr(dir, "://")) != NULL) {
                   1149:                dir = p + 3;
                   1150:        }
                   1151: 
                   1152:        if (!recursive) {
                   1153:                ret = php_mkdir(dir, mode TSRMLS_CC);
                   1154:        } else {
                   1155:                /* we look for directory separator from the end of string, thus hopefuly reducing our work load */
1.1.1.2   misho    1156:                char *e;
1.1       misho    1157:                struct stat sb;
                   1158:                int dir_len = strlen(dir);
                   1159:                int offset = 0;
1.1.1.2   misho    1160:                char buf[MAXPATHLEN];
1.1       misho    1161: 
1.1.1.2   misho    1162:                if (!expand_filepath_with_mode(dir, buf, NULL, 0, CWD_EXPAND  TSRMLS_CC)) {
                   1163:                        php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid path");
                   1164:                        return 0;
1.1       misho    1165:                }
1.1.1.2   misho    1166: 
                   1167:                e = buf +  strlen(buf);
1.1       misho    1168: 
                   1169:                if ((p = memchr(buf, DEFAULT_SLASH, dir_len))) {
                   1170:                        offset = p - buf + 1;
                   1171:                }
                   1172: 
                   1173:                if (p && dir_len == 1) {
1.1.1.2   misho    1174:                        /* buf == "DEFAULT_SLASH" */
1.1       misho    1175:                }
                   1176:                else {
                   1177:                        /* find a top level directory we need to create */
                   1178:                        while ( (p = strrchr(buf + offset, DEFAULT_SLASH)) || (offset != 1 && (p = strrchr(buf, DEFAULT_SLASH))) ) {
                   1179:                                int n = 0;
                   1180: 
                   1181:                                *p = '\0';
                   1182:                                while (p > buf && *(p-1) == DEFAULT_SLASH) {
                   1183:                                        ++n;
                   1184:                                        --p;
                   1185:                                        *p = '\0';
                   1186:                                }
                   1187:                                if (VCWD_STAT(buf, &sb) == 0) {
                   1188:                                        while (1) {
                   1189:                                                *p = DEFAULT_SLASH;
                   1190:                                                if (!n) break;
                   1191:                                                --n;
                   1192:                                                ++p;
                   1193:                                        }
                   1194:                                        break;
                   1195:                                }
                   1196:                        }
                   1197:                }
                   1198: 
                   1199:                if (p == buf) {
                   1200:                        ret = php_mkdir(dir, mode TSRMLS_CC);
                   1201:                } else if (!(ret = php_mkdir(buf, mode TSRMLS_CC))) {
                   1202:                        if (!p) {
                   1203:                                p = buf;
                   1204:                        }
                   1205:                        /* create any needed directories if the creation of the 1st directory worked */
                   1206:                        while (++p != e) {
                   1207:                                if (*p == '\0') {
                   1208:                                        *p = DEFAULT_SLASH;
                   1209:                                        if ((*(p+1) != '\0') &&
                   1210:                                                (ret = VCWD_MKDIR(buf, (mode_t)mode)) < 0) {
                   1211:                                                if (options & REPORT_ERRORS) {
                   1212:                                                        php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", strerror(errno));
                   1213:                                                }
                   1214:                                                break;
                   1215:                                        }
                   1216:                                }
                   1217:                        }
                   1218:                }
                   1219:        }
                   1220:        if (ret < 0) {
                   1221:                /* Failure */
                   1222:                return 0;
                   1223:        } else {
                   1224:                /* Success */
                   1225:                return 1;
                   1226:        }
                   1227: }
                   1228: 
                   1229: static int php_plain_files_rmdir(php_stream_wrapper *wrapper, char *url, int options, php_stream_context *context TSRMLS_DC)
                   1230: {
                   1231: #if PHP_WIN32
                   1232:        int url_len = strlen(url);
                   1233: #endif
                   1234:        if (php_check_open_basedir(url TSRMLS_CC)) {
                   1235:                return 0;
                   1236:        }
                   1237: 
                   1238: #if PHP_WIN32
                   1239:        if (!php_win32_check_trailing_space(url, url_len)) {
                   1240:                php_error_docref1(NULL TSRMLS_CC, url, E_WARNING, "%s", strerror(ENOENT));
                   1241:                return 0;
                   1242:        }
                   1243: #endif
                   1244: 
                   1245:        if (VCWD_RMDIR(url) < 0) {
                   1246:                php_error_docref1(NULL TSRMLS_CC, url, E_WARNING, "%s", strerror(errno));
                   1247:                return 0;
                   1248:        }
                   1249: 
                   1250:        /* Clear stat cache (and realpath cache) */
                   1251:        php_clear_stat_cache(1, NULL, 0 TSRMLS_CC);
                   1252: 
                   1253:        return 1;
                   1254: }
                   1255: 
1.1.1.2   misho    1256: static int php_plain_files_metadata(php_stream_wrapper *wrapper, char *url, int option, void *value, php_stream_context *context TSRMLS_DC)
                   1257: {
                   1258:        struct utimbuf *newtime;
                   1259:        char *p;
                   1260: #if !defined(WINDOWS) && !defined(NETWARE)
                   1261:        uid_t uid;
                   1262:        gid_t gid;
                   1263: #endif
                   1264:        mode_t mode;
                   1265:        int ret = 0;
                   1266: #if PHP_WIN32
                   1267:        int url_len = strlen(url);
                   1268: #endif
                   1269: 
                   1270: #if PHP_WIN32
                   1271:        if (!php_win32_check_trailing_space(url, url_len)) {
                   1272:                php_error_docref1(NULL TSRMLS_CC, url, E_WARNING, "%s", strerror(ENOENT));
                   1273:                return 0;
                   1274:        }
                   1275: #endif
                   1276: 
                   1277:        if ((p = strstr(url, "://")) != NULL) {
                   1278:                url = p + 3;
                   1279:        }
                   1280: 
                   1281:        if (php_check_open_basedir(url TSRMLS_CC)) {
                   1282:                return 0;
                   1283:        }
                   1284: 
                   1285:        switch(option) {
                   1286:                case PHP_STREAM_META_TOUCH:
                   1287:                        newtime = (struct utimbuf *)value;
                   1288:                        if (VCWD_ACCESS(url, F_OK) != 0) {
                   1289:                                FILE *file = VCWD_FOPEN(url, "w");
                   1290:                                if (file == NULL) {
                   1291:                                        php_error_docref1(NULL TSRMLS_CC, url, E_WARNING, "Unable to create file %s because %s", url, strerror(errno));
                   1292:                                        return 0;
                   1293:                                }
                   1294:                                fclose(file);
                   1295:                        }
                   1296: 
                   1297:                        ret = VCWD_UTIME(url, newtime);
                   1298:                        break;
                   1299: #if !defined(WINDOWS) && !defined(NETWARE)
                   1300:                case PHP_STREAM_META_OWNER_NAME:
                   1301:                case PHP_STREAM_META_OWNER:
                   1302:                        if(option == PHP_STREAM_META_OWNER_NAME) {
                   1303:                                if(php_get_uid_by_name((char *)value, &uid TSRMLS_CC) != SUCCESS) {
                   1304:                                        php_error_docref1(NULL TSRMLS_CC, url, E_WARNING, "Unable to find uid for %s", (char *)value);
                   1305:                                        return 0;
                   1306:                                }
                   1307:                        } else {
                   1308:                                uid = (uid_t)*(long *)value;
                   1309:                        }
                   1310:                        ret = VCWD_CHOWN(url, uid, -1);
                   1311:                        break;
                   1312:                case PHP_STREAM_META_GROUP:
                   1313:                case PHP_STREAM_META_GROUP_NAME:
                   1314:                        if(option == PHP_STREAM_META_OWNER_NAME) {
                   1315:                                if(php_get_gid_by_name((char *)value, &gid TSRMLS_CC) != SUCCESS) {
                   1316:                                        php_error_docref1(NULL TSRMLS_CC, url, E_WARNING, "Unable to find gid for %s", (char *)value);
                   1317:                                        return 0;
                   1318:                                }
                   1319:                        } else {
                   1320:                                gid = (gid_t)*(long *)value;
                   1321:                        }
                   1322:                        ret = VCWD_CHOWN(url, -1, gid);
                   1323:                        break;
                   1324: #endif
                   1325:                case PHP_STREAM_META_ACCESS:
                   1326:                        mode = (mode_t)*(long *)value;
                   1327:                        ret = VCWD_CHMOD(url, mode);
                   1328:                        break;
                   1329:                default:
                   1330:                        php_error_docref1(NULL TSRMLS_CC, url, E_WARNING, "Unknown option %d for stream_metadata", option);
                   1331:                        return 0;
                   1332:        }
                   1333:        if (ret == -1) {
                   1334:                php_error_docref1(NULL TSRMLS_CC, url, E_WARNING, "Operation failed: %s", strerror(errno));
                   1335:                return 0;
                   1336:        }
                   1337:        php_clear_stat_cache(0, NULL, 0 TSRMLS_CC);
                   1338:        return 1;
                   1339: }
                   1340: 
                   1341: 
1.1       misho    1342: static php_stream_wrapper_ops php_plain_files_wrapper_ops = {
                   1343:        php_plain_files_stream_opener,
                   1344:        NULL,
                   1345:        NULL,
                   1346:        php_plain_files_url_stater,
                   1347:        php_plain_files_dir_opener,
                   1348:        "plainfile",
                   1349:        php_plain_files_unlink,
                   1350:        php_plain_files_rename,
                   1351:        php_plain_files_mkdir,
1.1.1.2   misho    1352:        php_plain_files_rmdir,
                   1353:        php_plain_files_metadata
1.1       misho    1354: };
                   1355: 
                   1356: php_stream_wrapper php_plain_files_wrapper = {
                   1357:        &php_plain_files_wrapper_ops,
                   1358:        NULL,
                   1359:        0
                   1360: };
                   1361: 
                   1362: /* {{{ php_stream_fopen_with_path */
                   1363: PHPAPI php_stream *_php_stream_fopen_with_path(char *filename, char *mode, char *path, char **opened_path, int options STREAMS_DC TSRMLS_DC)
                   1364: {
                   1365:        /* code ripped off from fopen_wrappers.c */
                   1366:        char *pathbuf, *ptr, *end;
1.1.1.2   misho    1367:        const char *exec_fname;
1.1       misho    1368:        char trypath[MAXPATHLEN];
                   1369:        php_stream *stream;
                   1370:        int path_length;
                   1371:        int filename_length;
                   1372:        int exec_fname_length;
                   1373: 
                   1374:        if (opened_path) {
                   1375:                *opened_path = NULL;
                   1376:        }
                   1377: 
                   1378:        if(!filename) {
                   1379:                return NULL;
                   1380:        }
                   1381: 
                   1382:        filename_length = strlen(filename);
                   1383: 
                   1384:        /* Relative path open */
                   1385:        if (*filename == '.' && (IS_SLASH(filename[1]) || filename[1] == '.')) {
                   1386:                /* further checks, we could have ....... filenames */
                   1387:                ptr = filename + 1;
                   1388:                if (*ptr == '.') {
                   1389:                        while (*(++ptr) == '.');
                   1390:                        if (!IS_SLASH(*ptr)) { /* not a relative path after all */
                   1391:                                goto not_relative_path;
                   1392:                        }
                   1393:                }
                   1394: 
                   1395: 
                   1396:                if (((options & STREAM_DISABLE_OPEN_BASEDIR) == 0) && php_check_open_basedir(filename TSRMLS_CC)) {
                   1397:                        return NULL;
                   1398:                }
                   1399: 
                   1400:                return php_stream_fopen_rel(filename, mode, opened_path, options);
                   1401:        }
                   1402: 
                   1403: not_relative_path:
                   1404: 
                   1405:        /* Absolute path open */
                   1406:        if (IS_ABSOLUTE_PATH(filename, filename_length)) {
                   1407: 
                   1408:                if (((options & STREAM_DISABLE_OPEN_BASEDIR) == 0) && php_check_open_basedir(filename TSRMLS_CC)) {
                   1409:                        return NULL;
                   1410:                }
                   1411: 
                   1412:                return php_stream_fopen_rel(filename, mode, opened_path, options);
                   1413:        }
1.1.1.2   misho    1414: 
1.1       misho    1415: #ifdef PHP_WIN32
                   1416:        if (IS_SLASH(filename[0])) {
                   1417:                size_t cwd_len;
                   1418:                char *cwd;
                   1419:                cwd = virtual_getcwd_ex(&cwd_len TSRMLS_CC);
                   1420:                /* getcwd() will return always return [DRIVE_LETTER]:/) on windows. */
                   1421:                *(cwd+3) = '\0';
1.1.1.2   misho    1422: 
1.1       misho    1423:                if (snprintf(trypath, MAXPATHLEN, "%s%s", cwd, filename) >= MAXPATHLEN) {
                   1424:                        php_error_docref(NULL TSRMLS_CC, E_NOTICE, "%s/%s path was truncated to %d", cwd, filename, MAXPATHLEN);
                   1425:                }
1.1.1.2   misho    1426: 
1.1       misho    1427:                free(cwd);
1.1.1.2   misho    1428: 
1.1       misho    1429:                if (((options & STREAM_DISABLE_OPEN_BASEDIR) == 0) && php_check_open_basedir(trypath TSRMLS_CC)) {
                   1430:                        return NULL;
                   1431:                }
1.1.1.2   misho    1432: 
1.1       misho    1433:                return php_stream_fopen_rel(trypath, mode, opened_path, options);
                   1434:        }
                   1435: #endif
                   1436: 
                   1437:        if (!path || (path && !*path)) {
                   1438:                return php_stream_fopen_rel(filename, mode, opened_path, options);
                   1439:        }
                   1440: 
                   1441:        /* check in provided path */
                   1442:        /* append the calling scripts' current working directory
                   1443:         * as a fall back case
                   1444:         */
                   1445:        if (zend_is_executing(TSRMLS_C)) {
                   1446:                exec_fname = zend_get_executed_filename(TSRMLS_C);
                   1447:                exec_fname_length = strlen(exec_fname);
                   1448:                path_length = strlen(path);
                   1449: 
                   1450:                while ((--exec_fname_length >= 0) && !IS_SLASH(exec_fname[exec_fname_length]));
                   1451:                if ((exec_fname && exec_fname[0] == '[')
                   1452:                                || exec_fname_length<=0) {
                   1453:                        /* [no active file] or no path */
                   1454:                        pathbuf = estrdup(path);
                   1455:                } else {
                   1456:                        pathbuf = (char *) emalloc(exec_fname_length + path_length +1 +1);
                   1457:                        memcpy(pathbuf, path, path_length);
                   1458:                        pathbuf[path_length] = DEFAULT_DIR_SEPARATOR;
                   1459:                        memcpy(pathbuf+path_length+1, exec_fname, exec_fname_length);
                   1460:                        pathbuf[path_length + exec_fname_length +1] = '\0';
                   1461:                }
                   1462:        } else {
                   1463:                pathbuf = estrdup(path);
                   1464:        }
                   1465: 
                   1466:        ptr = pathbuf;
                   1467: 
                   1468:        while (ptr && *ptr) {
                   1469:                end = strchr(ptr, DEFAULT_DIR_SEPARATOR);
                   1470:                if (end != NULL) {
                   1471:                        *end = '\0';
                   1472:                        end++;
                   1473:                }
                   1474:                if (*ptr == '\0') {
                   1475:                        goto stream_skip;
                   1476:                }
                   1477:                if (snprintf(trypath, MAXPATHLEN, "%s/%s", ptr, filename) >= MAXPATHLEN) {
                   1478:                        php_error_docref(NULL TSRMLS_CC, E_NOTICE, "%s/%s path was truncated to %d", ptr, filename, MAXPATHLEN);
                   1479:                }
                   1480: 
                   1481:                if (((options & STREAM_DISABLE_OPEN_BASEDIR) == 0) && php_check_open_basedir_ex(trypath, 0 TSRMLS_CC)) {
                   1482:                        goto stream_skip;
                   1483:                }
                   1484: 
                   1485:                stream = php_stream_fopen_rel(trypath, mode, opened_path, options);
                   1486:                if (stream) {
                   1487:                        efree(pathbuf);
                   1488:                        return stream;
                   1489:                }
                   1490: stream_skip:
                   1491:                ptr = end;
                   1492:        } /* end provided path */
                   1493: 
                   1494:        efree(pathbuf);
                   1495:        return NULL;
                   1496: 
                   1497: }
                   1498: /* }}} */
                   1499: 
                   1500: /*
                   1501:  * Local variables:
                   1502:  * tab-width: 4
                   1503:  * c-basic-offset: 4
                   1504:  * End:
                   1505:  * vim600: noet sw=4 ts=4 fdm=marker
                   1506:  * vim<600: noet sw=4 ts=4
                   1507:  */

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>