12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718 |
- /*
- +----------------------------------------------------------------------+
- | Copyright (c) The PHP Group |
- +----------------------------------------------------------------------+
- | This source file is subject to version 3.01 of the PHP license, |
- | that is bundled with this package in the file LICENSE, and is |
- | available through the world-wide-web at the following url: |
- | https://www.php.net/license/3_01.txt |
- | If you did not receive a copy of the PHP license and are unable to |
- | obtain it through the world-wide-web, please send a note to |
- | license@php.net so we can mail you a copy immediately. |
- +----------------------------------------------------------------------+
- | Authors: Wez Furlong <wez@thebrainroom.com> |
- +----------------------------------------------------------------------+
- */
- #include "php.h"
- #include "php_globals.h"
- #include "php_network.h"
- #include "php_open_temporary_file.h"
- #include "ext/standard/file.h"
- #include "ext/standard/flock_compat.h"
- #include "ext/standard/php_filestat.h"
- #include <stddef.h>
- #include <fcntl.h>
- #if HAVE_SYS_WAIT_H
- #include <sys/wait.h>
- #endif
- #if HAVE_SYS_FILE_H
- #include <sys/file.h>
- #endif
- #ifdef HAVE_SYS_MMAN_H
- #include <sys/mman.h>
- #endif
- #include "SAPI.h"
- #include "php_streams_int.h"
- #ifdef PHP_WIN32
- # include "win32/winutil.h"
- # include "win32/time.h"
- # include "win32/ioutil.h"
- # include "win32/readdir.h"
- #endif
- #define php_stream_fopen_from_fd_int(fd, mode, persistent_id) _php_stream_fopen_from_fd_int((fd), (mode), (persistent_id) STREAMS_CC)
- #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)
- #define php_stream_fopen_from_file_int(file, mode) _php_stream_fopen_from_file_int((file), (mode) STREAMS_CC)
- #define php_stream_fopen_from_file_int_rel(file, mode) _php_stream_fopen_from_file_int((file), (mode) STREAMS_REL_CC)
- #ifndef PHP_WIN32
- extern int php_get_uid_by_name(const char *name, uid_t *uid);
- extern int php_get_gid_by_name(const char *name, gid_t *gid);
- #endif
- #if defined(PHP_WIN32)
- # define PLAIN_WRAP_BUF_SIZE(st) (((st) > UINT_MAX) ? UINT_MAX : (unsigned int)(st))
- #define fsync _commit
- #define fdatasync fsync
- #else
- # define PLAIN_WRAP_BUF_SIZE(st) (st)
- # if !defined(HAVE_FDATASYNC)
- # define fdatasync fsync
- # elif defined(__APPLE__)
- // The symbol is present, however not in the headers
- extern int fdatasync(int);
- # endif
- #endif
- /* parse standard "fopen" modes into open() flags */
- PHPAPI int php_stream_parse_fopen_modes(const char *mode, int *open_flags)
- {
- int flags;
- switch (mode[0]) {
- case 'r':
- flags = 0;
- break;
- case 'w':
- flags = O_TRUNC|O_CREAT;
- break;
- case 'a':
- flags = O_CREAT|O_APPEND;
- break;
- case 'x':
- flags = O_CREAT|O_EXCL;
- break;
- case 'c':
- flags = O_CREAT;
- break;
- default:
- /* unknown mode */
- return FAILURE;
- }
- if (strchr(mode, '+')) {
- flags |= O_RDWR;
- } else if (flags) {
- flags |= O_WRONLY;
- } else {
- flags |= O_RDONLY;
- }
- #if defined(O_CLOEXEC)
- if (strchr(mode, 'e')) {
- flags |= O_CLOEXEC;
- }
- #endif
- #if defined(O_NONBLOCK)
- if (strchr(mode, 'n')) {
- flags |= O_NONBLOCK;
- }
- #endif
- #if defined(_O_TEXT) && defined(O_BINARY)
- if (strchr(mode, 't')) {
- flags |= _O_TEXT;
- } else {
- flags |= O_BINARY;
- }
- #endif
- *open_flags = flags;
- return SUCCESS;
- }
- /* {{{ ------- STDIO stream implementation -------*/
- typedef struct {
- FILE *file;
- int fd; /* underlying file descriptor */
- unsigned is_process_pipe:1; /* use pclose instead of fclose */
- unsigned is_pipe:1; /* stream is an actual pipe, currently Windows only*/
- unsigned cached_fstat:1; /* sb is valid */
- unsigned is_pipe_blocking:1; /* allow blocking read() on pipes, currently Windows only */
- unsigned no_forced_fstat:1; /* Use fstat cache even if forced */
- unsigned is_seekable:1; /* don't try and seek, if not set */
- unsigned _reserved:26;
- int lock_flag; /* stores the lock state */
- zend_string *temp_name; /* if non-null, this is the path to a temporary file that
- * is to be deleted when the stream is closed */
- #ifdef HAVE_FLUSHIO
- char last_op;
- #endif
- #ifdef HAVE_MMAP
- char *last_mapped_addr;
- size_t last_mapped_len;
- #endif
- #ifdef PHP_WIN32
- char *last_mapped_addr;
- HANDLE file_mapping;
- #endif
- zend_stat_t sb;
- } php_stdio_stream_data;
- #define PHP_STDIOP_GET_FD(anfd, data) anfd = (data)->file ? fileno((data)->file) : (data)->fd
- static int do_fstat(php_stdio_stream_data *d, int force)
- {
- if (!d->cached_fstat || (force && !d->no_forced_fstat)) {
- int fd;
- int r;
- PHP_STDIOP_GET_FD(fd, d);
- r = zend_fstat(fd, &d->sb);
- d->cached_fstat = r == 0;
- return r;
- }
- return 0;
- }
- static php_stream *_php_stream_fopen_from_fd_int(int fd, const char *mode, const char *persistent_id STREAMS_DC)
- {
- php_stdio_stream_data *self;
- self = pemalloc_rel_orig(sizeof(*self), persistent_id);
- memset(self, 0, sizeof(*self));
- self->file = NULL;
- self->is_seekable = 1;
- self->is_pipe = 0;
- self->lock_flag = LOCK_UN;
- self->is_process_pipe = 0;
- self->temp_name = NULL;
- self->fd = fd;
- #ifdef PHP_WIN32
- self->is_pipe_blocking = 0;
- #endif
- return php_stream_alloc_rel(&php_stream_stdio_ops, self, persistent_id, mode);
- }
- static php_stream *_php_stream_fopen_from_file_int(FILE *file, const char *mode STREAMS_DC)
- {
- php_stdio_stream_data *self;
- self = emalloc_rel_orig(sizeof(*self));
- memset(self, 0, sizeof(*self));
- self->file = file;
- self->is_seekable = 1;
- self->is_pipe = 0;
- self->lock_flag = LOCK_UN;
- self->is_process_pipe = 0;
- self->temp_name = NULL;
- self->fd = fileno(file);
- #ifdef PHP_WIN32
- self->is_pipe_blocking = 0;
- #endif
- return php_stream_alloc_rel(&php_stream_stdio_ops, self, 0, mode);
- }
- PHPAPI php_stream *_php_stream_fopen_temporary_file(const char *dir, const char *pfx, zend_string **opened_path_ptr STREAMS_DC)
- {
- zend_string *opened_path = NULL;
- int fd;
- fd = php_open_temporary_fd(dir, pfx, &opened_path);
- if (fd != -1) {
- php_stream *stream;
- if (opened_path_ptr) {
- *opened_path_ptr = opened_path;
- }
- stream = php_stream_fopen_from_fd_int_rel(fd, "r+b", NULL);
- if (stream) {
- php_stdio_stream_data *self = (php_stdio_stream_data*)stream->abstract;
- stream->wrapper = (php_stream_wrapper*)&php_plain_files_wrapper;
- stream->orig_path = estrndup(ZSTR_VAL(opened_path), ZSTR_LEN(opened_path));
- self->temp_name = opened_path;
- self->lock_flag = LOCK_UN;
- return stream;
- }
- close(fd);
- php_error_docref(NULL, E_WARNING, "Unable to allocate stream");
- return NULL;
- }
- return NULL;
- }
- PHPAPI php_stream *_php_stream_fopen_tmpfile(int dummy STREAMS_DC)
- {
- return php_stream_fopen_temporary_file(NULL, "php", NULL);
- }
- static void detect_is_seekable(php_stdio_stream_data *self) {
- #if defined(S_ISFIFO) && defined(S_ISCHR)
- if (self->fd >= 0 && do_fstat(self, 0) == 0) {
- self->is_seekable = !(S_ISFIFO(self->sb.st_mode) || S_ISCHR(self->sb.st_mode));
- self->is_pipe = S_ISFIFO(self->sb.st_mode);
- }
- #elif defined(PHP_WIN32)
- zend_uintptr_t handle = _get_osfhandle(self->fd);
- if (handle != (zend_uintptr_t)INVALID_HANDLE_VALUE) {
- DWORD file_type = GetFileType((HANDLE)handle);
- self->is_seekable = !(file_type == FILE_TYPE_PIPE || file_type == FILE_TYPE_CHAR);
- self->is_pipe = file_type == FILE_TYPE_PIPE;
- /* Additional check needed to distinguish between pipes and sockets. */
- if (self->is_pipe && !GetNamedPipeInfo((HANDLE) handle, NULL, NULL, NULL, NULL)) {
- self->is_pipe = 0;
- }
- }
- #endif
- }
- PHPAPI php_stream *_php_stream_fopen_from_fd(int fd, const char *mode, const char *persistent_id STREAMS_DC)
- {
- php_stream *stream = php_stream_fopen_from_fd_int_rel(fd, mode, persistent_id);
- if (stream) {
- php_stdio_stream_data *self = (php_stdio_stream_data*)stream->abstract;
- detect_is_seekable(self);
- if (!self->is_seekable) {
- stream->flags |= PHP_STREAM_FLAG_NO_SEEK;
- stream->position = -1;
- } else {
- stream->position = zend_lseek(self->fd, 0, SEEK_CUR);
- #ifdef ESPIPE
- /* FIXME: Is this code still needed? */
- if (stream->position == (zend_off_t)-1 && errno == ESPIPE) {
- stream->flags |= PHP_STREAM_FLAG_NO_SEEK;
- self->is_seekable = 0;
- }
- #endif
- }
- }
- return stream;
- }
- PHPAPI php_stream *_php_stream_fopen_from_file(FILE *file, const char *mode STREAMS_DC)
- {
- php_stream *stream = php_stream_fopen_from_file_int_rel(file, mode);
- if (stream) {
- php_stdio_stream_data *self = (php_stdio_stream_data*)stream->abstract;
- detect_is_seekable(self);
- if (!self->is_seekable) {
- stream->flags |= PHP_STREAM_FLAG_NO_SEEK;
- stream->position = -1;
- } else {
- stream->position = zend_ftell(file);
- }
- }
- return stream;
- }
- PHPAPI php_stream *_php_stream_fopen_from_pipe(FILE *file, const char *mode STREAMS_DC)
- {
- php_stdio_stream_data *self;
- php_stream *stream;
- self = emalloc_rel_orig(sizeof(*self));
- memset(self, 0, sizeof(*self));
- self->file = file;
- self->is_seekable = 0;
- self->is_pipe = 1;
- self->lock_flag = LOCK_UN;
- self->is_process_pipe = 1;
- self->fd = fileno(file);
- self->temp_name = NULL;
- #ifdef PHP_WIN32
- self->is_pipe_blocking = 0;
- #endif
- stream = php_stream_alloc_rel(&php_stream_stdio_ops, self, 0, mode);
- stream->flags |= PHP_STREAM_FLAG_NO_SEEK;
- return stream;
- }
- static ssize_t php_stdiop_write(php_stream *stream, const char *buf, size_t count)
- {
- php_stdio_stream_data *data = (php_stdio_stream_data*)stream->abstract;
- assert(data != NULL);
- if (data->fd >= 0) {
- #ifdef PHP_WIN32
- ssize_t bytes_written;
- if (ZEND_SIZE_T_UINT_OVFL(count)) {
- count = UINT_MAX;
- }
- bytes_written = _write(data->fd, buf, (unsigned int)count);
- #else
- ssize_t bytes_written = write(data->fd, buf, count);
- #endif
- if (bytes_written < 0) {
- if (PHP_IS_TRANSIENT_ERROR(errno)) {
- return 0;
- }
- if (errno == EINTR) {
- /* TODO: Should this be treated as a proper error or not? */
- return bytes_written;
- }
- if (!(stream->flags & PHP_STREAM_FLAG_SUPPRESS_ERRORS)) {
- php_error_docref(NULL, E_NOTICE, "Write of %zu bytes failed with errno=%d %s", count, errno, strerror(errno));
- }
- }
- return bytes_written;
- } else {
- #ifdef HAVE_FLUSHIO
- if (data->is_seekable && data->last_op == 'r') {
- zend_fseek(data->file, 0, SEEK_CUR);
- }
- data->last_op = 'w';
- #endif
- return (ssize_t) fwrite(buf, 1, count, data->file);
- }
- }
- static ssize_t php_stdiop_read(php_stream *stream, char *buf, size_t count)
- {
- php_stdio_stream_data *data = (php_stdio_stream_data*)stream->abstract;
- ssize_t ret;
- assert(data != NULL);
- if (data->fd >= 0) {
- #ifdef PHP_WIN32
- php_stdio_stream_data *self = (php_stdio_stream_data*)stream->abstract;
- if ((self->is_pipe || self->is_process_pipe) && !self->is_pipe_blocking) {
- HANDLE ph = (HANDLE)_get_osfhandle(data->fd);
- int retry = 0;
- DWORD avail_read = 0;
- do {
- /* Look ahead to get the available data amount to read. Do the same
- as read() does, however not blocking forever. In case it failed,
- no data will be read (better than block). */
- if (!PeekNamedPipe(ph, NULL, 0, NULL, &avail_read, NULL)) {
- break;
- }
- /* If there's nothing to read, wait in 10us periods. */
- if (0 == avail_read) {
- usleep(10);
- }
- } while (0 == avail_read && retry++ < 3200000);
- /* Reduce the required data amount to what is available, otherwise read()
- will block.*/
- if (avail_read < count) {
- count = avail_read;
- }
- }
- #endif
- ret = read(data->fd, buf, PLAIN_WRAP_BUF_SIZE(count));
- if (ret == (size_t)-1 && errno == EINTR) {
- /* Read was interrupted, retry once,
- If read still fails, give up with feof==0
- so script can retry if desired */
- ret = read(data->fd, buf, PLAIN_WRAP_BUF_SIZE(count));
- }
- if (ret < 0) {
- if (PHP_IS_TRANSIENT_ERROR(errno)) {
- /* Not an error. */
- ret = 0;
- } else if (errno == EINTR) {
- /* TODO: Should this be treated as a proper error or not? */
- } else {
- if (!(stream->flags & PHP_STREAM_FLAG_SUPPRESS_ERRORS)) {
- php_error_docref(NULL, E_NOTICE, "Read of %zu bytes failed with errno=%d %s", count, errno, strerror(errno));
- }
- /* TODO: Remove this special-case? */
- if (errno != EBADF) {
- stream->eof = 1;
- }
- }
- } else if (ret == 0) {
- stream->eof = 1;
- }
- } else {
- #ifdef HAVE_FLUSHIO
- if (data->is_seekable && data->last_op == 'w')
- zend_fseek(data->file, 0, SEEK_CUR);
- data->last_op = 'r';
- #endif
- ret = fread(buf, 1, count, data->file);
- stream->eof = feof(data->file);
- }
- return ret;
- }
- static int php_stdiop_close(php_stream *stream, int close_handle)
- {
- int ret;
- php_stdio_stream_data *data = (php_stdio_stream_data*)stream->abstract;
- assert(data != NULL);
- #ifdef HAVE_MMAP
- if (data->last_mapped_addr) {
- munmap(data->last_mapped_addr, data->last_mapped_len);
- data->last_mapped_addr = NULL;
- }
- #elif defined(PHP_WIN32)
- if (data->last_mapped_addr) {
- UnmapViewOfFile(data->last_mapped_addr);
- data->last_mapped_addr = NULL;
- }
- if (data->file_mapping) {
- CloseHandle(data->file_mapping);
- data->file_mapping = NULL;
- }
- #endif
- if (close_handle) {
- if (data->file) {
- if (data->is_process_pipe) {
- errno = 0;
- ret = pclose(data->file);
- #ifdef HAVE_SYS_WAIT_H
- if (WIFEXITED(ret)) {
- ret = WEXITSTATUS(ret);
- }
- #endif
- } else {
- ret = fclose(data->file);
- data->file = NULL;
- }
- } else if (data->fd != -1) {
- ret = close(data->fd);
- data->fd = -1;
- } else {
- return 0; /* everything should be closed already -> success */
- }
- if (data->temp_name) {
- #ifdef PHP_WIN32
- php_win32_ioutil_unlink(ZSTR_VAL(data->temp_name));
- #else
- unlink(ZSTR_VAL(data->temp_name));
- #endif
- /* temporary streams are never persistent */
- zend_string_release_ex(data->temp_name, 0);
- data->temp_name = NULL;
- }
- } else {
- ret = 0;
- data->file = NULL;
- data->fd = -1;
- }
- pefree(data, stream->is_persistent);
- return ret;
- }
- static int php_stdiop_flush(php_stream *stream)
- {
- php_stdio_stream_data *data = (php_stdio_stream_data*)stream->abstract;
- assert(data != NULL);
- /*
- * stdio buffers data in user land. By calling fflush(3), this
- * data is send to the kernel using write(2). fsync'ing is
- * something completely different.
- */
- if (data->file) {
- return fflush(data->file);
- }
- return 0;
- }
- static int php_stdiop_sync(php_stream *stream, bool dataonly)
- {
- php_stdio_stream_data *data = (php_stdio_stream_data*)stream->abstract;
- FILE *fp;
- int fd;
- if (php_stream_cast(stream, PHP_STREAM_AS_STDIO, (void**)&fp, REPORT_ERRORS) == FAILURE) {
- return -1;
- }
- if (php_stdiop_flush(stream) == 0) {
- PHP_STDIOP_GET_FD(fd, data);
- if (dataonly) {
- return fdatasync(fd);
- } else {
- return fsync(fd);
- }
- }
- return -1;
- }
- static int php_stdiop_seek(php_stream *stream, zend_off_t offset, int whence, zend_off_t *newoffset)
- {
- php_stdio_stream_data *data = (php_stdio_stream_data*)stream->abstract;
- int ret;
- assert(data != NULL);
- if (!data->is_seekable) {
- php_error_docref(NULL, E_WARNING, "Cannot seek on this stream");
- return -1;
- }
- if (data->fd >= 0) {
- zend_off_t result;
- result = zend_lseek(data->fd, offset, whence);
- if (result == (zend_off_t)-1)
- return -1;
- *newoffset = result;
- return 0;
- } else {
- ret = zend_fseek(data->file, offset, whence);
- *newoffset = zend_ftell(data->file);
- return ret;
- }
- }
- static int php_stdiop_cast(php_stream *stream, int castas, void **ret)
- {
- php_socket_t fd;
- php_stdio_stream_data *data = (php_stdio_stream_data*) stream->abstract;
- assert(data != NULL);
- /* as soon as someone touches the stdio layer, buffering may ensue,
- * so we need to stop using the fd directly in that case */
- switch (castas) {
- case PHP_STREAM_AS_STDIO:
- if (ret) {
- if (data->file == NULL) {
- /* we were opened as a plain file descriptor, so we
- * need fdopen now */
- char fixed_mode[5];
- php_stream_mode_sanitize_fdopen_fopencookie(stream, fixed_mode);
- data->file = fdopen(data->fd, fixed_mode);
- if (data->file == NULL) {
- return FAILURE;
- }
- }
- *(FILE**)ret = data->file;
- data->fd = SOCK_ERR;
- }
- return SUCCESS;
- case PHP_STREAM_AS_FD_FOR_SELECT:
- PHP_STDIOP_GET_FD(fd, data);
- if (SOCK_ERR == fd) {
- return FAILURE;
- }
- if (ret) {
- *(php_socket_t *)ret = fd;
- }
- return SUCCESS;
- case PHP_STREAM_AS_FD:
- PHP_STDIOP_GET_FD(fd, data);
- if (SOCK_ERR == fd) {
- return FAILURE;
- }
- if (data->file) {
- fflush(data->file);
- }
- if (ret) {
- *(php_socket_t *)ret = fd;
- }
- return SUCCESS;
- default:
- return FAILURE;
- }
- }
- static int php_stdiop_stat(php_stream *stream, php_stream_statbuf *ssb)
- {
- int ret;
- php_stdio_stream_data *data = (php_stdio_stream_data*) stream->abstract;
- assert(data != NULL);
- if((ret = do_fstat(data, 1)) == 0) {
- memcpy(&ssb->sb, &data->sb, sizeof(ssb->sb));
- }
- return ret;
- }
- static int php_stdiop_set_option(php_stream *stream, int option, int value, void *ptrparam)
- {
- php_stdio_stream_data *data = (php_stdio_stream_data*) stream->abstract;
- size_t size;
- int fd;
- #ifdef O_NONBLOCK
- /* FIXME: make this work for win32 */
- int flags;
- int oldval;
- #endif
- PHP_STDIOP_GET_FD(fd, data);
- switch(option) {
- case PHP_STREAM_OPTION_BLOCKING:
- if (fd == -1)
- return -1;
- #ifdef O_NONBLOCK
- flags = fcntl(fd, F_GETFL, 0);
- oldval = (flags & O_NONBLOCK) ? 0 : 1;
- if (value)
- flags &= ~O_NONBLOCK;
- else
- flags |= O_NONBLOCK;
- if (-1 == fcntl(fd, F_SETFL, flags))
- return -1;
- return oldval;
- #else
- return -1; /* not yet implemented */
- #endif
- case PHP_STREAM_OPTION_WRITE_BUFFER:
- if (data->file == NULL) {
- return -1;
- }
- if (ptrparam)
- size = *(size_t *)ptrparam;
- else
- size = BUFSIZ;
- switch(value) {
- case PHP_STREAM_BUFFER_NONE:
- return setvbuf(data->file, NULL, _IONBF, 0);
- case PHP_STREAM_BUFFER_LINE:
- return setvbuf(data->file, NULL, _IOLBF, size);
- case PHP_STREAM_BUFFER_FULL:
- return setvbuf(data->file, NULL, _IOFBF, size);
- default:
- return -1;
- }
- break;
- case PHP_STREAM_OPTION_LOCKING:
- if (fd == -1) {
- return -1;
- }
- if ((zend_uintptr_t) ptrparam == PHP_STREAM_LOCK_SUPPORTED) {
- return 0;
- }
- if (!flock(fd, value)) {
- data->lock_flag = value;
- return 0;
- } else {
- return -1;
- }
- break;
- case PHP_STREAM_OPTION_MMAP_API:
- #ifdef HAVE_MMAP
- {
- php_stream_mmap_range *range = (php_stream_mmap_range*)ptrparam;
- int prot, flags;
- switch (value) {
- case PHP_STREAM_MMAP_SUPPORTED:
- return fd == -1 ? PHP_STREAM_OPTION_RETURN_ERR : PHP_STREAM_OPTION_RETURN_OK;
- case PHP_STREAM_MMAP_MAP_RANGE:
- if (do_fstat(data, 1) != 0) {
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- if (range->offset > data->sb.st_size) {
- range->offset = data->sb.st_size;
- }
- if (range->length == 0 ||
- range->length > data->sb.st_size - range->offset) {
- range->length = data->sb.st_size - range->offset;
- }
- switch (range->mode) {
- case PHP_STREAM_MAP_MODE_READONLY:
- prot = PROT_READ;
- flags = MAP_PRIVATE;
- break;
- case PHP_STREAM_MAP_MODE_READWRITE:
- prot = PROT_READ | PROT_WRITE;
- flags = MAP_PRIVATE;
- break;
- case PHP_STREAM_MAP_MODE_SHARED_READONLY:
- prot = PROT_READ;
- flags = MAP_SHARED;
- break;
- case PHP_STREAM_MAP_MODE_SHARED_READWRITE:
- prot = PROT_READ | PROT_WRITE;
- flags = MAP_SHARED;
- break;
- default:
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- range->mapped = (char*)mmap(NULL, range->length, prot, flags, fd, range->offset);
- if (range->mapped == (char*)MAP_FAILED) {
- range->mapped = NULL;
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- /* remember the mapping */
- data->last_mapped_addr = range->mapped;
- data->last_mapped_len = range->length;
- return PHP_STREAM_OPTION_RETURN_OK;
- case PHP_STREAM_MMAP_UNMAP:
- if (data->last_mapped_addr) {
- munmap(data->last_mapped_addr, data->last_mapped_len);
- data->last_mapped_addr = NULL;
- return PHP_STREAM_OPTION_RETURN_OK;
- }
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- }
- #elif defined(PHP_WIN32)
- {
- php_stream_mmap_range *range = (php_stream_mmap_range*)ptrparam;
- HANDLE hfile = (HANDLE)_get_osfhandle(fd);
- DWORD prot, acc, loffs = 0, hoffs = 0, delta = 0;
- LARGE_INTEGER file_size;
- switch (value) {
- case PHP_STREAM_MMAP_SUPPORTED:
- return hfile == INVALID_HANDLE_VALUE ? PHP_STREAM_OPTION_RETURN_ERR : PHP_STREAM_OPTION_RETURN_OK;
- case PHP_STREAM_MMAP_MAP_RANGE:
- switch (range->mode) {
- case PHP_STREAM_MAP_MODE_READONLY:
- prot = PAGE_READONLY;
- acc = FILE_MAP_READ;
- break;
- case PHP_STREAM_MAP_MODE_READWRITE:
- prot = PAGE_READWRITE;
- acc = FILE_MAP_READ | FILE_MAP_WRITE;
- break;
- case PHP_STREAM_MAP_MODE_SHARED_READONLY:
- prot = PAGE_READONLY;
- acc = FILE_MAP_READ;
- /* TODO: we should assign a name for the mapping */
- break;
- case PHP_STREAM_MAP_MODE_SHARED_READWRITE:
- prot = PAGE_READWRITE;
- acc = FILE_MAP_READ | FILE_MAP_WRITE;
- /* TODO: we should assign a name for the mapping */
- break;
- default:
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- /* create a mapping capable of viewing the whole file (this costs no real resources) */
- data->file_mapping = CreateFileMapping(hfile, NULL, prot, 0, 0, NULL);
- if (data->file_mapping == NULL) {
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- if (!GetFileSizeEx(hfile, &file_size)) {
- CloseHandle(data->file_mapping);
- data->file_mapping = NULL;
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- # if defined(_WIN64)
- size = file_size.QuadPart;
- # else
- if (file_size.HighPart) {
- CloseHandle(data->file_mapping);
- data->file_mapping = NULL;
- return PHP_STREAM_OPTION_RETURN_ERR;
- } else {
- size = file_size.LowPart;
- }
- # endif
- if (range->offset > size) {
- range->offset = size;
- }
- if (range->length == 0 || range->length > size - range->offset) {
- range->length = size - range->offset;
- }
- /* figure out how big a chunk to map to be able to view the part that we need */
- if (range->offset != 0) {
- SYSTEM_INFO info;
- DWORD gran;
- GetSystemInfo(&info);
- gran = info.dwAllocationGranularity;
- ZEND_ASSERT(gran != 0 && (gran & (gran - 1)) == 0);
- size_t rounded_offset = (range->offset / gran) * gran;
- delta = range->offset - rounded_offset;
- loffs = (DWORD)rounded_offset;
- hoffs = (DWORD)(rounded_offset >> 32);
- }
- /* MapViewOfFile()ing zero bytes would map to the end of the file; match *nix behavior instead */
- if (range->length + delta == 0) {
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- data->last_mapped_addr = MapViewOfFile(data->file_mapping, acc, hoffs, loffs, range->length + delta);
- if (data->last_mapped_addr) {
- /* give them back the address of the start offset they requested */
- range->mapped = data->last_mapped_addr + delta;
- return PHP_STREAM_OPTION_RETURN_OK;
- }
- CloseHandle(data->file_mapping);
- data->file_mapping = NULL;
- return PHP_STREAM_OPTION_RETURN_ERR;
- case PHP_STREAM_MMAP_UNMAP:
- if (data->last_mapped_addr) {
- UnmapViewOfFile(data->last_mapped_addr);
- data->last_mapped_addr = NULL;
- CloseHandle(data->file_mapping);
- data->file_mapping = NULL;
- return PHP_STREAM_OPTION_RETURN_OK;
- }
- return PHP_STREAM_OPTION_RETURN_ERR;
- default:
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- }
- #endif
- return PHP_STREAM_OPTION_RETURN_NOTIMPL;
- case PHP_STREAM_OPTION_SYNC_API:
- switch (value) {
- case PHP_STREAM_SYNC_SUPPORTED:
- return fd == -1 ? PHP_STREAM_OPTION_RETURN_ERR : PHP_STREAM_OPTION_RETURN_OK;
- case PHP_STREAM_SYNC_FSYNC:
- return php_stdiop_sync(stream, 0) == 0 ? PHP_STREAM_OPTION_RETURN_OK : PHP_STREAM_OPTION_RETURN_ERR;
- case PHP_STREAM_SYNC_FDSYNC:
- return php_stdiop_sync(stream, 1) == 0 ? PHP_STREAM_OPTION_RETURN_OK : PHP_STREAM_OPTION_RETURN_ERR;
- }
- /* Invalid option passed */
- return PHP_STREAM_OPTION_RETURN_ERR;
- case PHP_STREAM_OPTION_TRUNCATE_API:
- switch (value) {
- case PHP_STREAM_TRUNCATE_SUPPORTED:
- return fd == -1 ? PHP_STREAM_OPTION_RETURN_ERR : PHP_STREAM_OPTION_RETURN_OK;
- case PHP_STREAM_TRUNCATE_SET_SIZE: {
- ptrdiff_t new_size = *(ptrdiff_t*)ptrparam;
- if (new_size < 0) {
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- #ifdef PHP_WIN32
- HANDLE h = (HANDLE) _get_osfhandle(fd);
- if (INVALID_HANDLE_VALUE == h) {
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- LARGE_INTEGER sz, old_sz;
- sz.QuadPart = 0;
- if (!SetFilePointerEx(h, sz, &old_sz, FILE_CURRENT)) {
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- #ifdef _WIN64
- sz.QuadPart = new_size;
- #else
- sz.HighPart = 0;
- sz.LowPart = new_size;
- #endif
- if (!SetFilePointerEx(h, sz, NULL, FILE_BEGIN)) {
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- if (0 == SetEndOfFile(h)) {
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- if (!SetFilePointerEx(h, old_sz, NULL, FILE_BEGIN)) {
- return PHP_STREAM_OPTION_RETURN_ERR;
- }
- return PHP_STREAM_OPTION_RETURN_OK;
- #else
- return ftruncate(fd, new_size) == 0 ? PHP_STREAM_OPTION_RETURN_OK : PHP_STREAM_OPTION_RETURN_ERR;
- #endif
- }
- }
- return PHP_STREAM_OPTION_RETURN_NOTIMPL;
- #ifdef PHP_WIN32
- case PHP_STREAM_OPTION_PIPE_BLOCKING:
- data->is_pipe_blocking = value;
- return PHP_STREAM_OPTION_RETURN_OK;
- #endif
- case PHP_STREAM_OPTION_META_DATA_API:
- if (fd == -1)
- return -1;
- #ifdef O_NONBLOCK
- flags = fcntl(fd, F_GETFL, 0);
- add_assoc_bool((zval*)ptrparam, "timed_out", 0);
- add_assoc_bool((zval*)ptrparam, "blocked", (flags & O_NONBLOCK)? 0 : 1);
- add_assoc_bool((zval*)ptrparam, "eof", stream->eof);
- return PHP_STREAM_OPTION_RETURN_OK;
- #endif
- return -1;
- default:
- return PHP_STREAM_OPTION_RETURN_NOTIMPL;
- }
- }
- /* This should be "const", but phpdbg overwrite it */
- PHPAPI php_stream_ops php_stream_stdio_ops = {
- php_stdiop_write, php_stdiop_read,
- php_stdiop_close, php_stdiop_flush,
- "STDIO",
- php_stdiop_seek,
- php_stdiop_cast,
- php_stdiop_stat,
- php_stdiop_set_option
- };
- /* }}} */
- /* {{{ plain files opendir/readdir implementation */
- static ssize_t php_plain_files_dirstream_read(php_stream *stream, char *buf, size_t count)
- {
- DIR *dir = (DIR*)stream->abstract;
- struct dirent *result;
- php_stream_dirent *ent = (php_stream_dirent*)buf;
- /* avoid problems if someone mis-uses the stream */
- if (count != sizeof(php_stream_dirent))
- return -1;
- result = readdir(dir);
- if (result) {
- PHP_STRLCPY(ent->d_name, result->d_name, sizeof(ent->d_name), strlen(result->d_name));
- return sizeof(php_stream_dirent);
- }
- return 0;
- }
- static int php_plain_files_dirstream_close(php_stream *stream, int close_handle)
- {
- return closedir((DIR *)stream->abstract);
- }
- static int php_plain_files_dirstream_rewind(php_stream *stream, zend_off_t offset, int whence, zend_off_t *newoffs)
- {
- rewinddir((DIR *)stream->abstract);
- return 0;
- }
- static const php_stream_ops php_plain_files_dirstream_ops = {
- NULL, php_plain_files_dirstream_read,
- php_plain_files_dirstream_close, NULL,
- "dir",
- php_plain_files_dirstream_rewind,
- NULL, /* cast */
- NULL, /* stat */
- NULL /* set_option */
- };
- static php_stream *php_plain_files_dir_opener(php_stream_wrapper *wrapper, const char *path, const char *mode,
- int options, zend_string **opened_path, php_stream_context *context STREAMS_DC)
- {
- DIR *dir = NULL;
- php_stream *stream = NULL;
- #ifdef HAVE_GLOB
- if (options & STREAM_USE_GLOB_DIR_OPEN) {
- return php_glob_stream_wrapper.wops->dir_opener((php_stream_wrapper*)&php_glob_stream_wrapper, path, mode, options, opened_path, context STREAMS_REL_CC);
- }
- #endif
- if (((options & STREAM_DISABLE_OPEN_BASEDIR) == 0) && php_check_open_basedir(path)) {
- return NULL;
- }
- dir = VCWD_OPENDIR(path);
- #ifdef PHP_WIN32
- if (!dir) {
- php_win32_docref1_from_error(GetLastError(), path);
- }
- if (dir && dir->finished) {
- closedir(dir);
- dir = NULL;
- }
- #endif
- if (dir) {
- stream = php_stream_alloc(&php_plain_files_dirstream_ops, dir, 0, mode);
- if (stream == NULL)
- closedir(dir);
- }
- return stream;
- }
- /* }}} */
- /* {{{ php_stream_fopen */
- PHPAPI php_stream *_php_stream_fopen(const char *filename, const char *mode, zend_string **opened_path, int options STREAMS_DC)
- {
- char realpath[MAXPATHLEN];
- int open_flags;
- int fd;
- php_stream *ret;
- int persistent = options & STREAM_OPEN_PERSISTENT;
- char *persistent_id = NULL;
- if (FAILURE == php_stream_parse_fopen_modes(mode, &open_flags)) {
- php_stream_wrapper_log_error(&php_plain_files_wrapper, options, "`%s' is not a valid mode for fopen", mode);
- return NULL;
- }
- if (options & STREAM_ASSUME_REALPATH) {
- strlcpy(realpath, filename, sizeof(realpath));
- } else {
- if (expand_filepath(filename, realpath) == NULL) {
- return NULL;
- }
- }
- if (persistent) {
- spprintf(&persistent_id, 0, "streams_stdio_%d_%s", open_flags, realpath);
- switch (php_stream_from_persistent_id(persistent_id, &ret)) {
- case PHP_STREAM_PERSISTENT_SUCCESS:
- if (opened_path) {
- //TODO: avoid reallocation???
- *opened_path = zend_string_init(realpath, strlen(realpath), 0);
- }
- ZEND_FALLTHROUGH;
- case PHP_STREAM_PERSISTENT_FAILURE:
- efree(persistent_id);
- return ret;
- }
- }
- #ifdef PHP_WIN32
- fd = php_win32_ioutil_open(realpath, open_flags, 0666);
- #else
- fd = open(realpath, open_flags, 0666);
- #endif
- if (fd != -1) {
- if (options & STREAM_OPEN_FOR_INCLUDE) {
- ret = php_stream_fopen_from_fd_int_rel(fd, mode, persistent_id);
- } else {
- ret = php_stream_fopen_from_fd_rel(fd, mode, persistent_id);
- }
- if (ret) {
- if (opened_path) {
- *opened_path = zend_string_init(realpath, strlen(realpath), 0);
- }
- if (persistent_id) {
- efree(persistent_id);
- }
- /* WIN32 always set ISREG flag */
- #ifndef PHP_WIN32
- /* sanity checks for include/require.
- * We check these after opening the stream, so that we save
- * on fstat() syscalls */
- if (options & STREAM_OPEN_FOR_INCLUDE) {
- php_stdio_stream_data *self = (php_stdio_stream_data*)ret->abstract;
- int r;
- r = do_fstat(self, 0);
- if ((r == 0 && !S_ISREG(self->sb.st_mode))) {
- if (opened_path) {
- zend_string_release_ex(*opened_path, 0);
- *opened_path = NULL;
- }
- php_stream_close(ret);
- return NULL;
- }
- /* Make sure the fstat result is reused when we later try to get the
- * file size. */
- self->no_forced_fstat = 1;
- }
- if (options & STREAM_USE_BLOCKING_PIPE) {
- php_stdio_stream_data *self = (php_stdio_stream_data*)ret->abstract;
- self->is_pipe_blocking = 1;
- }
- #endif
- return ret;
- }
- close(fd);
- }
- if (persistent_id) {
- efree(persistent_id);
- }
- return NULL;
- }
- /* }}} */
- static php_stream *php_plain_files_stream_opener(php_stream_wrapper *wrapper, const char *path, const char *mode,
- int options, zend_string **opened_path, php_stream_context *context STREAMS_DC)
- {
- if (((options & STREAM_DISABLE_OPEN_BASEDIR) == 0) && php_check_open_basedir(path)) {
- return NULL;
- }
- return php_stream_fopen_rel(path, mode, opened_path, options);
- }
- static int php_plain_files_url_stater(php_stream_wrapper *wrapper, const char *url, int flags, php_stream_statbuf *ssb, php_stream_context *context)
- {
- if (!(flags & PHP_STREAM_URL_STAT_IGNORE_OPEN_BASEDIR)) {
- if (strncasecmp(url, "file://", sizeof("file://") - 1) == 0) {
- url += sizeof("file://") - 1;
- }
- if (php_check_open_basedir_ex(url, (flags & PHP_STREAM_URL_STAT_QUIET) ? 0 : 1)) {
- return -1;
- }
- }
- #ifdef PHP_WIN32
- if (flags & PHP_STREAM_URL_STAT_LINK) {
- return VCWD_LSTAT(url, &ssb->sb);
- }
- #else
- # ifdef HAVE_SYMLINK
- if (flags & PHP_STREAM_URL_STAT_LINK) {
- return VCWD_LSTAT(url, &ssb->sb);
- } else
- # endif
- #endif
- return VCWD_STAT(url, &ssb->sb);
- }
- static int php_plain_files_unlink(php_stream_wrapper *wrapper, const char *url, int options, php_stream_context *context)
- {
- int ret;
- if (strncasecmp(url, "file://", sizeof("file://") - 1) == 0) {
- url += sizeof("file://") - 1;
- }
- if (php_check_open_basedir(url)) {
- return 0;
- }
- ret = VCWD_UNLINK(url);
- if (ret == -1) {
- if (options & REPORT_ERRORS) {
- php_error_docref1(NULL, url, E_WARNING, "%s", strerror(errno));
- }
- return 0;
- }
- /* Clear stat cache (and realpath cache) */
- php_clear_stat_cache(1, NULL, 0);
- return 1;
- }
- static int php_plain_files_rename(php_stream_wrapper *wrapper, const char *url_from, const char *url_to, int options, php_stream_context *context)
- {
- int ret;
- if (!url_from || !url_to) {
- return 0;
- }
- #ifdef PHP_WIN32
- if (!php_win32_check_trailing_space(url_from, strlen(url_from))) {
- php_win32_docref2_from_error(ERROR_INVALID_NAME, url_from, url_to);
- return 0;
- }
- if (!php_win32_check_trailing_space(url_to, strlen(url_to))) {
- php_win32_docref2_from_error(ERROR_INVALID_NAME, url_from, url_to);
- return 0;
- }
- #endif
- if (strncasecmp(url_from, "file://", sizeof("file://") - 1) == 0) {
- url_from += sizeof("file://") - 1;
- }
- if (strncasecmp(url_to, "file://", sizeof("file://") - 1) == 0) {
- url_to += sizeof("file://") - 1;
- }
- if (php_check_open_basedir(url_from) || php_check_open_basedir(url_to)) {
- return 0;
- }
- ret = VCWD_RENAME(url_from, url_to);
- if (ret == -1) {
- #ifndef PHP_WIN32
- # ifdef EXDEV
- if (errno == EXDEV) {
- zend_stat_t sb;
- # if !defined(ZTS) && !defined(TSRM_WIN32)
- /* not sure what to do in ZTS case, umask is not thread-safe */
- int oldmask = umask(077);
- # endif
- int success = 0;
- if (php_copy_file(url_from, url_to) == SUCCESS) {
- if (VCWD_STAT(url_from, &sb) == 0) {
- success = 1;
- # ifndef TSRM_WIN32
- /*
- * Try to set user and permission info on the target.
- * If we're not root, then some of these may fail.
- * We try chown first, to set proper group info, relying
- * on the system environment to have proper umask to not allow
- * access to the file in the meantime.
- */
- if (VCWD_CHOWN(url_to, sb.st_uid, sb.st_gid)) {
- php_error_docref2(NULL, url_from, url_to, E_WARNING, "%s", strerror(errno));
- if (errno != EPERM) {
- success = 0;
- }
- }
- if (success) {
- if (VCWD_CHMOD(url_to, sb.st_mode)) {
- php_error_docref2(NULL, url_from, url_to, E_WARNING, "%s", strerror(errno));
- if (errno != EPERM) {
- success = 0;
- }
- }
- }
- # endif
- if (success) {
- VCWD_UNLINK(url_from);
- }
- } else {
- php_error_docref2(NULL, url_from, url_to, E_WARNING, "%s", strerror(errno));
- }
- } else {
- php_error_docref2(NULL, url_from, url_to, E_WARNING, "%s", strerror(errno));
- }
- # if !defined(ZTS) && !defined(TSRM_WIN32)
- umask(oldmask);
- # endif
- return success;
- }
- # endif
- #endif
- #ifdef PHP_WIN32
- php_win32_docref2_from_error(GetLastError(), url_from, url_to);
- #else
- php_error_docref2(NULL, url_from, url_to, E_WARNING, "%s", strerror(errno));
- #endif
- return 0;
- }
- /* Clear stat cache (and realpath cache) */
- php_clear_stat_cache(1, NULL, 0);
- return 1;
- }
- static int php_plain_files_mkdir(php_stream_wrapper *wrapper, const char *dir, int mode, int options, php_stream_context *context)
- {
- if (strncasecmp(dir, "file://", sizeof("file://") - 1) == 0) {
- dir += sizeof("file://") - 1;
- }
- if (!(options & PHP_STREAM_MKDIR_RECURSIVE)) {
- return php_mkdir(dir, mode) == 0;
- }
- char buf[MAXPATHLEN];
- if (!expand_filepath_with_mode(dir, buf, NULL, 0, CWD_EXPAND)) {
- php_error_docref(NULL, E_WARNING, "Invalid path");
- return 0;
- }
- if (php_check_open_basedir(buf)) {
- return 0;
- }
- /* we look for directory separator from the end of string, thus hopefully reducing our work load */
- char *p;
- zend_stat_t sb;
- size_t dir_len = strlen(dir), offset = 0;
- char *e = buf + strlen(buf);
- if ((p = memchr(buf, DEFAULT_SLASH, dir_len))) {
- offset = p - buf + 1;
- }
- if (p && dir_len == 1) {
- /* buf == "DEFAULT_SLASH" */
- }
- else {
- /* find a top level directory we need to create */
- while ( (p = strrchr(buf + offset, DEFAULT_SLASH)) || (offset != 1 && (p = strrchr(buf, DEFAULT_SLASH))) ) {
- int n = 0;
- *p = '\0';
- while (p > buf && *(p-1) == DEFAULT_SLASH) {
- ++n;
- --p;
- *p = '\0';
- }
- if (VCWD_STAT(buf, &sb) == 0) {
- while (1) {
- *p = DEFAULT_SLASH;
- if (!n) break;
- --n;
- ++p;
- }
- break;
- }
- }
- }
- if (!p) {
- p = buf;
- }
- while (true) {
- int ret = VCWD_MKDIR(buf, (mode_t) mode);
- if (ret < 0 && errno != EEXIST) {
- if (options & REPORT_ERRORS) {
- php_error_docref(NULL, E_WARNING, "%s", strerror(errno));
- }
- return 0;
- }
- bool replaced_slash = false;
- while (++p != e) {
- if (*p == '\0') {
- replaced_slash = true;
- *p = DEFAULT_SLASH;
- if (*(p+1) != '\0') {
- break;
- }
- }
- }
- if (p == e || !replaced_slash) {
- /* No more directories to create */
- /* issue a warning to client when the last directory was created failed */
- if (ret < 0) {
- if (options & REPORT_ERRORS) {
- php_error_docref(NULL, E_WARNING, "%s", strerror(errno));
- }
- return 0;
- }
- return 1;
- }
- }
- }
- static int php_plain_files_rmdir(php_stream_wrapper *wrapper, const char *url, int options, php_stream_context *context)
- {
- if (strncasecmp(url, "file://", sizeof("file://") - 1) == 0) {
- url += sizeof("file://") - 1;
- }
- if (php_check_open_basedir(url)) {
- return 0;
- }
- #ifdef PHP_WIN32
- if (!php_win32_check_trailing_space(url, strlen(url))) {
- php_error_docref1(NULL, url, E_WARNING, "%s", strerror(ENOENT));
- return 0;
- }
- #endif
- if (VCWD_RMDIR(url) < 0) {
- php_error_docref1(NULL, url, E_WARNING, "%s", strerror(errno));
- return 0;
- }
- /* Clear stat cache (and realpath cache) */
- php_clear_stat_cache(1, NULL, 0);
- return 1;
- }
- static int php_plain_files_metadata(php_stream_wrapper *wrapper, const char *url, int option, void *value, php_stream_context *context)
- {
- struct utimbuf *newtime;
- #ifndef PHP_WIN32
- uid_t uid;
- gid_t gid;
- #endif
- mode_t mode;
- int ret = 0;
- #ifdef PHP_WIN32
- if (!php_win32_check_trailing_space(url, strlen(url))) {
- php_error_docref1(NULL, url, E_WARNING, "%s", strerror(ENOENT));
- return 0;
- }
- #endif
- if (strncasecmp(url, "file://", sizeof("file://") - 1) == 0) {
- url += sizeof("file://") - 1;
- }
- if (php_check_open_basedir(url)) {
- return 0;
- }
- switch(option) {
- case PHP_STREAM_META_TOUCH:
- newtime = (struct utimbuf *)value;
- if (VCWD_ACCESS(url, F_OK) != 0) {
- FILE *file = VCWD_FOPEN(url, "w");
- if (file == NULL) {
- php_error_docref1(NULL, url, E_WARNING, "Unable to create file %s because %s", url, strerror(errno));
- return 0;
- }
- fclose(file);
- }
- ret = VCWD_UTIME(url, newtime);
- break;
- #ifndef PHP_WIN32
- case PHP_STREAM_META_OWNER_NAME:
- case PHP_STREAM_META_OWNER:
- if(option == PHP_STREAM_META_OWNER_NAME) {
- if(php_get_uid_by_name((char *)value, &uid) != SUCCESS) {
- php_error_docref1(NULL, url, E_WARNING, "Unable to find uid for %s", (char *)value);
- return 0;
- }
- } else {
- uid = (uid_t)*(long *)value;
- }
- ret = VCWD_CHOWN(url, uid, -1);
- break;
- case PHP_STREAM_META_GROUP:
- case PHP_STREAM_META_GROUP_NAME:
- if(option == PHP_STREAM_META_GROUP_NAME) {
- if(php_get_gid_by_name((char *)value, &gid) != SUCCESS) {
- php_error_docref1(NULL, url, E_WARNING, "Unable to find gid for %s", (char *)value);
- return 0;
- }
- } else {
- gid = (gid_t)*(long *)value;
- }
- ret = VCWD_CHOWN(url, -1, gid);
- break;
- #endif
- case PHP_STREAM_META_ACCESS:
- mode = (mode_t)*(zend_long *)value;
- ret = VCWD_CHMOD(url, mode);
- break;
- default:
- zend_value_error("Unknown option %d for stream_metadata", option);
- return 0;
- }
- if (ret == -1) {
- php_error_docref1(NULL, url, E_WARNING, "Operation failed: %s", strerror(errno));
- return 0;
- }
- php_clear_stat_cache(0, NULL, 0);
- return 1;
- }
- static const php_stream_wrapper_ops php_plain_files_wrapper_ops = {
- php_plain_files_stream_opener,
- NULL,
- NULL,
- php_plain_files_url_stater,
- php_plain_files_dir_opener,
- "plainfile",
- php_plain_files_unlink,
- php_plain_files_rename,
- php_plain_files_mkdir,
- php_plain_files_rmdir,
- php_plain_files_metadata
- };
- /* TODO: We have to make php_plain_files_wrapper writable to support SWOOLE */
- PHPAPI /*const*/ php_stream_wrapper php_plain_files_wrapper = {
- &php_plain_files_wrapper_ops,
- NULL,
- 0
- };
- /* {{{ php_stream_fopen_with_path */
- PHPAPI php_stream *_php_stream_fopen_with_path(const char *filename, const char *mode, const char *path, zend_string **opened_path, int options STREAMS_DC)
- {
- /* code ripped off from fopen_wrappers.c */
- char *pathbuf, *end;
- const char *ptr;
- char trypath[MAXPATHLEN];
- php_stream *stream;
- size_t filename_length;
- zend_string *exec_filename;
- if (opened_path) {
- *opened_path = NULL;
- }
- if(!filename) {
- return NULL;
- }
- filename_length = strlen(filename);
- #ifndef PHP_WIN32
- (void) filename_length;
- #endif
- /* Relative path open */
- if (*filename == '.' && (IS_SLASH(filename[1]) || filename[1] == '.')) {
- /* further checks, we could have ....... filenames */
- ptr = filename + 1;
- if (*ptr == '.') {
- while (*(++ptr) == '.');
- if (!IS_SLASH(*ptr)) { /* not a relative path after all */
- goto not_relative_path;
- }
- }
- if (((options & STREAM_DISABLE_OPEN_BASEDIR) == 0) && php_check_open_basedir(filename)) {
- return NULL;
- }
- return php_stream_fopen_rel(filename, mode, opened_path, options);
- }
- not_relative_path:
- /* Absolute path open */
- if (IS_ABSOLUTE_PATH(filename, filename_length)) {
- if (((options & STREAM_DISABLE_OPEN_BASEDIR) == 0) && php_check_open_basedir(filename)) {
- return NULL;
- }
- return php_stream_fopen_rel(filename, mode, opened_path, options);
- }
- #ifdef PHP_WIN32
- if (IS_SLASH(filename[0])) {
- size_t cwd_len;
- char *cwd;
- cwd = virtual_getcwd_ex(&cwd_len);
- /* getcwd() will return always return [DRIVE_LETTER]:/) on windows. */
- *(cwd+3) = '\0';
- if (snprintf(trypath, MAXPATHLEN, "%s%s", cwd, filename) >= MAXPATHLEN) {
- php_error_docref(NULL, E_NOTICE, "%s/%s path was truncated to %d", cwd, filename, MAXPATHLEN);
- }
- efree(cwd);
- if (((options & STREAM_DISABLE_OPEN_BASEDIR) == 0) && php_check_open_basedir(trypath)) {
- return NULL;
- }
- return php_stream_fopen_rel(trypath, mode, opened_path, options);
- }
- #endif
- if (!path || !*path) {
- return php_stream_fopen_rel(filename, mode, opened_path, options);
- }
- /* check in provided path */
- /* append the calling scripts' current working directory
- * as a fall back case
- */
- if (zend_is_executing() &&
- (exec_filename = zend_get_executed_filename_ex()) != NULL) {
- const char *exec_fname = ZSTR_VAL(exec_filename);
- size_t exec_fname_length = ZSTR_LEN(exec_filename);
- while ((--exec_fname_length < SIZE_MAX) && !IS_SLASH(exec_fname[exec_fname_length]));
- if (exec_fname_length<=0) {
- /* no path */
- pathbuf = estrdup(path);
- } else {
- size_t path_length = strlen(path);
- pathbuf = (char *) emalloc(exec_fname_length + path_length +1 +1);
- memcpy(pathbuf, path, path_length);
- pathbuf[path_length] = DEFAULT_DIR_SEPARATOR;
- memcpy(pathbuf+path_length+1, exec_fname, exec_fname_length);
- pathbuf[path_length + exec_fname_length +1] = '\0';
- }
- } else {
- pathbuf = estrdup(path);
- }
- ptr = pathbuf;
- while (ptr && *ptr) {
- end = strchr(ptr, DEFAULT_DIR_SEPARATOR);
- if (end != NULL) {
- *end = '\0';
- end++;
- }
- if (*ptr == '\0') {
- goto stream_skip;
- }
- if (snprintf(trypath, MAXPATHLEN, "%s/%s", ptr, filename) >= MAXPATHLEN) {
- php_error_docref(NULL, E_NOTICE, "%s/%s path was truncated to %d", ptr, filename, MAXPATHLEN);
- }
- if (((options & STREAM_DISABLE_OPEN_BASEDIR) == 0) && php_check_open_basedir_ex(trypath, 0)) {
- goto stream_skip;
- }
- stream = php_stream_fopen_rel(trypath, mode, opened_path, options);
- if (stream) {
- efree(pathbuf);
- return stream;
- }
- stream_skip:
- ptr = end;
- } /* end provided path */
- efree(pathbuf);
- return NULL;
- }
- /* }}} */
|