func_interceptors.c 40 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216
  1. /*
  2. +----------------------------------------------------------------------+
  3. | phar php single-file executable PHP extension |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 2005-2016 The PHP Group |
  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: Gregory Beaver <cellog@php.net> |
  16. +----------------------------------------------------------------------+
  17. */
  18. /* $Id$ */
  19. #include "phar_internal.h"
  20. #define PHAR_FUNC(name) \
  21. static PHP_NAMED_FUNCTION(name)
  22. PHAR_FUNC(phar_opendir) /* {{{ */
  23. {
  24. char *filename;
  25. int filename_len;
  26. zval *zcontext = NULL;
  27. if (!PHAR_G(intercepted)) {
  28. goto skip_phar;
  29. }
  30. if ((PHAR_GLOBALS->phar_fname_map.arBuckets && !zend_hash_num_elements(&(PHAR_GLOBALS->phar_fname_map)))
  31. && !cached_phars.arBuckets) {
  32. goto skip_phar;
  33. }
  34. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p|z", &filename, &filename_len, &zcontext) == FAILURE) {
  35. return;
  36. }
  37. if (!IS_ABSOLUTE_PATH(filename, filename_len) && !strstr(filename, "://")) {
  38. char *arch, *entry, *fname;
  39. int arch_len, entry_len, fname_len;
  40. fname = (char*)zend_get_executed_filename(TSRMLS_C);
  41. /* we are checking for existence of a file within the relative path. Chances are good that this is
  42. retrieving something from within the phar archive */
  43. if (strncasecmp(fname, "phar://", 7)) {
  44. goto skip_phar;
  45. }
  46. fname_len = strlen(fname);
  47. if (SUCCESS == phar_split_fname(fname, fname_len, &arch, &arch_len, &entry, &entry_len, 2, 0 TSRMLS_CC)) {
  48. php_stream_context *context = NULL;
  49. php_stream *stream;
  50. char *name;
  51. efree(entry);
  52. entry = estrndup(filename, filename_len);
  53. /* fopen within phar, if :// is not in the url, then prepend phar://<archive>/ */
  54. entry_len = filename_len;
  55. /* retrieving a file within the current directory, so use this if possible */
  56. entry = phar_fix_filepath(entry, &entry_len, 1 TSRMLS_CC);
  57. if (entry[0] == '/') {
  58. spprintf(&name, 4096, "phar://%s%s", arch, entry);
  59. } else {
  60. spprintf(&name, 4096, "phar://%s/%s", arch, entry);
  61. }
  62. efree(entry);
  63. efree(arch);
  64. if (zcontext) {
  65. context = php_stream_context_from_zval(zcontext, 0);
  66. }
  67. stream = php_stream_opendir(name, REPORT_ERRORS, context);
  68. efree(name);
  69. if (!stream) {
  70. RETURN_FALSE;
  71. }
  72. php_stream_to_zval(stream, return_value);
  73. return;
  74. }
  75. }
  76. skip_phar:
  77. PHAR_G(orig_opendir)(INTERNAL_FUNCTION_PARAM_PASSTHRU);
  78. return;
  79. }
  80. /* }}} */
  81. PHAR_FUNC(phar_file_get_contents) /* {{{ */
  82. {
  83. char *filename;
  84. int filename_len;
  85. char *contents;
  86. zend_bool use_include_path = 0;
  87. php_stream *stream;
  88. int len;
  89. long offset = -1;
  90. long maxlen = PHP_STREAM_COPY_ALL;
  91. zval *zcontext = NULL;
  92. if (!PHAR_G(intercepted)) {
  93. goto skip_phar;
  94. }
  95. if ((PHAR_GLOBALS->phar_fname_map.arBuckets && !zend_hash_num_elements(&(PHAR_GLOBALS->phar_fname_map)))
  96. && !cached_phars.arBuckets) {
  97. goto skip_phar;
  98. }
  99. /* Parse arguments */
  100. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS() TSRMLS_CC, "p|br!ll", &filename, &filename_len, &use_include_path, &zcontext, &offset, &maxlen) == FAILURE) {
  101. goto skip_phar;
  102. }
  103. if (use_include_path || (!IS_ABSOLUTE_PATH(filename, filename_len) && !strstr(filename, "://"))) {
  104. char *arch, *entry, *fname;
  105. int arch_len, entry_len, fname_len;
  106. php_stream_context *context = NULL;
  107. fname = (char*)zend_get_executed_filename(TSRMLS_C);
  108. if (strncasecmp(fname, "phar://", 7)) {
  109. goto skip_phar;
  110. }
  111. fname_len = strlen(fname);
  112. if (SUCCESS == phar_split_fname(fname, fname_len, &arch, &arch_len, &entry, &entry_len, 2, 0 TSRMLS_CC)) {
  113. char *name;
  114. phar_archive_data *phar;
  115. efree(entry);
  116. entry = filename;
  117. /* fopen within phar, if :// is not in the url, then prepend phar://<archive>/ */
  118. entry_len = filename_len;
  119. if (ZEND_NUM_ARGS() == 5 && maxlen < 0) {
  120. efree(arch);
  121. php_error_docref(NULL TSRMLS_CC, E_WARNING, "length must be greater than or equal to zero");
  122. RETURN_FALSE;
  123. }
  124. /* retrieving a file defaults to within the current directory, so use this if possible */
  125. if (FAILURE == phar_get_archive(&phar, arch, arch_len, NULL, 0, NULL TSRMLS_CC)) {
  126. efree(arch);
  127. goto skip_phar;
  128. }
  129. if (use_include_path) {
  130. if ((entry = phar_find_in_include_path(entry, entry_len, NULL TSRMLS_CC))) {
  131. name = entry;
  132. goto phar_it;
  133. } else {
  134. /* this file is not in the phar, use the original path */
  135. efree(arch);
  136. goto skip_phar;
  137. }
  138. } else {
  139. entry = phar_fix_filepath(estrndup(entry, entry_len), &entry_len, 1 TSRMLS_CC);
  140. if (entry[0] == '/') {
  141. if (!zend_hash_exists(&(phar->manifest), entry + 1, entry_len - 1)) {
  142. /* this file is not in the phar, use the original path */
  143. notfound:
  144. efree(arch);
  145. efree(entry);
  146. goto skip_phar;
  147. }
  148. } else {
  149. if (!zend_hash_exists(&(phar->manifest), entry, entry_len)) {
  150. goto notfound;
  151. }
  152. }
  153. /* auto-convert to phar:// */
  154. if (entry[0] == '/') {
  155. spprintf(&name, 4096, "phar://%s%s", arch, entry);
  156. } else {
  157. spprintf(&name, 4096, "phar://%s/%s", arch, entry);
  158. }
  159. if (entry != filename) {
  160. efree(entry);
  161. }
  162. }
  163. phar_it:
  164. efree(arch);
  165. if (zcontext) {
  166. context = php_stream_context_from_zval(zcontext, 0);
  167. }
  168. stream = php_stream_open_wrapper_ex(name, "rb", 0 | REPORT_ERRORS, NULL, context);
  169. efree(name);
  170. if (!stream) {
  171. RETURN_FALSE;
  172. }
  173. if (offset > 0 && php_stream_seek(stream, offset, SEEK_SET) < 0) {
  174. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Failed to seek to position %ld in the stream", offset);
  175. php_stream_close(stream);
  176. RETURN_FALSE;
  177. }
  178. /* uses mmap if possible */
  179. if ((len = php_stream_copy_to_mem(stream, &contents, maxlen, 0)) > 0) {
  180. #if PHP_API_VERSION < 20100412
  181. if (PG(magic_quotes_runtime)) {
  182. int newlen;
  183. contents = php_addslashes(contents, len, &newlen, 1 TSRMLS_CC); /* 1 = free source string */
  184. len = newlen;
  185. }
  186. #endif
  187. RETVAL_STRINGL(contents, len, 0);
  188. } else if (len == 0) {
  189. RETVAL_EMPTY_STRING();
  190. } else {
  191. RETVAL_FALSE;
  192. }
  193. php_stream_close(stream);
  194. return;
  195. }
  196. }
  197. skip_phar:
  198. PHAR_G(orig_file_get_contents)(INTERNAL_FUNCTION_PARAM_PASSTHRU);
  199. return;
  200. }
  201. /* }}} */
  202. PHAR_FUNC(phar_readfile) /* {{{ */
  203. {
  204. char *filename;
  205. int filename_len;
  206. int size = 0;
  207. zend_bool use_include_path = 0;
  208. zval *zcontext = NULL;
  209. php_stream *stream;
  210. if (!PHAR_G(intercepted)) {
  211. goto skip_phar;
  212. }
  213. if ((PHAR_GLOBALS->phar_fname_map.arBuckets && !zend_hash_num_elements(&(PHAR_GLOBALS->phar_fname_map)))
  214. && !cached_phars.arBuckets) {
  215. goto skip_phar;
  216. }
  217. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS() TSRMLS_CC, "p|br!", &filename, &filename_len, &use_include_path, &zcontext) == FAILURE) {
  218. goto skip_phar;
  219. }
  220. if (use_include_path || (!IS_ABSOLUTE_PATH(filename, filename_len) && !strstr(filename, "://"))) {
  221. char *arch, *entry, *fname;
  222. int arch_len, entry_len, fname_len;
  223. php_stream_context *context = NULL;
  224. char *name;
  225. phar_archive_data *phar;
  226. fname = (char*)zend_get_executed_filename(TSRMLS_C);
  227. if (strncasecmp(fname, "phar://", 7)) {
  228. goto skip_phar;
  229. }
  230. fname_len = strlen(fname);
  231. if (FAILURE == phar_split_fname(fname, fname_len, &arch, &arch_len, &entry, &entry_len, 2, 0 TSRMLS_CC)) {
  232. goto skip_phar;
  233. }
  234. efree(entry);
  235. entry = filename;
  236. /* fopen within phar, if :// is not in the url, then prepend phar://<archive>/ */
  237. entry_len = filename_len;
  238. /* retrieving a file defaults to within the current directory, so use this if possible */
  239. if (FAILURE == phar_get_archive(&phar, arch, arch_len, NULL, 0, NULL TSRMLS_CC)) {
  240. efree(arch);
  241. goto skip_phar;
  242. }
  243. if (use_include_path) {
  244. if (!(entry = phar_find_in_include_path(entry, entry_len, NULL TSRMLS_CC))) {
  245. /* this file is not in the phar, use the original path */
  246. efree(arch);
  247. goto skip_phar;
  248. } else {
  249. name = entry;
  250. }
  251. } else {
  252. entry = phar_fix_filepath(estrndup(entry, entry_len), &entry_len, 1 TSRMLS_CC);
  253. if (entry[0] == '/') {
  254. if (!zend_hash_exists(&(phar->manifest), entry + 1, entry_len - 1)) {
  255. /* this file is not in the phar, use the original path */
  256. notfound:
  257. efree(entry);
  258. efree(arch);
  259. goto skip_phar;
  260. }
  261. } else {
  262. if (!zend_hash_exists(&(phar->manifest), entry, entry_len)) {
  263. goto notfound;
  264. }
  265. }
  266. /* auto-convert to phar:// */
  267. if (entry[0] == '/') {
  268. spprintf(&name, 4096, "phar://%s%s", arch, entry);
  269. } else {
  270. spprintf(&name, 4096, "phar://%s/%s", arch, entry);
  271. }
  272. efree(entry);
  273. }
  274. efree(arch);
  275. context = php_stream_context_from_zval(zcontext, 0);
  276. stream = php_stream_open_wrapper_ex(name, "rb", 0 | REPORT_ERRORS, NULL, context);
  277. efree(name);
  278. if (stream == NULL) {
  279. RETURN_FALSE;
  280. }
  281. size = php_stream_passthru(stream);
  282. php_stream_close(stream);
  283. RETURN_LONG(size);
  284. }
  285. skip_phar:
  286. PHAR_G(orig_readfile)(INTERNAL_FUNCTION_PARAM_PASSTHRU);
  287. return;
  288. }
  289. /* }}} */
  290. PHAR_FUNC(phar_fopen) /* {{{ */
  291. {
  292. char *filename, *mode;
  293. int filename_len, mode_len;
  294. zend_bool use_include_path = 0;
  295. zval *zcontext = NULL;
  296. php_stream *stream;
  297. if (!PHAR_G(intercepted)) {
  298. goto skip_phar;
  299. }
  300. if ((PHAR_GLOBALS->phar_fname_map.arBuckets && !zend_hash_num_elements(&(PHAR_GLOBALS->phar_fname_map)))
  301. && !cached_phars.arBuckets) {
  302. /* no need to check, include_path not even specified in fopen/ no active phars */
  303. goto skip_phar;
  304. }
  305. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS() TSRMLS_CC, "ps|br", &filename, &filename_len, &mode, &mode_len, &use_include_path, &zcontext) == FAILURE) {
  306. goto skip_phar;
  307. }
  308. if (use_include_path || (!IS_ABSOLUTE_PATH(filename, filename_len) && !strstr(filename, "://"))) {
  309. char *arch, *entry, *fname;
  310. int arch_len, entry_len, fname_len;
  311. php_stream_context *context = NULL;
  312. char *name;
  313. phar_archive_data *phar;
  314. fname = (char*)zend_get_executed_filename(TSRMLS_C);
  315. if (strncasecmp(fname, "phar://", 7)) {
  316. goto skip_phar;
  317. }
  318. fname_len = strlen(fname);
  319. if (FAILURE == phar_split_fname(fname, fname_len, &arch, &arch_len, &entry, &entry_len, 2, 0 TSRMLS_CC)) {
  320. goto skip_phar;
  321. }
  322. efree(entry);
  323. entry = filename;
  324. /* fopen within phar, if :// is not in the url, then prepend phar://<archive>/ */
  325. entry_len = filename_len;
  326. /* retrieving a file defaults to within the current directory, so use this if possible */
  327. if (FAILURE == phar_get_archive(&phar, arch, arch_len, NULL, 0, NULL TSRMLS_CC)) {
  328. efree(arch);
  329. goto skip_phar;
  330. }
  331. if (use_include_path) {
  332. if (!(entry = phar_find_in_include_path(entry, entry_len, NULL TSRMLS_CC))) {
  333. /* this file is not in the phar, use the original path */
  334. efree(arch);
  335. goto skip_phar;
  336. } else {
  337. name = entry;
  338. }
  339. } else {
  340. entry = phar_fix_filepath(estrndup(entry, entry_len), &entry_len, 1 TSRMLS_CC);
  341. if (entry[0] == '/') {
  342. if (!zend_hash_exists(&(phar->manifest), entry + 1, entry_len - 1)) {
  343. /* this file is not in the phar, use the original path */
  344. notfound:
  345. efree(entry);
  346. efree(arch);
  347. goto skip_phar;
  348. }
  349. } else {
  350. if (!zend_hash_exists(&(phar->manifest), entry, entry_len)) {
  351. /* this file is not in the phar, use the original path */
  352. goto notfound;
  353. }
  354. }
  355. /* auto-convert to phar:// */
  356. if (entry[0] == '/') {
  357. spprintf(&name, 4096, "phar://%s%s", arch, entry);
  358. } else {
  359. spprintf(&name, 4096, "phar://%s/%s", arch, entry);
  360. }
  361. efree(entry);
  362. }
  363. efree(arch);
  364. context = php_stream_context_from_zval(zcontext, 0);
  365. stream = php_stream_open_wrapper_ex(name, mode, 0 | REPORT_ERRORS, NULL, context);
  366. efree(name);
  367. if (stream == NULL) {
  368. RETURN_FALSE;
  369. }
  370. php_stream_to_zval(stream, return_value);
  371. if (zcontext) {
  372. zend_list_addref(Z_RESVAL_P(zcontext));
  373. }
  374. return;
  375. }
  376. skip_phar:
  377. PHAR_G(orig_fopen)(INTERNAL_FUNCTION_PARAM_PASSTHRU);
  378. return;
  379. }
  380. /* }}} */
  381. #ifndef S_ISDIR
  382. #define S_ISDIR(mode) (((mode)&S_IFMT) == S_IFDIR)
  383. #endif
  384. #ifndef S_ISREG
  385. #define S_ISREG(mode) (((mode)&S_IFMT) == S_IFREG)
  386. #endif
  387. #ifndef S_ISLNK
  388. #define S_ISLNK(mode) (((mode)&S_IFMT) == S_IFLNK)
  389. #endif
  390. #define S_IXROOT ( S_IXUSR | S_IXGRP | S_IXOTH )
  391. #define IS_LINK_OPERATION(__t) ((__t) == FS_TYPE || (__t) == FS_IS_LINK || (__t) == FS_LSTAT)
  392. #define IS_EXISTS_CHECK(__t) ((__t) == FS_EXISTS || (__t) == FS_IS_W || (__t) == FS_IS_R || (__t) == FS_IS_X || (__t) == FS_IS_FILE || (__t) == FS_IS_DIR || (__t) == FS_IS_LINK)
  393. #define IS_ABLE_CHECK(__t) ((__t) == FS_IS_R || (__t) == FS_IS_W || (__t) == FS_IS_X)
  394. #define IS_ACCESS_CHECK(__t) (IS_ABLE_CHECK(type) || (__t) == FS_EXISTS)
  395. /* {{{ php_stat
  396. */
  397. static void phar_fancy_stat(struct stat *stat_sb, int type, zval *return_value TSRMLS_DC)
  398. {
  399. zval *stat_dev, *stat_ino, *stat_mode, *stat_nlink, *stat_uid, *stat_gid, *stat_rdev,
  400. *stat_size, *stat_atime, *stat_mtime, *stat_ctime, *stat_blksize, *stat_blocks;
  401. int rmask=S_IROTH, wmask=S_IWOTH, xmask=S_IXOTH; /* access rights defaults to other */
  402. char *stat_sb_names[13] = {
  403. "dev", "ino", "mode", "nlink", "uid", "gid", "rdev",
  404. "size", "atime", "mtime", "ctime", "blksize", "blocks"
  405. };
  406. #ifndef NETWARE
  407. if (type >= FS_IS_W && type <= FS_IS_X) {
  408. if(stat_sb->st_uid==getuid()) {
  409. rmask=S_IRUSR;
  410. wmask=S_IWUSR;
  411. xmask=S_IXUSR;
  412. } else if(stat_sb->st_gid==getgid()) {
  413. rmask=S_IRGRP;
  414. wmask=S_IWGRP;
  415. xmask=S_IXGRP;
  416. } else {
  417. int groups, n, i;
  418. gid_t *gids;
  419. groups = getgroups(0, NULL);
  420. if(groups > 0) {
  421. gids=(gid_t *)safe_emalloc(groups, sizeof(gid_t), 0);
  422. n=getgroups(groups, gids);
  423. for(i=0;i<n;++i){
  424. if(stat_sb->st_gid==gids[i]) {
  425. rmask=S_IRGRP;
  426. wmask=S_IWGRP;
  427. xmask=S_IXGRP;
  428. break;
  429. }
  430. }
  431. efree(gids);
  432. }
  433. }
  434. }
  435. #endif
  436. switch (type) {
  437. case FS_PERMS:
  438. RETURN_LONG((long)stat_sb->st_mode);
  439. case FS_INODE:
  440. RETURN_LONG((long)stat_sb->st_ino);
  441. case FS_SIZE:
  442. RETURN_LONG((long)stat_sb->st_size);
  443. case FS_OWNER:
  444. RETURN_LONG((long)stat_sb->st_uid);
  445. case FS_GROUP:
  446. RETURN_LONG((long)stat_sb->st_gid);
  447. case FS_ATIME:
  448. #ifdef NETWARE
  449. RETURN_LONG((long)stat_sb->st_atime.tv_sec);
  450. #else
  451. RETURN_LONG((long)stat_sb->st_atime);
  452. #endif
  453. case FS_MTIME:
  454. #ifdef NETWARE
  455. RETURN_LONG((long)stat_sb->st_mtime.tv_sec);
  456. #else
  457. RETURN_LONG((long)stat_sb->st_mtime);
  458. #endif
  459. case FS_CTIME:
  460. #ifdef NETWARE
  461. RETURN_LONG((long)stat_sb->st_ctime.tv_sec);
  462. #else
  463. RETURN_LONG((long)stat_sb->st_ctime);
  464. #endif
  465. case FS_TYPE:
  466. if (S_ISLNK(stat_sb->st_mode)) {
  467. RETURN_STRING("link", 1);
  468. }
  469. switch(stat_sb->st_mode & S_IFMT) {
  470. case S_IFDIR: RETURN_STRING("dir", 1);
  471. case S_IFREG: RETURN_STRING("file", 1);
  472. }
  473. php_error_docref(NULL TSRMLS_CC, E_NOTICE, "Unknown file type (%u)", stat_sb->st_mode & S_IFMT);
  474. RETURN_STRING("unknown", 1);
  475. case FS_IS_W:
  476. RETURN_BOOL((stat_sb->st_mode & wmask) != 0);
  477. case FS_IS_R:
  478. RETURN_BOOL((stat_sb->st_mode&rmask)!=0);
  479. case FS_IS_X:
  480. RETURN_BOOL((stat_sb->st_mode&xmask)!=0 && !S_ISDIR(stat_sb->st_mode));
  481. case FS_IS_FILE:
  482. RETURN_BOOL(S_ISREG(stat_sb->st_mode));
  483. case FS_IS_DIR:
  484. RETURN_BOOL(S_ISDIR(stat_sb->st_mode));
  485. case FS_IS_LINK:
  486. RETURN_BOOL(S_ISLNK(stat_sb->st_mode));
  487. case FS_EXISTS:
  488. RETURN_TRUE; /* the false case was done earlier */
  489. case FS_LSTAT:
  490. /* FALLTHROUGH */
  491. case FS_STAT:
  492. array_init(return_value);
  493. MAKE_LONG_ZVAL_INCREF(stat_dev, stat_sb->st_dev);
  494. MAKE_LONG_ZVAL_INCREF(stat_ino, stat_sb->st_ino);
  495. MAKE_LONG_ZVAL_INCREF(stat_mode, stat_sb->st_mode);
  496. MAKE_LONG_ZVAL_INCREF(stat_nlink, stat_sb->st_nlink);
  497. MAKE_LONG_ZVAL_INCREF(stat_uid, stat_sb->st_uid);
  498. MAKE_LONG_ZVAL_INCREF(stat_gid, stat_sb->st_gid);
  499. #ifdef HAVE_ST_RDEV
  500. MAKE_LONG_ZVAL_INCREF(stat_rdev, stat_sb->st_rdev);
  501. #else
  502. MAKE_LONG_ZVAL_INCREF(stat_rdev, -1);
  503. #endif
  504. MAKE_LONG_ZVAL_INCREF(stat_size, stat_sb->st_size);
  505. #ifdef NETWARE
  506. MAKE_LONG_ZVAL_INCREF(stat_atime, (stat_sb->st_atime).tv_sec);
  507. MAKE_LONG_ZVAL_INCREF(stat_mtime, (stat_sb->st_mtime).tv_sec);
  508. MAKE_LONG_ZVAL_INCREF(stat_ctime, (stat_sb->st_ctime).tv_sec);
  509. #else
  510. MAKE_LONG_ZVAL_INCREF(stat_atime, stat_sb->st_atime);
  511. MAKE_LONG_ZVAL_INCREF(stat_mtime, stat_sb->st_mtime);
  512. MAKE_LONG_ZVAL_INCREF(stat_ctime, stat_sb->st_ctime);
  513. #endif
  514. #ifdef HAVE_ST_BLKSIZE
  515. MAKE_LONG_ZVAL_INCREF(stat_blksize, stat_sb->st_blksize);
  516. #else
  517. MAKE_LONG_ZVAL_INCREF(stat_blksize,-1);
  518. #endif
  519. #ifdef HAVE_ST_BLOCKS
  520. MAKE_LONG_ZVAL_INCREF(stat_blocks, stat_sb->st_blocks);
  521. #else
  522. MAKE_LONG_ZVAL_INCREF(stat_blocks,-1);
  523. #endif
  524. /* Store numeric indexes in proper order */
  525. zend_hash_next_index_insert(HASH_OF(return_value), (void *)&stat_dev, sizeof(zval *), NULL);
  526. zend_hash_next_index_insert(HASH_OF(return_value), (void *)&stat_ino, sizeof(zval *), NULL);
  527. zend_hash_next_index_insert(HASH_OF(return_value), (void *)&stat_mode, sizeof(zval *), NULL);
  528. zend_hash_next_index_insert(HASH_OF(return_value), (void *)&stat_nlink, sizeof(zval *), NULL);
  529. zend_hash_next_index_insert(HASH_OF(return_value), (void *)&stat_uid, sizeof(zval *), NULL);
  530. zend_hash_next_index_insert(HASH_OF(return_value), (void *)&stat_gid, sizeof(zval *), NULL);
  531. zend_hash_next_index_insert(HASH_OF(return_value), (void *)&stat_rdev, sizeof(zval *), NULL);
  532. zend_hash_next_index_insert(HASH_OF(return_value), (void *)&stat_size, sizeof(zval *), NULL);
  533. zend_hash_next_index_insert(HASH_OF(return_value), (void *)&stat_atime, sizeof(zval *), NULL);
  534. zend_hash_next_index_insert(HASH_OF(return_value), (void *)&stat_mtime, sizeof(zval *), NULL);
  535. zend_hash_next_index_insert(HASH_OF(return_value), (void *)&stat_ctime, sizeof(zval *), NULL);
  536. zend_hash_next_index_insert(HASH_OF(return_value), (void *)&stat_blksize, sizeof(zval *), NULL);
  537. zend_hash_next_index_insert(HASH_OF(return_value), (void *)&stat_blocks, sizeof(zval *), NULL);
  538. /* Store string indexes referencing the same zval*/
  539. zend_hash_update(HASH_OF(return_value), stat_sb_names[0], strlen(stat_sb_names[0])+1, (void *) &stat_dev, sizeof(zval *), NULL);
  540. zend_hash_update(HASH_OF(return_value), stat_sb_names[1], strlen(stat_sb_names[1])+1, (void *) &stat_ino, sizeof(zval *), NULL);
  541. zend_hash_update(HASH_OF(return_value), stat_sb_names[2], strlen(stat_sb_names[2])+1, (void *) &stat_mode, sizeof(zval *), NULL);
  542. zend_hash_update(HASH_OF(return_value), stat_sb_names[3], strlen(stat_sb_names[3])+1, (void *) &stat_nlink, sizeof(zval *), NULL);
  543. zend_hash_update(HASH_OF(return_value), stat_sb_names[4], strlen(stat_sb_names[4])+1, (void *) &stat_uid, sizeof(zval *), NULL);
  544. zend_hash_update(HASH_OF(return_value), stat_sb_names[5], strlen(stat_sb_names[5])+1, (void *) &stat_gid, sizeof(zval *), NULL);
  545. zend_hash_update(HASH_OF(return_value), stat_sb_names[6], strlen(stat_sb_names[6])+1, (void *) &stat_rdev, sizeof(zval *), NULL);
  546. zend_hash_update(HASH_OF(return_value), stat_sb_names[7], strlen(stat_sb_names[7])+1, (void *) &stat_size, sizeof(zval *), NULL);
  547. zend_hash_update(HASH_OF(return_value), stat_sb_names[8], strlen(stat_sb_names[8])+1, (void *) &stat_atime, sizeof(zval *), NULL);
  548. zend_hash_update(HASH_OF(return_value), stat_sb_names[9], strlen(stat_sb_names[9])+1, (void *) &stat_mtime, sizeof(zval *), NULL);
  549. zend_hash_update(HASH_OF(return_value), stat_sb_names[10], strlen(stat_sb_names[10])+1, (void *) &stat_ctime, sizeof(zval *), NULL);
  550. zend_hash_update(HASH_OF(return_value), stat_sb_names[11], strlen(stat_sb_names[11])+1, (void *) &stat_blksize, sizeof(zval *), NULL);
  551. zend_hash_update(HASH_OF(return_value), stat_sb_names[12], strlen(stat_sb_names[12])+1, (void *) &stat_blocks, sizeof(zval *), NULL);
  552. return;
  553. }
  554. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Didn't understand stat call");
  555. RETURN_FALSE;
  556. }
  557. /* }}} */
  558. static void phar_file_stat(const char *filename, php_stat_len filename_length, int type, void (*orig_stat_func)(INTERNAL_FUNCTION_PARAMETERS), INTERNAL_FUNCTION_PARAMETERS) /* {{{ */
  559. {
  560. if (!filename_length) {
  561. RETURN_FALSE;
  562. }
  563. if (!IS_ABSOLUTE_PATH(filename, filename_length) && !strstr(filename, "://")) {
  564. char *arch, *entry, *fname;
  565. int arch_len, entry_len, fname_len;
  566. struct stat sb = {0};
  567. phar_entry_info *data = NULL;
  568. phar_archive_data *phar;
  569. fname = (char*)zend_get_executed_filename(TSRMLS_C);
  570. /* we are checking for existence of a file within the relative path. Chances are good that this is
  571. retrieving something from within the phar archive */
  572. if (strncasecmp(fname, "phar://", 7)) {
  573. goto skip_phar;
  574. }
  575. fname_len = strlen(fname);
  576. if (PHAR_G(last_phar) && fname_len - 7 >= PHAR_G(last_phar_name_len) && !memcmp(fname + 7, PHAR_G(last_phar_name), PHAR_G(last_phar_name_len))) {
  577. arch = estrndup(PHAR_G(last_phar_name), PHAR_G(last_phar_name_len));
  578. arch_len = PHAR_G(last_phar_name_len);
  579. entry = estrndup(filename, filename_length);
  580. /* fopen within phar, if :// is not in the url, then prepend phar://<archive>/ */
  581. entry_len = (int) filename_length;
  582. phar = PHAR_G(last_phar);
  583. goto splitted;
  584. }
  585. if (SUCCESS == phar_split_fname(fname, fname_len, &arch, &arch_len, &entry, &entry_len, 2, 0 TSRMLS_CC)) {
  586. efree(entry);
  587. entry = estrndup(filename, filename_length);
  588. /* fopen within phar, if :// is not in the url, then prepend phar://<archive>/ */
  589. entry_len = (int) filename_length;
  590. if (FAILURE == phar_get_archive(&phar, arch, arch_len, NULL, 0, NULL TSRMLS_CC)) {
  591. efree(arch);
  592. efree(entry);
  593. goto skip_phar;
  594. }
  595. splitted:
  596. entry = phar_fix_filepath(entry, &entry_len, 1 TSRMLS_CC);
  597. if (entry[0] == '/') {
  598. if (SUCCESS == zend_hash_find(&(phar->manifest), entry + 1, entry_len - 1, (void **) &data)) {
  599. efree(entry);
  600. goto stat_entry;
  601. }
  602. goto notfound;
  603. }
  604. if (SUCCESS == zend_hash_find(&(phar->manifest), entry, entry_len, (void **) &data)) {
  605. efree(entry);
  606. goto stat_entry;
  607. }
  608. if (zend_hash_exists(&(phar->virtual_dirs), entry, entry_len)) {
  609. efree(entry);
  610. efree(arch);
  611. if (IS_EXISTS_CHECK(type)) {
  612. RETURN_TRUE;
  613. }
  614. sb.st_size = 0;
  615. sb.st_mode = 0777;
  616. sb.st_mode |= S_IFDIR; /* regular directory */
  617. #ifdef NETWARE
  618. sb.st_mtime.tv_sec = phar->max_timestamp;
  619. sb.st_atime.tv_sec = phar->max_timestamp;
  620. sb.st_ctime.tv_sec = phar->max_timestamp;
  621. #else
  622. sb.st_mtime = phar->max_timestamp;
  623. sb.st_atime = phar->max_timestamp;
  624. sb.st_ctime = phar->max_timestamp;
  625. #endif
  626. goto statme_baby;
  627. } else {
  628. char *save;
  629. int save_len;
  630. notfound:
  631. efree(entry);
  632. save = PHAR_G(cwd);
  633. save_len = PHAR_G(cwd_len);
  634. /* this file is not in the current directory, use the original path */
  635. entry = estrndup(filename, filename_length);
  636. entry_len = filename_length;
  637. PHAR_G(cwd) = "/";
  638. PHAR_G(cwd_len) = 0;
  639. /* clean path without cwd */
  640. entry = phar_fix_filepath(entry, &entry_len, 1 TSRMLS_CC);
  641. if (SUCCESS == zend_hash_find(&(phar->manifest), entry + 1, entry_len - 1, (void **) &data)) {
  642. PHAR_G(cwd) = save;
  643. PHAR_G(cwd_len) = save_len;
  644. efree(entry);
  645. if (IS_EXISTS_CHECK(type)) {
  646. efree(arch);
  647. RETURN_TRUE;
  648. }
  649. goto stat_entry;
  650. }
  651. if (zend_hash_exists(&(phar->virtual_dirs), entry + 1, entry_len - 1)) {
  652. PHAR_G(cwd) = save;
  653. PHAR_G(cwd_len) = save_len;
  654. efree(entry);
  655. efree(arch);
  656. if (IS_EXISTS_CHECK(type)) {
  657. RETURN_TRUE;
  658. }
  659. sb.st_size = 0;
  660. sb.st_mode = 0777;
  661. sb.st_mode |= S_IFDIR; /* regular directory */
  662. #ifdef NETWARE
  663. sb.st_mtime.tv_sec = phar->max_timestamp;
  664. sb.st_atime.tv_sec = phar->max_timestamp;
  665. sb.st_ctime.tv_sec = phar->max_timestamp;
  666. #else
  667. sb.st_mtime = phar->max_timestamp;
  668. sb.st_atime = phar->max_timestamp;
  669. sb.st_ctime = phar->max_timestamp;
  670. #endif
  671. goto statme_baby;
  672. }
  673. PHAR_G(cwd) = save;
  674. PHAR_G(cwd_len) = save_len;
  675. efree(entry);
  676. efree(arch);
  677. /* Error Occurred */
  678. if (!IS_EXISTS_CHECK(type)) {
  679. php_error_docref(NULL TSRMLS_CC, E_WARNING, "%sstat failed for %s", IS_LINK_OPERATION(type) ? "L" : "", filename);
  680. }
  681. RETURN_FALSE;
  682. }
  683. stat_entry:
  684. efree(arch);
  685. if (!data->is_dir) {
  686. sb.st_size = data->uncompressed_filesize;
  687. sb.st_mode = data->flags & PHAR_ENT_PERM_MASK;
  688. if (data->link) {
  689. sb.st_mode |= S_IFREG|S_IFLNK; /* regular file */
  690. } else {
  691. sb.st_mode |= S_IFREG; /* regular file */
  692. }
  693. /* timestamp is just the timestamp when this was added to the phar */
  694. #ifdef NETWARE
  695. sb.st_mtime.tv_sec = data->timestamp;
  696. sb.st_atime.tv_sec = data->timestamp;
  697. sb.st_ctime.tv_sec = data->timestamp;
  698. #else
  699. sb.st_mtime = data->timestamp;
  700. sb.st_atime = data->timestamp;
  701. sb.st_ctime = data->timestamp;
  702. #endif
  703. } else {
  704. sb.st_size = 0;
  705. sb.st_mode = data->flags & PHAR_ENT_PERM_MASK;
  706. sb.st_mode |= S_IFDIR; /* regular directory */
  707. if (data->link) {
  708. sb.st_mode |= S_IFLNK;
  709. }
  710. /* timestamp is just the timestamp when this was added to the phar */
  711. #ifdef NETWARE
  712. sb.st_mtime.tv_sec = data->timestamp;
  713. sb.st_atime.tv_sec = data->timestamp;
  714. sb.st_ctime.tv_sec = data->timestamp;
  715. #else
  716. sb.st_mtime = data->timestamp;
  717. sb.st_atime = data->timestamp;
  718. sb.st_ctime = data->timestamp;
  719. #endif
  720. }
  721. statme_baby:
  722. if (!phar->is_writeable) {
  723. sb.st_mode = (sb.st_mode & 0555) | (sb.st_mode & ~0777);
  724. }
  725. sb.st_nlink = 1;
  726. sb.st_rdev = -1;
  727. /* this is only for APC, so use /dev/null device - no chance of conflict there! */
  728. sb.st_dev = 0xc;
  729. /* generate unique inode number for alias/filename, so no phars will conflict */
  730. if (data) {
  731. sb.st_ino = data->inode;
  732. }
  733. #ifndef PHP_WIN32
  734. sb.st_blksize = -1;
  735. sb.st_blocks = -1;
  736. #endif
  737. phar_fancy_stat(&sb, type, return_value TSRMLS_CC);
  738. return;
  739. }
  740. }
  741. skip_phar:
  742. orig_stat_func(INTERNAL_FUNCTION_PARAM_PASSTHRU);
  743. return;
  744. }
  745. /* }}} */
  746. #define PharFileFunction(fname, funcnum, orig) \
  747. void fname(INTERNAL_FUNCTION_PARAMETERS) { \
  748. if (!PHAR_G(intercepted)) { \
  749. PHAR_G(orig)(INTERNAL_FUNCTION_PARAM_PASSTHRU); \
  750. } else { \
  751. char *filename; \
  752. int filename_len; \
  753. \
  754. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p", &filename, &filename_len) == FAILURE) { \
  755. return; \
  756. } \
  757. \
  758. phar_file_stat(filename, (php_stat_len) filename_len, funcnum, PHAR_G(orig), INTERNAL_FUNCTION_PARAM_PASSTHRU); \
  759. } \
  760. }
  761. /* }}} */
  762. /* {{{ proto int fileperms(string filename)
  763. Get file permissions */
  764. PharFileFunction(phar_fileperms, FS_PERMS, orig_fileperms)
  765. /* }}} */
  766. /* {{{ proto int fileinode(string filename)
  767. Get file inode */
  768. PharFileFunction(phar_fileinode, FS_INODE, orig_fileinode)
  769. /* }}} */
  770. /* {{{ proto int filesize(string filename)
  771. Get file size */
  772. PharFileFunction(phar_filesize, FS_SIZE, orig_filesize)
  773. /* }}} */
  774. /* {{{ proto int fileowner(string filename)
  775. Get file owner */
  776. PharFileFunction(phar_fileowner, FS_OWNER, orig_fileowner)
  777. /* }}} */
  778. /* {{{ proto int filegroup(string filename)
  779. Get file group */
  780. PharFileFunction(phar_filegroup, FS_GROUP, orig_filegroup)
  781. /* }}} */
  782. /* {{{ proto int fileatime(string filename)
  783. Get last access time of file */
  784. PharFileFunction(phar_fileatime, FS_ATIME, orig_fileatime)
  785. /* }}} */
  786. /* {{{ proto int filemtime(string filename)
  787. Get last modification time of file */
  788. PharFileFunction(phar_filemtime, FS_MTIME, orig_filemtime)
  789. /* }}} */
  790. /* {{{ proto int filectime(string filename)
  791. Get inode modification time of file */
  792. PharFileFunction(phar_filectime, FS_CTIME, orig_filectime)
  793. /* }}} */
  794. /* {{{ proto string filetype(string filename)
  795. Get file type */
  796. PharFileFunction(phar_filetype, FS_TYPE, orig_filetype)
  797. /* }}} */
  798. /* {{{ proto bool is_writable(string filename)
  799. Returns true if file can be written */
  800. PharFileFunction(phar_is_writable, FS_IS_W, orig_is_writable)
  801. /* }}} */
  802. /* {{{ proto bool is_readable(string filename)
  803. Returns true if file can be read */
  804. PharFileFunction(phar_is_readable, FS_IS_R, orig_is_readable)
  805. /* }}} */
  806. /* {{{ proto bool is_executable(string filename)
  807. Returns true if file is executable */
  808. PharFileFunction(phar_is_executable, FS_IS_X, orig_is_executable)
  809. /* }}} */
  810. /* {{{ proto bool file_exists(string filename)
  811. Returns true if filename exists */
  812. PharFileFunction(phar_file_exists, FS_EXISTS, orig_file_exists)
  813. /* }}} */
  814. /* {{{ proto bool is_dir(string filename)
  815. Returns true if file is directory */
  816. PharFileFunction(phar_is_dir, FS_IS_DIR, orig_is_dir)
  817. /* }}} */
  818. PHAR_FUNC(phar_is_file) /* {{{ */
  819. {
  820. char *filename;
  821. int filename_len;
  822. if (!PHAR_G(intercepted)) {
  823. goto skip_phar;
  824. }
  825. if ((PHAR_GLOBALS->phar_fname_map.arBuckets && !zend_hash_num_elements(&(PHAR_GLOBALS->phar_fname_map)))
  826. && !cached_phars.arBuckets) {
  827. goto skip_phar;
  828. }
  829. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS() TSRMLS_CC, "p", &filename, &filename_len) == FAILURE) {
  830. goto skip_phar;
  831. }
  832. if (!IS_ABSOLUTE_PATH(filename, filename_len) && !strstr(filename, "://")) {
  833. char *arch, *entry, *fname;
  834. int arch_len, entry_len, fname_len;
  835. fname = (char*)zend_get_executed_filename(TSRMLS_C);
  836. /* we are checking for existence of a file within the relative path. Chances are good that this is
  837. retrieving something from within the phar archive */
  838. if (strncasecmp(fname, "phar://", 7)) {
  839. goto skip_phar;
  840. }
  841. fname_len = strlen(fname);
  842. if (SUCCESS == phar_split_fname(fname, fname_len, &arch, &arch_len, &entry, &entry_len, 2, 0 TSRMLS_CC)) {
  843. phar_archive_data *phar;
  844. efree(entry);
  845. entry = filename;
  846. /* fopen within phar, if :// is not in the url, then prepend phar://<archive>/ */
  847. entry_len = filename_len;
  848. /* retrieving a file within the current directory, so use this if possible */
  849. if (SUCCESS == phar_get_archive(&phar, arch, arch_len, NULL, 0, NULL TSRMLS_CC)) {
  850. phar_entry_info *etemp;
  851. entry = phar_fix_filepath(estrndup(entry, entry_len), &entry_len, 1 TSRMLS_CC);
  852. if (entry[0] == '/') {
  853. if (SUCCESS == zend_hash_find(&(phar->manifest), entry + 1, entry_len - 1, (void **) &etemp)) {
  854. /* this file is not in the current directory, use the original path */
  855. found_it:
  856. efree(entry);
  857. efree(arch);
  858. RETURN_BOOL(!etemp->is_dir);
  859. }
  860. } else {
  861. if (SUCCESS == zend_hash_find(&(phar->manifest), entry, entry_len, (void **) &etemp)) {
  862. goto found_it;
  863. }
  864. }
  865. }
  866. if (entry != filename) {
  867. efree(entry);
  868. }
  869. efree(arch);
  870. RETURN_FALSE;
  871. }
  872. }
  873. skip_phar:
  874. PHAR_G(orig_is_file)(INTERNAL_FUNCTION_PARAM_PASSTHRU);
  875. return;
  876. }
  877. /* }}} */
  878. PHAR_FUNC(phar_is_link) /* {{{ */
  879. {
  880. char *filename;
  881. int filename_len;
  882. if (!PHAR_G(intercepted)) {
  883. goto skip_phar;
  884. }
  885. if ((PHAR_GLOBALS->phar_fname_map.arBuckets && !zend_hash_num_elements(&(PHAR_GLOBALS->phar_fname_map)))
  886. && !cached_phars.arBuckets) {
  887. goto skip_phar;
  888. }
  889. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS() TSRMLS_CC, "p", &filename, &filename_len) == FAILURE) {
  890. goto skip_phar;
  891. }
  892. if (!IS_ABSOLUTE_PATH(filename, filename_len) && !strstr(filename, "://")) {
  893. char *arch, *entry, *fname;
  894. int arch_len, entry_len, fname_len;
  895. fname = (char*)zend_get_executed_filename(TSRMLS_C);
  896. /* we are checking for existence of a file within the relative path. Chances are good that this is
  897. retrieving something from within the phar archive */
  898. if (strncasecmp(fname, "phar://", 7)) {
  899. goto skip_phar;
  900. }
  901. fname_len = strlen(fname);
  902. if (SUCCESS == phar_split_fname(fname, fname_len, &arch, &arch_len, &entry, &entry_len, 2, 0 TSRMLS_CC)) {
  903. phar_archive_data *phar;
  904. efree(entry);
  905. entry = filename;
  906. /* fopen within phar, if :// is not in the url, then prepend phar://<archive>/ */
  907. entry_len = filename_len;
  908. /* retrieving a file within the current directory, so use this if possible */
  909. if (SUCCESS == phar_get_archive(&phar, arch, arch_len, NULL, 0, NULL TSRMLS_CC)) {
  910. phar_entry_info *etemp;
  911. entry = phar_fix_filepath(estrndup(entry, entry_len), &entry_len, 1 TSRMLS_CC);
  912. if (entry[0] == '/') {
  913. if (SUCCESS == zend_hash_find(&(phar->manifest), entry + 1, entry_len - 1, (void **) &etemp)) {
  914. /* this file is not in the current directory, use the original path */
  915. found_it:
  916. efree(entry);
  917. efree(arch);
  918. RETURN_BOOL(etemp->link);
  919. }
  920. } else {
  921. if (SUCCESS == zend_hash_find(&(phar->manifest), entry, entry_len, (void **) &etemp)) {
  922. goto found_it;
  923. }
  924. }
  925. }
  926. efree(entry);
  927. efree(arch);
  928. RETURN_FALSE;
  929. }
  930. }
  931. skip_phar:
  932. PHAR_G(orig_is_link)(INTERNAL_FUNCTION_PARAM_PASSTHRU);
  933. return;
  934. }
  935. /* }}} */
  936. /* {{{ proto array lstat(string filename)
  937. Give information about a file or symbolic link */
  938. PharFileFunction(phar_lstat, FS_LSTAT, orig_lstat)
  939. /* }}} */
  940. /* {{{ proto array stat(string filename)
  941. Give information about a file */
  942. PharFileFunction(phar_stat, FS_STAT, orig_stat)
  943. /* }}} */
  944. /* {{{ void phar_intercept_functions(TSRMLS_D) */
  945. void phar_intercept_functions(TSRMLS_D)
  946. {
  947. if (!PHAR_G(request_init)) {
  948. PHAR_G(cwd) = NULL;
  949. PHAR_G(cwd_len) = 0;
  950. }
  951. PHAR_G(intercepted) = 1;
  952. }
  953. /* }}} */
  954. /* {{{ void phar_release_functions(TSRMLS_D) */
  955. void phar_release_functions(TSRMLS_D)
  956. {
  957. PHAR_G(intercepted) = 0;
  958. }
  959. /* }}} */
  960. /* {{{ void phar_intercept_functions_init(TSRMLS_D) */
  961. #define PHAR_INTERCEPT(func) \
  962. PHAR_G(orig_##func) = NULL; \
  963. if (SUCCESS == zend_hash_find(CG(function_table), #func, sizeof(#func), (void **)&orig)) { \
  964. PHAR_G(orig_##func) = orig->internal_function.handler; \
  965. orig->internal_function.handler = phar_##func; \
  966. }
  967. void phar_intercept_functions_init(TSRMLS_D)
  968. {
  969. zend_function *orig;
  970. PHAR_INTERCEPT(fopen);
  971. PHAR_INTERCEPT(file_get_contents);
  972. PHAR_INTERCEPT(is_file);
  973. PHAR_INTERCEPT(is_link);
  974. PHAR_INTERCEPT(is_dir);
  975. PHAR_INTERCEPT(opendir);
  976. PHAR_INTERCEPT(file_exists);
  977. PHAR_INTERCEPT(fileperms);
  978. PHAR_INTERCEPT(fileinode);
  979. PHAR_INTERCEPT(filesize);
  980. PHAR_INTERCEPT(fileowner);
  981. PHAR_INTERCEPT(filegroup);
  982. PHAR_INTERCEPT(fileatime);
  983. PHAR_INTERCEPT(filemtime);
  984. PHAR_INTERCEPT(filectime);
  985. PHAR_INTERCEPT(filetype);
  986. PHAR_INTERCEPT(is_writable);
  987. PHAR_INTERCEPT(is_readable);
  988. PHAR_INTERCEPT(is_executable);
  989. PHAR_INTERCEPT(lstat);
  990. PHAR_INTERCEPT(stat);
  991. PHAR_INTERCEPT(readfile);
  992. PHAR_G(intercepted) = 0;
  993. }
  994. /* }}} */
  995. /* {{{ void phar_intercept_functions_shutdown(TSRMLS_D) */
  996. #define PHAR_RELEASE(func) \
  997. if (PHAR_G(orig_##func) && SUCCESS == zend_hash_find(CG(function_table), #func, sizeof(#func), (void **)&orig)) { \
  998. orig->internal_function.handler = PHAR_G(orig_##func); \
  999. } \
  1000. PHAR_G(orig_##func) = NULL;
  1001. void phar_intercept_functions_shutdown(TSRMLS_D)
  1002. {
  1003. zend_function *orig;
  1004. PHAR_RELEASE(fopen);
  1005. PHAR_RELEASE(file_get_contents);
  1006. PHAR_RELEASE(is_file);
  1007. PHAR_RELEASE(is_dir);
  1008. PHAR_RELEASE(opendir);
  1009. PHAR_RELEASE(file_exists);
  1010. PHAR_RELEASE(fileperms);
  1011. PHAR_RELEASE(fileinode);
  1012. PHAR_RELEASE(filesize);
  1013. PHAR_RELEASE(fileowner);
  1014. PHAR_RELEASE(filegroup);
  1015. PHAR_RELEASE(fileatime);
  1016. PHAR_RELEASE(filemtime);
  1017. PHAR_RELEASE(filectime);
  1018. PHAR_RELEASE(filetype);
  1019. PHAR_RELEASE(is_writable);
  1020. PHAR_RELEASE(is_readable);
  1021. PHAR_RELEASE(is_executable);
  1022. PHAR_RELEASE(lstat);
  1023. PHAR_RELEASE(stat);
  1024. PHAR_RELEASE(readfile);
  1025. PHAR_G(intercepted) = 0;
  1026. }
  1027. /* }}} */
  1028. static struct _phar_orig_functions {
  1029. void (*orig_fopen)(INTERNAL_FUNCTION_PARAMETERS);
  1030. void (*orig_file_get_contents)(INTERNAL_FUNCTION_PARAMETERS);
  1031. void (*orig_is_file)(INTERNAL_FUNCTION_PARAMETERS);
  1032. void (*orig_is_link)(INTERNAL_FUNCTION_PARAMETERS);
  1033. void (*orig_is_dir)(INTERNAL_FUNCTION_PARAMETERS);
  1034. void (*orig_opendir)(INTERNAL_FUNCTION_PARAMETERS);
  1035. void (*orig_file_exists)(INTERNAL_FUNCTION_PARAMETERS);
  1036. void (*orig_fileperms)(INTERNAL_FUNCTION_PARAMETERS);
  1037. void (*orig_fileinode)(INTERNAL_FUNCTION_PARAMETERS);
  1038. void (*orig_filesize)(INTERNAL_FUNCTION_PARAMETERS);
  1039. void (*orig_fileowner)(INTERNAL_FUNCTION_PARAMETERS);
  1040. void (*orig_filegroup)(INTERNAL_FUNCTION_PARAMETERS);
  1041. void (*orig_fileatime)(INTERNAL_FUNCTION_PARAMETERS);
  1042. void (*orig_filemtime)(INTERNAL_FUNCTION_PARAMETERS);
  1043. void (*orig_filectime)(INTERNAL_FUNCTION_PARAMETERS);
  1044. void (*orig_filetype)(INTERNAL_FUNCTION_PARAMETERS);
  1045. void (*orig_is_writable)(INTERNAL_FUNCTION_PARAMETERS);
  1046. void (*orig_is_readable)(INTERNAL_FUNCTION_PARAMETERS);
  1047. void (*orig_is_executable)(INTERNAL_FUNCTION_PARAMETERS);
  1048. void (*orig_lstat)(INTERNAL_FUNCTION_PARAMETERS);
  1049. void (*orig_readfile)(INTERNAL_FUNCTION_PARAMETERS);
  1050. void (*orig_stat)(INTERNAL_FUNCTION_PARAMETERS);
  1051. } phar_orig_functions = {NULL};
  1052. void phar_save_orig_functions(TSRMLS_D) /* {{{ */
  1053. {
  1054. phar_orig_functions.orig_fopen = PHAR_G(orig_fopen);
  1055. phar_orig_functions.orig_file_get_contents = PHAR_G(orig_file_get_contents);
  1056. phar_orig_functions.orig_is_file = PHAR_G(orig_is_file);
  1057. phar_orig_functions.orig_is_link = PHAR_G(orig_is_link);
  1058. phar_orig_functions.orig_is_dir = PHAR_G(orig_is_dir);
  1059. phar_orig_functions.orig_opendir = PHAR_G(orig_opendir);
  1060. phar_orig_functions.orig_file_exists = PHAR_G(orig_file_exists);
  1061. phar_orig_functions.orig_fileperms = PHAR_G(orig_fileperms);
  1062. phar_orig_functions.orig_fileinode = PHAR_G(orig_fileinode);
  1063. phar_orig_functions.orig_filesize = PHAR_G(orig_filesize);
  1064. phar_orig_functions.orig_fileowner = PHAR_G(orig_fileowner);
  1065. phar_orig_functions.orig_filegroup = PHAR_G(orig_filegroup);
  1066. phar_orig_functions.orig_fileatime = PHAR_G(orig_fileatime);
  1067. phar_orig_functions.orig_filemtime = PHAR_G(orig_filemtime);
  1068. phar_orig_functions.orig_filectime = PHAR_G(orig_filectime);
  1069. phar_orig_functions.orig_filetype = PHAR_G(orig_filetype);
  1070. phar_orig_functions.orig_is_writable = PHAR_G(orig_is_writable);
  1071. phar_orig_functions.orig_is_readable = PHAR_G(orig_is_readable);
  1072. phar_orig_functions.orig_is_executable = PHAR_G(orig_is_executable);
  1073. phar_orig_functions.orig_lstat = PHAR_G(orig_lstat);
  1074. phar_orig_functions.orig_readfile = PHAR_G(orig_readfile);
  1075. phar_orig_functions.orig_stat = PHAR_G(orig_stat);
  1076. }
  1077. /* }}} */
  1078. void phar_restore_orig_functions(TSRMLS_D) /* {{{ */
  1079. {
  1080. PHAR_G(orig_fopen) = phar_orig_functions.orig_fopen;
  1081. PHAR_G(orig_file_get_contents) = phar_orig_functions.orig_file_get_contents;
  1082. PHAR_G(orig_is_file) = phar_orig_functions.orig_is_file;
  1083. PHAR_G(orig_is_link) = phar_orig_functions.orig_is_link;
  1084. PHAR_G(orig_is_dir) = phar_orig_functions.orig_is_dir;
  1085. PHAR_G(orig_opendir) = phar_orig_functions.orig_opendir;
  1086. PHAR_G(orig_file_exists) = phar_orig_functions.orig_file_exists;
  1087. PHAR_G(orig_fileperms) = phar_orig_functions.orig_fileperms;
  1088. PHAR_G(orig_fileinode) = phar_orig_functions.orig_fileinode;
  1089. PHAR_G(orig_filesize) = phar_orig_functions.orig_filesize;
  1090. PHAR_G(orig_fileowner) = phar_orig_functions.orig_fileowner;
  1091. PHAR_G(orig_filegroup) = phar_orig_functions.orig_filegroup;
  1092. PHAR_G(orig_fileatime) = phar_orig_functions.orig_fileatime;
  1093. PHAR_G(orig_filemtime) = phar_orig_functions.orig_filemtime;
  1094. PHAR_G(orig_filectime) = phar_orig_functions.orig_filectime;
  1095. PHAR_G(orig_filetype) = phar_orig_functions.orig_filetype;
  1096. PHAR_G(orig_is_writable) = phar_orig_functions.orig_is_writable;
  1097. PHAR_G(orig_is_readable) = phar_orig_functions.orig_is_readable;
  1098. PHAR_G(orig_is_executable) = phar_orig_functions.orig_is_executable;
  1099. PHAR_G(orig_lstat) = phar_orig_functions.orig_lstat;
  1100. PHAR_G(orig_readfile) = phar_orig_functions.orig_readfile;
  1101. PHAR_G(orig_stat) = phar_orig_functions.orig_stat;
  1102. }
  1103. /* }}} */
  1104. /*
  1105. * Local variables:
  1106. * tab-width: 4
  1107. * c-basic-offset: 4
  1108. * End:
  1109. * vim600: noet sw=4 ts=4 fdm=marker
  1110. * vim<600: noet sw=4 ts=4
  1111. */