mirror of
https://github.com/php/php-src.git
synced 2026-04-12 18:43:37 +02:00
Squashed commit of the following: commit 106c815fffb8eb3efe00a27a5229cb1f8ffc9736 Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Nov 14 16:36:44 2018 +0300 Added NEWS entry commit 1dacd5e20b7043368ef9e80db296d1781134b6fd Merge: d516139abfba99aa133cAuthor: Dmitry Stogov <dmitry@zend.com> Date: Wed Nov 14 16:33:37 2018 +0300 Merge branch 'master' into preload * master: Fixed issues related to optimization and persitence of classes linked with interfaces, traits or internal classes. Added possiblity to avoid signal handlers reinitialization on each request. commit d516139abf5ffbd495ee6037f1dc04a1cfe588a7 Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Nov 14 16:13:15 2018 +0300 Override opcache.preload for testing commit 162b154d0bbfbaf8ef93975f7e56a1353236903d Merge:45fdd034ce8bda22592eAuthor: Dmitry Stogov <dmitry@zend.com> Date: Wed Nov 14 15:38:09 2018 +0300 Merge branch 'master' into preload * master: (34 commits) Eliminate useless $this related check Eliminate useless $this related checks Replace zend_parse_method_parameters() by zend_parse_parameters() and avoid useless checks. Replace getThis() by EX(This), when additional check is not necessary. Fixed tests Validate length on socket_write Fix compilation on x32 Fix #77141: Signedness issue in SOAP when precision=-1 Support SQLite3 @name notation Remove lexer files generated by RE2C Update libmagic.patch [ci skip] Update libmagic.patch [ci skip] Fork test with pcre.jit=0 Rework magic data Fix regex Fix regex Rework magic data Sync one more upstream libmagic piece Suppress already used warning Ignore getaddrinfo failed message ... commit45fdd034ceAuthor: Dmitry Stogov <dmitry@zend.com> Date: Fri Nov 9 13:07:03 2018 +0300 Properly resolve magic method of preloaded classes inherited from internal ones. commit34645aeb42Author: Dmitry Stogov <dmitry@zend.com> Date: Thu Nov 8 15:29:17 2018 +0300 Don't preload constants defined during preload script excution. commitcef0d67c3eAuthor: Dmitry Stogov <dmitry@zend.com> Date: Wed Nov 7 15:56:54 2018 +0300 Support for class aliasses commit08ffc9a552Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Nov 7 15:34:39 2018 +0300 Resolve constants only in linked classes commit8d3429cda8Author: Dmitry Stogov <dmitry@zend.com> Date: Tue Nov 6 11:56:39 2018 +0300 Fixed preloading of references to internal classes. commit7ae3a47d20Merge:9b0a53ed1c049f239cfcAuthor: Dmitry Stogov <dmitry@zend.com> Date: Tue Nov 6 11:37:15 2018 +0300 Merge branch 'master' into preload * master: Update NEWS [ci skip] Update NEWS [ci skip] Update libmagic.patch [ci skip] Update libmagic.patch [ci skip] Declare function proto in header Declare function proto in header Fix #76825: Undefined symbols ___cpuid_count NEWS Fix: #77110 undefined symbol zend_string_equal_val in C++ build Fix #77105: Use position:sticky for <th> in `phpinfo()` Implement handling for JIT recognition when cross compiling Backport7f5f4601for 7.2 Fix #76348: WSDL_CACHE_MEMORY causes Segmentation fault Rework places in libmagic regarding previous CVE-2014-3538 fixes Change the way JIT availability is checked Fix a test for ldap extension Fixed bug #77092 Future-proof email addresses commit9b0a53ed1cAuthor: Dmitry Stogov <dmitry@zend.com> Date: Fri Nov 2 14:54:44 2018 +0300 We don't need preload_restart() here commit0bd17bd438Author: Dmitry Stogov <dmitry@zend.com> Date: Fri Nov 2 14:44:30 2018 +0300 EG(*) may be not initializd at this point - use CG(*). commitb610467051Merge:3a9d90f74a67e0138c0dAuthor: Dmitry Stogov <dmitry@zend.com> Date: Fri Nov 2 11:33:37 2018 +0300 Merge branch 'master' into preload * master: Future-proof email addresses... Update email addresses. We're still @Zend, but future proofing it... commit3a9d90f74aAuthor: Dmitry Stogov <dmitry@zend.com> Date: Thu Nov 1 15:19:48 2018 +0300 Fexed resolution of method clones commitaea85c65bdAuthor: Dmitry Stogov <dmitry@zend.com> Date: Thu Nov 1 11:45:50 2018 +0300 Prevent inlining of method copied from trait commit36b644fbb7Merge:7a20781d2eb91690c892Author: Dmitry Stogov <dmitry@zend.com> Date: Thu Nov 1 10:56:02 2018 +0300 Merge branch 'master' into preload * master: Fix stray newline that caused this test to fail Fix session tests that fail if error_log is set This test needs to log to stdout Fix error condition Fixed bug #77081 ftruncate() changes seek pointer in c mode Fix and improve test case commit7a20781d2eAuthor: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 31 00:52:46 2018 +0300 Added test commit4a57b5d563Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 31 00:50:21 2018 +0300 Fixed preloading of classes linked with traits commit68c4f99e23Author: Dmitry Stogov <dmitry@zend.com> Date: Tue Oct 30 16:25:14 2018 +0300 Added test commit38ab7ef4cfMerge:eb6e2c529fbf38e6c10aAuthor: Dmitry Stogov <dmitry@zend.com> Date: Tue Oct 30 16:14:39 2018 +0300 Merge branch 'master' into preload * master: Keep original value of "prototype" commiteb6e2c529fMerge:562049510f2fefa8c61eAuthor: Dmitry Stogov <dmitry@zend.com> Date: Tue Oct 30 15:35:39 2018 +0300 Merge branch 'master' into preload * master: Call function_add_ref() in proper place Updated to version 2018.7 (2018g) Updated to version 2018.7 (2018g) Updated to version 2018.7 (2018g) Reslove inherited op_array references once afrer all optimizations. commit562049510fMerge:e806cb732a4828fb7b6bAuthor: Dmitry Stogov <dmitry@zend.com> Date: Tue Oct 30 10:29:49 2018 +0300 Merge branch 'master' into preload * master: [ci skip] Update NEWS [ci skip] Update NEWS [ci skip] Update NEWS fix bug #77079 Add missing null initialization Remove redundant mbfl_string_init calls Use zend_string for mbstring last encoding cache commite806cb732aAuthor: Dmitry Stogov <dmitry@zend.com> Date: Mon Oct 29 22:32:15 2018 +0300 Fixed double-free commit2f697ef8afAuthor: Dmitry Stogov <dmitry@zend.com> Date: Mon Oct 29 22:07:32 2018 +0300 typo commitc559f22b3eMerge:310631cc05ea2e67876aAuthor: Dmitry Stogov <dmitry@zend.com> Date: Mon Oct 29 21:59:27 2018 +0300 Merge branch 'master' into preload * master: Stop Apache if PHP wasn't started successful. Execute zend_post_startup() with module_initialized flag set. Removed dead code Fix mb_strrpos() with encoding passed as 3rd param commit310631cc05Author: Dmitry Stogov <dmitry@zend.com> Date: Mon Oct 29 16:48:42 2018 +0300 Stop Apache if PHP wasn't started successful. commit0a24d7ba8fAuthor: Dmitry Stogov <dmitry@zend.com> Date: Mon Oct 29 16:25:49 2018 +0300 Avoid use-after-free in main thread commit17a3cb4a2aAuthor: Dmitry Stogov <dmitry@zend.com> Date: Mon Oct 29 15:25:17 2018 +0300 Execute zend_post_startup() with module_initialized flag set. commit6d4b22c518Author: Dmitry Stogov <dmitry@zend.com> Date: Mon Oct 29 14:12:20 2018 +0300 Override SAPI.ub_write and SAPI.flush for preloading commit386c9d3470Merge:d7fbb4d402359f19edc9Author: Dmitry Stogov <dmitry@zend.com> Date: Mon Oct 29 13:49:24 2018 +0300 Merge branch 'master' into preload * master: Optimize substr() edge-case conditions [ci skip] Update UPGRADING Fix #71592: External entity processing never fails Add TIDY_TAG_* constants supported by libtidy 5 Add is_iterable to opcache Optimizer commitd7fbb4d402Author: Dmitry Stogov <dmitry@zend.com> Date: Fri Oct 26 13:11:54 2018 +0300 Restore preload state if it was already loaded in another process. commit0fe9ea1c07Author: Dmitry Stogov <dmitry@zend.com> Date: Fri Oct 26 12:29:06 2018 +0300 Removed dead code commit3a2d1bcc1fAuthor: Dmitry Stogov <dmitry@zend.com> Date: Fri Oct 26 00:19:40 2018 +0300 Support for builds without ZEND_SIGNALS commite6b76ecb4bMerge:4531fbf93168694c9997Author: Dmitry Stogov <dmitry@zend.com> Date: Thu Oct 25 23:43:25 2018 +0300 Merge branch 'master' into preload * master: Don't wrap php_module_shutdown() with zend_try. executor_globals are released in ZTS build, and this leads to crash. [ci skip] Fix indentation in UPGRADING. commit4531fbf931Author: Dmitry Stogov <dmitry@zend.com> Date: Thu Oct 25 22:44:49 2018 +0300 Disable linking and preloading of classes those parent or one of interface or trait is an internal class. commita594a618ceAuthor: Dmitry Stogov <dmitry@zend.com> Date: Thu Oct 25 22:30:51 2018 +0300 Cleanup - remove useless ZCSG(saved_map_ptr_last) - move preloaded classes/functions clean-up code back into better place commitab9a40f63cAuthor: Dmitry Stogov <dmitry@zend.com> Date: Thu Oct 25 20:52:51 2018 +0300 Added support for preloaded classes/functions in ZTS build commite3c65db099Merge:4f57c1e02933e777acbfAuthor: Dmitry Stogov <dmitry@zend.com> Date: Thu Oct 25 20:52:26 2018 +0300 Merge branch 'master' into preload * master: Improved shared interned strings handling. The previous implementation worked incorrectly in ZTS build. It changed strings only in function/class tables of one thread. Now all threads gets the same shared interned strings. Also, on shutdown, we don't try to replace SHM interned strings back to process strings, but delay dettachment of SHM instead. Don't use request heap at shutdown Don't optimize function if inference failed Fixed bug #77058 Improve "narrowing" error message bump versions commit4f57c1e029Author: Dmitry Stogov <dmitry@zend.com> Date: Thu Oct 25 15:29:58 2018 +0300 Cleanup (move preload_shutdown() call to better place) commit26587a95c0Author: Dmitry Stogov <dmitry@zend.com> Date: Thu Oct 25 14:30:51 2018 +0300 eol commitd70cb10480Author: Dmitry Stogov <dmitry@zend.com> Date: Thu Oct 25 14:30:20 2018 +0300 cleanup commitaabe685dbbMerge:d9fc51bc3b40808ac41eAuthor: Dmitry Stogov <dmitry@zend.com> Date: Thu Oct 25 12:42:51 2018 +0300 Merge branch 'master' into preload * master: Remove unused var Remove ext/json parser files generated by bison Fix run-tests.php for running phpdbg and certain test sections Normalize .gitignore commitd9fc51bc3bMerge:b5ffba0fafb6ef8998d5Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 24 15:59:24 2018 +0300 Merge branch 'master' into preload * master: Fixed reseting of interned strings buffer. commitb5ffba0fafMerge:e4a7ef0c43a404383118Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 24 12:46:28 2018 +0300 Merge branch 'master' into preload * master: Fixed build in directory different from source commite4a7ef0c43Merge:811f20aaa5d1e14e2cc0Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 24 11:59:43 2018 +0300 Merge branch 'master' into preload * master: (29 commits) Make php_plain_files_wrapper to be writable (workaround for swoole) Remove phpdbg parser files generated by bison Fix conflicts in phpdbg parser Refetetch function name on exceptional path to allow better code on fast code path. fix typo in sysvsem.c Fixed bug #50675 bump to 7.2.13-dev [ci skip] Update NEWS wrt. php-7.3.0RC4 tagging Inlining in the most frequently used code paths Fixed test failurs introduced by9c144e0d82Use persistent strings only for persistent connections Fix accessibility checks for dynamic properties Updated to version 2018.6 (2018f) Updated to version 2018.6 (2018f) Updated to version 2018.6 (2018f) Fix arginfo and clean up fpm_get_status Defragment two Zend related Makefile fragments together [ci skip] Remove automake and aclocal in comments Fix #75282: xmlrpc_encode_request() crashes Fix tests for ICU 63.1 ... commit811f20aaa5Author: Dmitry Stogov <dmitry@zend.com> Date: Mon Oct 22 14:10:49 2018 +0300 Added information about preloading to opcache_get_status() commit093e8b1bbfAuthor: Dmitry Stogov <dmitry@zend.com> Date: Fri Oct 19 13:46:23 2018 +0300 Added warning message commita2ba970ce3Author: Dmitry Stogov <dmitry@zend.com> Date: Fri Oct 19 13:35:40 2018 +0300 Added test commitb67e28367cAuthor: Dmitry Stogov <dmitry@zend.com> Date: Fri Oct 19 13:33:12 2018 +0300 Don't preload functions declared at run-time. commitb0139dc228Merge:a609520adb3fe698b904Author: Dmitry Stogov <dmitry@zend.com> Date: Fri Oct 19 13:23:14 2018 +0300 Merge branch 'master' into preload * master: Mark "top-level" functions. Don't initialize static_member_tables during start-up, when inherit internal classes. [ci skip] Update NEWS [ci skip] Update NEWS [ci skip] Update NEWS Fix #77035: The phpize and ./configure create redundant .deps file Remove outdated PEAR artefacts Fix tests/output/bug74815.phpt generating errors.log Revert "Use C++ symbols, when C++11 or upper is compiled" Use C++ symbols, when C++11 or upper is compiled Added new line Remove stamp-h Move all testing docs to qa.php.net Fix a typo in UPGRADING.INTERNALS Fix test when it's run on another drive [ci skip] Update UPGRADING wrt. tidyp support Fixed incorrect reallocation Fix #77027: tidy::getOptDoc() not available on Windows Run CI tests under opcache.protect_memory=1 commita609520adbMerge:ac8f45f61bb6ac50f9e6Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 17:01:05 2018 +0300 Merge branch 'master' into preload * master: Fixed comment Micro optimizations Mark "top-level" classes commitac8f45f61bMerge:632b30b545d57cd36e47Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 15:53:41 2018 +0300 Merge branch 'master' into preload * master: Immutable clases and op_arrays. commit632b30b545Merge:d33908a99acd0c36c3f9Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 15:04:43 2018 +0300 Merge branch 'immutable' into preload * immutable: Remove the "auto" encoding Fixed bug #77025 Add vtbls for EUC-TW encoding commitcd0c36c3f9Merge:4740dabb84ad6738e886Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 14:43:38 2018 +0300 Merge branch 'master' into immutable * master: Remove the "auto" encoding Fixed bug #77025 Add vtbls for EUC-TW encoding commitd33908a99aMerge:21e0bebca34740dabb84Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 14:14:23 2018 +0300 Merge branch 'immutable' into preload * immutable: Reverted back ce->iterator_funcs_ptr. Initialize ce->iterator_funcs_ptr fields in immutable classes. commit4740dabb84Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 14:12:28 2018 +0300 Reverted back ce->iterator_funcs_ptr. Initialize ce->iterator_funcs_ptr fields in immutable classes. commit21e0bebca3Merge:c78277ae84ad7a78b253Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 12:29:59 2018 +0300 Merge branch 'immutable' into preload * immutable: Added comment Added type cast Moved static class members initialization into the proper place. Removed redundand assertion Removed duplicate code Hide offset encoding magic in ZEND_MAP_PTR_IS_OFFSET(), ZEND_MAP_PTR_OFFSET2PTR() and ZEND_MAP_PTR_PTR2OFFSET() macros. typo Remove unused variable makefile_am_files Classify object handlers are required/optional Add support for getting SKIP_TAGSTART and SKIP_WHITE options Remove some obsolete config_vars.mk occurrences Remove bsd_converted from .gitignore Remove configuration parser and scanners ignores Remove obsolete buildconf.stamp from .gitignore [ci skip] Add magicdata.patch exception to .gitignore Remove outdated ext/spl/examples items from .gitignore Remove unused test.inc in ext/iconv/tests commitad7a78b253Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 11:46:30 2018 +0300 Added comment commit0276ea5187Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 11:42:43 2018 +0300 Added type cast commitc63fc5d5f1Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 11:36:51 2018 +0300 Moved static class members initialization into the proper place. commitb945548e93Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 11:21:03 2018 +0300 Removed redundand assertion commitd5a4108840Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 11:19:13 2018 +0300 Removed duplicate code commit8dadca8864Author: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 11:05:43 2018 +0300 Hide offset encoding magic in ZEND_MAP_PTR_IS_OFFSET(), ZEND_MAP_PTR_OFFSET2PTR() and ZEND_MAP_PTR_PTR2OFFSET() macros. commit9ef07c88bdAuthor: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 10:48:29 2018 +0300 typo commita06f0f3d3aMerge:94099586ec3412345ffeAuthor: Dmitry Stogov <dmitry@zend.com> Date: Wed Oct 17 10:47:07 2018 +0300 Merge branch 'master' into immutable * master: Remove unused variable makefile_am_files Classify object handlers are required/optional Add support for getting SKIP_TAGSTART and SKIP_WHITE options Remove some obsolete config_vars.mk occurrences Remove bsd_converted from .gitignore Remove configuration parser and scanners ignores Remove obsolete buildconf.stamp from .gitignore [ci skip] Add magicdata.patch exception to .gitignore Remove outdated ext/spl/examples items from .gitignore Remove unused test.inc in ext/iconv/tests commitc78277ae84Author: Dmitry Stogov <dmitry@zend.com> Date: Tue Oct 16 17:25:35 2018 +0300 Preloadsing support for opcache restart commitf76a955c02Author: Dmitry Stogov <dmitry@zend.com> Date: Tue Oct 16 13:52:36 2018 +0300 Fixed incorrect signal handlers overriding commit0810ce0d81Author: Dmitry Stogov <dmitry@zend.com> Date: Mon Oct 15 23:38:48 2018 +0300 An attempt to implemnt "preloading" ability. commit94099586ecAuthor: Dmitry Stogov <dmitry@zend.com> Date: Mon Oct 15 23:34:01 2018 +0300 Immutable clases and op_arrays
3999 lines
117 KiB
C
3999 lines
117 KiB
C
/*
|
|
+----------------------------------------------------------------------+
|
|
| Zend OPcache |
|
|
+----------------------------------------------------------------------+
|
|
| Copyright (c) 1998-2018 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: |
|
|
| http://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: Andi Gutmans <andi@php.net> |
|
|
| Zeev Suraski <zeev@php.net> |
|
|
| Stanislav Malyshev <stas@zend.com> |
|
|
| Dmitry Stogov <dmitry@php.net> |
|
|
+----------------------------------------------------------------------+
|
|
*/
|
|
|
|
#include "main/php.h"
|
|
#include "main/php_globals.h"
|
|
#include "zend.h"
|
|
#include "zend_extensions.h"
|
|
#include "zend_compile.h"
|
|
#include "ZendAccelerator.h"
|
|
#include "zend_persist.h"
|
|
#include "zend_shared_alloc.h"
|
|
#include "zend_accelerator_module.h"
|
|
#include "zend_accelerator_blacklist.h"
|
|
#include "zend_list.h"
|
|
#include "zend_execute.h"
|
|
#include "zend_inheritance.h"
|
|
#include "main/php_main.h"
|
|
#include "main/SAPI.h"
|
|
#include "main/php_streams.h"
|
|
#include "main/php_open_temporary_file.h"
|
|
#include "zend_API.h"
|
|
#include "zend_ini.h"
|
|
#include "zend_virtual_cwd.h"
|
|
#include "zend_accelerator_util_funcs.h"
|
|
#include "zend_accelerator_hash.h"
|
|
#include "ext/pcre/php_pcre.h"
|
|
#include "ext/standard/md5.h"
|
|
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
# include "zend_file_cache.h"
|
|
#endif
|
|
|
|
#ifndef ZEND_WIN32
|
|
#include <netdb.h>
|
|
#endif
|
|
|
|
#ifdef ZEND_WIN32
|
|
typedef int uid_t;
|
|
typedef int gid_t;
|
|
#include <io.h>
|
|
#endif
|
|
|
|
#ifndef ZEND_WIN32
|
|
# include <sys/time.h>
|
|
#else
|
|
# include <process.h>
|
|
#endif
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
# include <unistd.h>
|
|
#endif
|
|
#include <fcntl.h>
|
|
#include <signal.h>
|
|
#include <time.h>
|
|
|
|
#ifndef ZEND_WIN32
|
|
# include <sys/types.h>
|
|
# include <sys/ipc.h>
|
|
#endif
|
|
|
|
#include <sys/stat.h>
|
|
#include <errno.h>
|
|
|
|
#ifdef __AVX__
|
|
#include <immintrin.h>
|
|
#endif
|
|
|
|
#define SHM_PROTECT() \
|
|
do { \
|
|
if (ZCG(accel_directives).protect_memory) { \
|
|
zend_accel_shared_protect(1); \
|
|
} \
|
|
} while (0)
|
|
#define SHM_UNPROTECT() \
|
|
do { \
|
|
if (ZCG(accel_directives).protect_memory) { \
|
|
zend_accel_shared_protect(0); \
|
|
} \
|
|
} while (0)
|
|
|
|
ZEND_EXTENSION();
|
|
|
|
#ifndef ZTS
|
|
zend_accel_globals accel_globals;
|
|
#else
|
|
int accel_globals_id;
|
|
#if defined(COMPILE_DL_OPCACHE)
|
|
ZEND_TSRMLS_CACHE_DEFINE()
|
|
#endif
|
|
#endif
|
|
|
|
/* Points to the structure shared across all PHP processes */
|
|
zend_accel_shared_globals *accel_shared_globals = NULL;
|
|
|
|
/* true globals, no need for thread safety */
|
|
zend_bool accel_startup_ok = 0;
|
|
static char *zps_failure_reason = NULL;
|
|
char *zps_api_failure_reason = NULL;
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
zend_bool file_cache_only = 0; /* process uses file cache only */
|
|
#endif
|
|
#if ENABLE_FILE_CACHE_FALLBACK
|
|
zend_bool fallback_process = 0; /* process uses file cache fallback */
|
|
#endif
|
|
|
|
static zend_op_array *(*accelerator_orig_compile_file)(zend_file_handle *file_handle, int type);
|
|
static int (*accelerator_orig_zend_stream_open_function)(const char *filename, zend_file_handle *handle );
|
|
static zend_string *(*accelerator_orig_zend_resolve_path)(const char *filename, size_t filename_len);
|
|
static zif_handler orig_chdir = NULL;
|
|
static ZEND_INI_MH((*orig_include_path_on_modify)) = NULL;
|
|
static int (*orig_post_startup_cb)(void);
|
|
|
|
static void accel_gen_system_id(void);
|
|
static int accel_post_startup(void);
|
|
static int accel_finish_startup(void);
|
|
|
|
static void preload_shutdown(void);
|
|
static void preload_activate(void);
|
|
static void preload_restart(void);
|
|
|
|
#ifdef ZEND_WIN32
|
|
# define INCREMENT(v) InterlockedIncrement64(&ZCSG(v))
|
|
# define DECREMENT(v) InterlockedDecrement64(&ZCSG(v))
|
|
# define LOCKVAL(v) (ZCSG(v))
|
|
#endif
|
|
|
|
#ifdef ZEND_WIN32
|
|
static time_t zend_accel_get_time(void)
|
|
{
|
|
FILETIME now;
|
|
GetSystemTimeAsFileTime(&now);
|
|
|
|
return (time_t) ((((((__int64)now.dwHighDateTime) << 32)|now.dwLowDateTime) - 116444736000000000L)/10000000);
|
|
}
|
|
#else
|
|
# define zend_accel_get_time() time(NULL)
|
|
#endif
|
|
|
|
static inline int is_stream_path(const char *filename)
|
|
{
|
|
const char *p;
|
|
|
|
for (p = filename;
|
|
(*p >= 'a' && *p <= 'z') ||
|
|
(*p >= 'A' && *p <= 'Z') ||
|
|
(*p >= '0' && *p <= '9') ||
|
|
*p == '+' || *p == '-' || *p == '.';
|
|
p++);
|
|
return ((p != filename) && (p[0] == ':') && (p[1] == '/') && (p[2] == '/'));
|
|
}
|
|
|
|
static inline int is_cacheable_stream_path(const char *filename)
|
|
{
|
|
return memcmp(filename, "file://", sizeof("file://") - 1) == 0 ||
|
|
memcmp(filename, "phar://", sizeof("phar://") - 1) == 0;
|
|
}
|
|
|
|
/* O+ overrides PHP chdir() function and remembers the current working directory
|
|
* in ZCG(cwd) and ZCG(cwd_len). Later accel_getcwd() can use stored value and
|
|
* avoid getcwd() call.
|
|
*/
|
|
static ZEND_FUNCTION(accel_chdir)
|
|
{
|
|
char cwd[MAXPATHLEN];
|
|
|
|
orig_chdir(INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
if (VCWD_GETCWD(cwd, MAXPATHLEN)) {
|
|
if (ZCG(cwd)) {
|
|
zend_string_release_ex(ZCG(cwd), 0);
|
|
}
|
|
ZCG(cwd) = zend_string_init(cwd, strlen(cwd), 0);
|
|
} else {
|
|
if (ZCG(cwd)) {
|
|
zend_string_release_ex(ZCG(cwd), 0);
|
|
ZCG(cwd) = NULL;
|
|
}
|
|
}
|
|
ZCG(cwd_key_len) = 0;
|
|
ZCG(cwd_check) = 1;
|
|
}
|
|
|
|
static inline zend_string* accel_getcwd(void)
|
|
{
|
|
if (ZCG(cwd)) {
|
|
return ZCG(cwd);
|
|
} else {
|
|
char cwd[MAXPATHLEN + 1];
|
|
|
|
if (!VCWD_GETCWD(cwd, MAXPATHLEN)) {
|
|
return NULL;
|
|
}
|
|
ZCG(cwd) = zend_string_init(cwd, strlen(cwd), 0);
|
|
ZCG(cwd_key_len) = 0;
|
|
ZCG(cwd_check) = 1;
|
|
return ZCG(cwd);
|
|
}
|
|
}
|
|
|
|
void zend_accel_schedule_restart_if_necessary(zend_accel_restart_reason reason)
|
|
{
|
|
if ((((double) ZSMMG(wasted_shared_memory)) / ZCG(accel_directives).memory_consumption) >= ZCG(accel_directives).max_wasted_percentage) {
|
|
zend_accel_schedule_restart(reason);
|
|
}
|
|
}
|
|
|
|
/* O+ tracks changes of "include_path" directive. It stores all the requested
|
|
* values in ZCG(include_paths) shared hash table, current value in
|
|
* ZCG(include_path)/ZCG(include_path_len) and one letter "path key" in
|
|
* ZCG(include_path_key).
|
|
*/
|
|
static ZEND_INI_MH(accel_include_path_on_modify)
|
|
{
|
|
int ret = orig_include_path_on_modify(entry, new_value, mh_arg1, mh_arg2, mh_arg3, stage);
|
|
|
|
if (ret == SUCCESS) {
|
|
ZCG(include_path) = new_value;
|
|
ZCG(include_path_key_len) = 0;
|
|
ZCG(include_path_check) = 1;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static inline void accel_restart_enter(void)
|
|
{
|
|
#ifdef ZEND_WIN32
|
|
INCREMENT(restart_in);
|
|
#else
|
|
# ifdef _AIX
|
|
static FLOCK_STRUCTURE(restart_in_progress, F_WRLCK, SEEK_SET, 2, 1);
|
|
# else
|
|
static const FLOCK_STRUCTURE(restart_in_progress, F_WRLCK, SEEK_SET, 2, 1);
|
|
#endif
|
|
|
|
if (fcntl(lock_file, F_SETLK, &restart_in_progress) == -1) {
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "RestartC(+1): %s (%d)", strerror(errno), errno);
|
|
}
|
|
#endif
|
|
ZCSG(restart_in_progress) = 1;
|
|
}
|
|
|
|
static inline void accel_restart_leave(void)
|
|
{
|
|
#ifdef ZEND_WIN32
|
|
ZCSG(restart_in_progress) = 0;
|
|
DECREMENT(restart_in);
|
|
#else
|
|
# ifdef _AIX
|
|
static FLOCK_STRUCTURE(restart_finished, F_UNLCK, SEEK_SET, 2, 1);
|
|
# else
|
|
static const FLOCK_STRUCTURE(restart_finished, F_UNLCK, SEEK_SET, 2, 1);
|
|
# endif
|
|
|
|
ZCSG(restart_in_progress) = 0;
|
|
if (fcntl(lock_file, F_SETLK, &restart_finished) == -1) {
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "RestartC(-1): %s (%d)", strerror(errno), errno);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static inline int accel_restart_is_active(void)
|
|
{
|
|
if (ZCSG(restart_in_progress)) {
|
|
#ifndef ZEND_WIN32
|
|
FLOCK_STRUCTURE(restart_check, F_WRLCK, SEEK_SET, 2, 1);
|
|
|
|
if (fcntl(lock_file, F_GETLK, &restart_check) == -1) {
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "RestartC: %s (%d)", strerror(errno), errno);
|
|
return FAILURE;
|
|
}
|
|
if (restart_check.l_type == F_UNLCK) {
|
|
ZCSG(restart_in_progress) = 0;
|
|
return 0;
|
|
} else {
|
|
return 1;
|
|
}
|
|
#else
|
|
return LOCKVAL(restart_in) != 0;
|
|
#endif
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* Creates a read lock for SHM access */
|
|
static inline int accel_activate_add(void)
|
|
{
|
|
#ifdef ZEND_WIN32
|
|
INCREMENT(mem_usage);
|
|
#else
|
|
# ifdef _AIX
|
|
static FLOCK_STRUCTURE(mem_usage_lock, F_RDLCK, SEEK_SET, 1, 1);
|
|
# else
|
|
static const FLOCK_STRUCTURE(mem_usage_lock, F_RDLCK, SEEK_SET, 1, 1);
|
|
# endif
|
|
|
|
if (fcntl(lock_file, F_SETLK, &mem_usage_lock) == -1) {
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "UpdateC(+1): %s (%d)", strerror(errno), errno);
|
|
return FAILURE;
|
|
}
|
|
#endif
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* Releases a lock for SHM access */
|
|
static inline void accel_deactivate_sub(void)
|
|
{
|
|
#ifdef ZEND_WIN32
|
|
if (ZCG(counted)) {
|
|
DECREMENT(mem_usage);
|
|
ZCG(counted) = 0;
|
|
}
|
|
#else
|
|
# ifdef _AIX
|
|
static FLOCK_STRUCTURE(mem_usage_unlock, F_UNLCK, SEEK_SET, 1, 1);
|
|
# else
|
|
static const FLOCK_STRUCTURE(mem_usage_unlock, F_UNLCK, SEEK_SET, 1, 1);
|
|
# endif
|
|
|
|
if (fcntl(lock_file, F_SETLK, &mem_usage_unlock) == -1) {
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "UpdateC(-1): %s (%d)", strerror(errno), errno);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static inline void accel_unlock_all(void)
|
|
{
|
|
#ifdef ZEND_WIN32
|
|
accel_deactivate_sub();
|
|
#else
|
|
# ifdef _AIX
|
|
static FLOCK_STRUCTURE(mem_usage_unlock_all, F_UNLCK, SEEK_SET, 0, 0);
|
|
# else
|
|
static const FLOCK_STRUCTURE(mem_usage_unlock_all, F_UNLCK, SEEK_SET, 0, 0);
|
|
# endif
|
|
|
|
if (fcntl(lock_file, F_SETLK, &mem_usage_unlock_all) == -1) {
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "UnlockAll: %s (%d)", strerror(errno), errno);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
/* Interned strings support */
|
|
|
|
/* O+ disables creation of interned strings by regular PHP compiler, instead,
|
|
* it creates interned strings in shared memory when saves a script.
|
|
* Such interned strings are shared across all PHP processes
|
|
*/
|
|
|
|
#define STRTAB_INVALID_POS 0
|
|
|
|
#define STRTAB_HASH_TO_SLOT(tab, h) \
|
|
((uint32_t*)((char*)(tab) + sizeof(*(tab)) + ((h) & (tab)->nTableMask)))
|
|
#define STRTAB_STR_TO_POS(tab, s) \
|
|
((uint32_t)((char*)s - (char*)(tab)))
|
|
#define STRTAB_POS_TO_STR(tab, pos) \
|
|
((zend_string*)((char*)(tab) + (pos)))
|
|
#define STRTAB_COLLISION(s) \
|
|
(*((uint32_t*)((char*)s - sizeof(uint32_t))))
|
|
#define STRTAB_STR_SIZE(s) \
|
|
ZEND_MM_ALIGNED_SIZE_EX(_ZSTR_HEADER_SIZE + ZSTR_LEN(s) + 5, 8)
|
|
#define STRTAB_NEXT(s) \
|
|
((zend_string*)((char*)(s) + STRTAB_STR_SIZE(s)))
|
|
|
|
static void accel_interned_strings_restore_state(void)
|
|
{
|
|
zend_string *s, *top;
|
|
uint32_t *hash_slot, n;
|
|
|
|
/* clear removed content */
|
|
memset(ZCSG(interned_strings).saved_top,
|
|
0, (char*)ZCSG(interned_strings).top - (char*)ZCSG(interned_strings).saved_top);
|
|
|
|
/* Reset "top" */
|
|
ZCSG(interned_strings).top = ZCSG(interned_strings).saved_top;
|
|
|
|
/* rehash */
|
|
memset((char*)&ZCSG(interned_strings) + sizeof(zend_string_table),
|
|
STRTAB_INVALID_POS,
|
|
(char*)ZCSG(interned_strings).start -
|
|
((char*)&ZCSG(interned_strings) + sizeof(zend_string_table)));
|
|
s = ZCSG(interned_strings).start;
|
|
top = ZCSG(interned_strings).top;
|
|
n = 0;
|
|
if (EXPECTED(s < top)) {
|
|
do {
|
|
hash_slot = STRTAB_HASH_TO_SLOT(&ZCSG(interned_strings), ZSTR_H(s));
|
|
STRTAB_COLLISION(s) = *hash_slot;
|
|
*hash_slot = STRTAB_STR_TO_POS(&ZCSG(interned_strings), s);
|
|
s = STRTAB_NEXT(s);
|
|
n++;
|
|
} while (s < top);
|
|
}
|
|
ZCSG(interned_strings).nNumOfElements = n;
|
|
}
|
|
|
|
static void accel_interned_strings_save_state(void)
|
|
{
|
|
ZCSG(interned_strings).saved_top = ZCSG(interned_strings).top;
|
|
}
|
|
|
|
static zend_always_inline zend_string *accel_find_interned_string(zend_string *str)
|
|
{
|
|
/* for now interned strings are supported only for non-ZTS build */
|
|
zend_ulong h;
|
|
uint32_t pos;
|
|
zend_string *s;
|
|
|
|
if (IS_ACCEL_INTERNED(str)) {
|
|
/* this is already an interned string */
|
|
return str;
|
|
}
|
|
|
|
if (!ZCG(counted)) {
|
|
if (accel_activate_add() == FAILURE) {
|
|
return str;
|
|
}
|
|
ZCG(counted) = 1;
|
|
}
|
|
|
|
h = zend_string_hash_val(str);
|
|
|
|
/* check for existing interned string */
|
|
pos = *STRTAB_HASH_TO_SLOT(&ZCSG(interned_strings), h);
|
|
if (EXPECTED(pos != STRTAB_INVALID_POS)) {
|
|
do {
|
|
s = STRTAB_POS_TO_STR(&ZCSG(interned_strings), pos);
|
|
if (EXPECTED(ZSTR_H(s) == h) && zend_string_equal_content(s, str)) {
|
|
return s;
|
|
}
|
|
pos = STRTAB_COLLISION(s);
|
|
} while (pos != STRTAB_INVALID_POS);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
zend_string* ZEND_FASTCALL accel_new_interned_string(zend_string *str)
|
|
{
|
|
zend_ulong h;
|
|
uint32_t pos, *hash_slot;
|
|
zend_string *s;
|
|
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
if (UNEXPECTED(file_cache_only)) {
|
|
return str;
|
|
}
|
|
#endif
|
|
|
|
if (IS_ACCEL_INTERNED(str)) {
|
|
/* this is already an interned string */
|
|
return str;
|
|
}
|
|
|
|
h = zend_string_hash_val(str);
|
|
|
|
/* check for existing interned string */
|
|
hash_slot = STRTAB_HASH_TO_SLOT(&ZCSG(interned_strings), h);
|
|
pos = *hash_slot;
|
|
if (EXPECTED(pos != STRTAB_INVALID_POS)) {
|
|
do {
|
|
s = STRTAB_POS_TO_STR(&ZCSG(interned_strings), pos);
|
|
if (EXPECTED(ZSTR_H(s) == h) && zend_string_equal_content(s, str)) {
|
|
zend_string_release(str);
|
|
return s;
|
|
}
|
|
pos = STRTAB_COLLISION(s);
|
|
} while (pos != STRTAB_INVALID_POS);
|
|
}
|
|
|
|
if (UNEXPECTED(ZCSG(interned_strings).end - ZCSG(interned_strings).top < STRTAB_STR_SIZE(str))) {
|
|
/* no memory, return the same non-interned string */
|
|
zend_accel_error(ACCEL_LOG_WARNING, "Interned string buffer overflow");
|
|
return str;
|
|
}
|
|
|
|
/* create new interning string in shared interned strings buffer */
|
|
ZCSG(interned_strings).nNumOfElements++;
|
|
s = ZCSG(interned_strings).top;
|
|
hash_slot = STRTAB_HASH_TO_SLOT(&ZCSG(interned_strings), h);
|
|
STRTAB_COLLISION(s) = *hash_slot;
|
|
*hash_slot = STRTAB_STR_TO_POS(&ZCSG(interned_strings), s);
|
|
GC_SET_REFCOUNT(s, 1);
|
|
GC_TYPE_INFO(s) = IS_STRING | ((IS_STR_INTERNED | IS_STR_PERMANENT) << GC_FLAGS_SHIFT);
|
|
ZSTR_H(s) = h;
|
|
ZSTR_LEN(s) = ZSTR_LEN(str);
|
|
memcpy(ZSTR_VAL(s), ZSTR_VAL(str), ZSTR_LEN(s) + 1);
|
|
ZCSG(interned_strings).top = STRTAB_NEXT(s);
|
|
|
|
zend_string_release(str);
|
|
return s;
|
|
}
|
|
|
|
static zend_string* ZEND_FASTCALL accel_new_interned_string_for_php(zend_string *str)
|
|
{
|
|
zend_string_hash_val(str);
|
|
if (ZCG(counted)) {
|
|
zend_string *ret = accel_find_interned_string(str);
|
|
|
|
if (ret) {
|
|
zend_string_release(str);
|
|
return ret;
|
|
}
|
|
}
|
|
return str;
|
|
}
|
|
|
|
static zend_always_inline zend_string *accel_find_interned_string_ex(zend_ulong h, const char *str, size_t size)
|
|
{
|
|
uint32_t pos;
|
|
zend_string *s;
|
|
|
|
/* check for existing interned string */
|
|
pos = *STRTAB_HASH_TO_SLOT(&ZCSG(interned_strings), h);
|
|
if (EXPECTED(pos != STRTAB_INVALID_POS)) {
|
|
do {
|
|
s = STRTAB_POS_TO_STR(&ZCSG(interned_strings), pos);
|
|
if (EXPECTED(ZSTR_H(s) == h) && EXPECTED(ZSTR_LEN(s) == size)) {
|
|
if (!memcmp(ZSTR_VAL(s), str, size)) {
|
|
return s;
|
|
}
|
|
}
|
|
pos = STRTAB_COLLISION(s);
|
|
} while (pos != STRTAB_INVALID_POS);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static zend_string* ZEND_FASTCALL accel_init_interned_string_for_php(const char *str, size_t size, int permanent)
|
|
{
|
|
if (ZCG(counted)) {
|
|
zend_ulong h = zend_inline_hash_func(str, size);
|
|
zend_string *ret = accel_find_interned_string_ex(h, str, size);
|
|
|
|
if (!ret) {
|
|
ret = zend_string_init(str, size, permanent);
|
|
ZSTR_H(ret) = h;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
return zend_string_init(str, size, permanent);
|
|
}
|
|
|
|
/* Copy PHP interned strings from PHP process memory into the shared memory */
|
|
static void accel_copy_permanent_strings(zend_new_interned_string_func_t new_interned_string)
|
|
{
|
|
uint32_t j;
|
|
Bucket *p, *q;
|
|
HashTable *ht;
|
|
|
|
/* empty string */
|
|
zend_empty_string = new_interned_string(zend_empty_string);
|
|
for (j = 0; j < 256; j++) {
|
|
char s[2];
|
|
s[0] = j;
|
|
s[1] = 0;
|
|
zend_one_char_string[j] = new_interned_string(zend_string_init(s, 1, 0));
|
|
}
|
|
for (j = 0; j < ZEND_STR_LAST_KNOWN; j++) {
|
|
zend_known_strings[j] = new_interned_string(zend_known_strings[j]);
|
|
}
|
|
|
|
/* function table hash keys */
|
|
ZEND_HASH_FOREACH_BUCKET(CG(function_table), p) {
|
|
if (p->key) {
|
|
p->key = new_interned_string(p->key);
|
|
}
|
|
if (Z_FUNC(p->val)->common.function_name) {
|
|
Z_FUNC(p->val)->common.function_name = new_interned_string(Z_FUNC(p->val)->common.function_name);
|
|
}
|
|
if (Z_FUNC(p->val)->common.arg_info &&
|
|
(Z_FUNC(p->val)->common.fn_flags & (ZEND_ACC_HAS_RETURN_TYPE|ZEND_ACC_HAS_TYPE_HINTS))) {
|
|
uint32_t i;
|
|
uint32_t num_args = Z_FUNC(p->val)->common.num_args + 1;
|
|
zend_arg_info *arg_info = Z_FUNC(p->val)->common.arg_info - 1;
|
|
|
|
if (Z_FUNC(p->val)->common.fn_flags & ZEND_ACC_VARIADIC) {
|
|
num_args++;
|
|
}
|
|
for (i = 0 ; i < num_args; i++) {
|
|
if (ZEND_TYPE_IS_CLASS(arg_info[i].type)) {
|
|
zend_bool allow_null = ZEND_TYPE_ALLOW_NULL(arg_info[i].type);
|
|
arg_info[i].type = ZEND_TYPE_ENCODE_CLASS(new_interned_string(ZEND_TYPE_NAME(arg_info[i].type)), allow_null);
|
|
}
|
|
}
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
/* class table hash keys, class names, properties, methods, constants, etc */
|
|
ZEND_HASH_FOREACH_BUCKET(CG(class_table), p) {
|
|
zend_class_entry *ce;
|
|
|
|
ce = (zend_class_entry*)Z_PTR(p->val);
|
|
|
|
if (p->key) {
|
|
p->key = new_interned_string(p->key);
|
|
}
|
|
|
|
if (ce->name) {
|
|
ce->name = new_interned_string(ce->name);
|
|
}
|
|
|
|
ZEND_HASH_FOREACH_BUCKET(&ce->properties_info, q) {
|
|
zend_property_info *info;
|
|
|
|
info = (zend_property_info*)Z_PTR(q->val);
|
|
|
|
if (q->key) {
|
|
q->key = new_interned_string(q->key);
|
|
}
|
|
|
|
if (info->name) {
|
|
info->name = new_interned_string(info->name);
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
ZEND_HASH_FOREACH_BUCKET(&ce->function_table, q) {
|
|
if (q->key) {
|
|
q->key = new_interned_string(q->key);
|
|
}
|
|
if (Z_FUNC(q->val)->common.function_name) {
|
|
Z_FUNC(q->val)->common.function_name = new_interned_string(Z_FUNC(q->val)->common.function_name);
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
ZEND_HASH_FOREACH_BUCKET(&ce->constants_table, q) {
|
|
if (q->key) {
|
|
q->key = new_interned_string(q->key);
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
/* constant hash keys */
|
|
ZEND_HASH_FOREACH_BUCKET(EG(zend_constants), p) {
|
|
zend_constant *c;
|
|
|
|
if (p->key) {
|
|
p->key = new_interned_string(p->key);
|
|
}
|
|
c = (zend_constant*)Z_PTR(p->val);
|
|
if (c->name) {
|
|
c->name = new_interned_string(c->name);
|
|
}
|
|
if (Z_TYPE(c->value) == IS_STRING) {
|
|
ZVAL_STR(&c->value, new_interned_string(Z_STR(c->value)));
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
/* auto globals hash keys and names */
|
|
ZEND_HASH_FOREACH_BUCKET(CG(auto_globals), p) {
|
|
zend_auto_global *auto_global;
|
|
|
|
auto_global = (zend_auto_global*)Z_PTR(p->val);
|
|
|
|
zend_string_addref(auto_global->name);
|
|
auto_global->name = new_interned_string(auto_global->name);
|
|
if (p->key) {
|
|
p->key = new_interned_string(p->key);
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
ZEND_HASH_FOREACH_BUCKET(&module_registry, p) {
|
|
if (p->key) {
|
|
p->key = new_interned_string(p->key);
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
ZEND_HASH_FOREACH_BUCKET(EG(ini_directives), p) {
|
|
zend_ini_entry *entry = (zend_ini_entry*)Z_PTR(p->val);
|
|
|
|
if (p->key) {
|
|
p->key = new_interned_string(p->key);
|
|
}
|
|
if (entry->name) {
|
|
entry->name = new_interned_string(entry->name);
|
|
}
|
|
if (entry->value) {
|
|
entry->value = new_interned_string(entry->value);
|
|
}
|
|
if (entry->orig_value) {
|
|
entry->orig_value = new_interned_string(entry->orig_value);
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
ht = php_get_stream_filters_hash_global();
|
|
ZEND_HASH_FOREACH_BUCKET(ht, p) {
|
|
if (p->key) {
|
|
p->key = new_interned_string(p->key);
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
ht = php_stream_get_url_stream_wrappers_hash_global();
|
|
ZEND_HASH_FOREACH_BUCKET(ht, p) {
|
|
if (p->key) {
|
|
p->key = new_interned_string(p->key);
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
ht = php_stream_xport_get_hash();
|
|
ZEND_HASH_FOREACH_BUCKET(ht, p) {
|
|
if (p->key) {
|
|
p->key = new_interned_string(p->key);
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
}
|
|
|
|
static zend_string* ZEND_FASTCALL accel_replace_string_by_shm_permanent(zend_string *str)
|
|
{
|
|
zend_string *ret = accel_find_interned_string(str);
|
|
|
|
if (ret) {
|
|
zend_string_release(str);
|
|
return ret;
|
|
}
|
|
return str;
|
|
}
|
|
|
|
static void accel_use_shm_interned_strings(void)
|
|
{
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
|
SHM_UNPROTECT();
|
|
zend_shared_alloc_lock();
|
|
|
|
if (ZCSG(interned_strings).saved_top == NULL) {
|
|
accel_copy_permanent_strings(accel_new_interned_string);
|
|
} else {
|
|
accel_copy_permanent_strings(accel_replace_string_by_shm_permanent);
|
|
if (ZCG(counted)) {
|
|
accel_deactivate_sub();
|
|
}
|
|
}
|
|
accel_interned_strings_save_state();
|
|
|
|
zend_shared_alloc_unlock();
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
}
|
|
|
|
#ifndef ZEND_WIN32
|
|
static inline void kill_all_lockers(struct flock *mem_usage_check)
|
|
{
|
|
int success, tries;
|
|
/* so that other process won't try to force while we are busy cleaning up */
|
|
ZCSG(force_restart_time) = 0;
|
|
while (mem_usage_check->l_pid > 0) {
|
|
/* Clear previous errno, reset success and tries */
|
|
errno = 0;
|
|
success = 0;
|
|
tries = 10;
|
|
|
|
while (tries--) {
|
|
zend_accel_error(ACCEL_LOG_WARNING, "Attempting to kill locker %d", mem_usage_check->l_pid);
|
|
if (kill(mem_usage_check->l_pid, SIGKILL)) {
|
|
if (errno == ESRCH) {
|
|
/* Process died before the signal was sent */
|
|
success = 1;
|
|
zend_accel_error(ACCEL_LOG_WARNING, "Process %d died before SIGKILL was sent", mem_usage_check->l_pid);
|
|
}
|
|
break;
|
|
}
|
|
/* give it a chance to die */
|
|
usleep(20000);
|
|
if (kill(mem_usage_check->l_pid, 0)) {
|
|
if (errno == ESRCH) {
|
|
/* successfully killed locker, process no longer exists */
|
|
success = 1;
|
|
zend_accel_error(ACCEL_LOG_WARNING, "Killed locker %d", mem_usage_check->l_pid);
|
|
}
|
|
break;
|
|
}
|
|
usleep(10000);
|
|
}
|
|
if (!success) {
|
|
/* errno is not ESRCH or we ran out of tries to kill the locker */
|
|
ZCSG(force_restart_time) = time(NULL); /* restore forced restart request */
|
|
/* cannot kill the locker, bail out with error */
|
|
zend_accel_error(ACCEL_LOG_ERROR, "Cannot kill process %d: %s!", mem_usage_check->l_pid, strerror(errno));
|
|
}
|
|
|
|
mem_usage_check->l_type = F_WRLCK;
|
|
mem_usage_check->l_whence = SEEK_SET;
|
|
mem_usage_check->l_start = 1;
|
|
mem_usage_check->l_len = 1;
|
|
mem_usage_check->l_pid = -1;
|
|
if (fcntl(lock_file, F_GETLK, mem_usage_check) == -1) {
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "KLockers: %s (%d)", strerror(errno), errno);
|
|
break;
|
|
}
|
|
|
|
if (mem_usage_check->l_type == F_UNLCK || mem_usage_check->l_pid <= 0) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
static inline int accel_is_inactive(void)
|
|
{
|
|
#ifdef ZEND_WIN32
|
|
if (LOCKVAL(mem_usage) == 0) {
|
|
return SUCCESS;
|
|
}
|
|
#else
|
|
FLOCK_STRUCTURE(mem_usage_check, F_WRLCK, SEEK_SET, 1, 1);
|
|
|
|
mem_usage_check.l_pid = -1;
|
|
if (fcntl(lock_file, F_GETLK, &mem_usage_check) == -1) {
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "UpdateC: %s (%d)", strerror(errno), errno);
|
|
return FAILURE;
|
|
}
|
|
if (mem_usage_check.l_type == F_UNLCK) {
|
|
return SUCCESS;
|
|
}
|
|
|
|
if (ZCG(accel_directives).force_restart_timeout
|
|
&& ZCSG(force_restart_time)
|
|
&& time(NULL) >= ZCSG(force_restart_time)) {
|
|
zend_accel_error(ACCEL_LOG_WARNING, "Forced restart at %ld (after " ZEND_LONG_FMT " seconds), locked by %d", (long)time(NULL), ZCG(accel_directives).force_restart_timeout, mem_usage_check.l_pid);
|
|
kill_all_lockers(&mem_usage_check);
|
|
|
|
return FAILURE; /* next request should be able to restart it */
|
|
}
|
|
#endif
|
|
|
|
return FAILURE;
|
|
}
|
|
|
|
static int zend_get_stream_timestamp(const char *filename, zend_stat_t *statbuf)
|
|
{
|
|
php_stream_wrapper *wrapper;
|
|
php_stream_statbuf stream_statbuf;
|
|
int ret, er;
|
|
|
|
if (!filename) {
|
|
return FAILURE;
|
|
}
|
|
|
|
wrapper = php_stream_locate_url_wrapper(filename, NULL, STREAM_LOCATE_WRAPPERS_ONLY);
|
|
if (!wrapper) {
|
|
return FAILURE;
|
|
}
|
|
if (!wrapper->wops || !wrapper->wops->url_stat) {
|
|
statbuf->st_mtime = 1;
|
|
return SUCCESS; /* anything other than 0 is considered to be a valid timestamp */
|
|
}
|
|
|
|
er = EG(error_reporting);
|
|
EG(error_reporting) = 0;
|
|
zend_try {
|
|
ret = wrapper->wops->url_stat(wrapper, (char*)filename, PHP_STREAM_URL_STAT_QUIET, &stream_statbuf, NULL);
|
|
} zend_catch {
|
|
ret = -1;
|
|
} zend_end_try();
|
|
EG(error_reporting) = er;
|
|
|
|
if (ret != 0) {
|
|
return FAILURE;
|
|
}
|
|
|
|
*statbuf = stream_statbuf.sb;
|
|
return SUCCESS;
|
|
}
|
|
|
|
#if ZEND_WIN32
|
|
static accel_time_t zend_get_file_handle_timestamp_win(zend_file_handle *file_handle, size_t *size)
|
|
{
|
|
static unsigned __int64 utc_base = 0;
|
|
static FILETIME utc_base_ft;
|
|
WIN32_FILE_ATTRIBUTE_DATA fdata;
|
|
|
|
if (!file_handle->opened_path) {
|
|
return 0;
|
|
}
|
|
|
|
if (!utc_base) {
|
|
SYSTEMTIME st;
|
|
|
|
st.wYear = 1970;
|
|
st.wMonth = 1;
|
|
st.wDay = 1;
|
|
st.wHour = 0;
|
|
st.wMinute = 0;
|
|
st.wSecond = 0;
|
|
st.wMilliseconds = 0;
|
|
|
|
SystemTimeToFileTime (&st, &utc_base_ft);
|
|
utc_base = (((unsigned __int64)utc_base_ft.dwHighDateTime) << 32) + utc_base_ft.dwLowDateTime;
|
|
}
|
|
|
|
if (file_handle->opened_path && GetFileAttributesEx(file_handle->opened_path->val, GetFileExInfoStandard, &fdata) != 0) {
|
|
unsigned __int64 ftime;
|
|
|
|
if (CompareFileTime (&fdata.ftLastWriteTime, &utc_base_ft) < 0) {
|
|
return 0;
|
|
}
|
|
|
|
ftime = (((unsigned __int64)fdata.ftLastWriteTime.dwHighDateTime) << 32) + fdata.ftLastWriteTime.dwLowDateTime - utc_base;
|
|
ftime /= 10000000L;
|
|
|
|
if (size) {
|
|
*size = (size_t)((((unsigned __int64)fdata.nFileSizeHigh) << 32) + (unsigned __int64)fdata.nFileSizeLow);
|
|
}
|
|
return (accel_time_t)ftime;
|
|
}
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
accel_time_t zend_get_file_handle_timestamp(zend_file_handle *file_handle, size_t *size)
|
|
{
|
|
zend_stat_t statbuf;
|
|
#ifdef ZEND_WIN32
|
|
accel_time_t res;
|
|
#endif
|
|
|
|
if (sapi_module.get_stat &&
|
|
!EG(current_execute_data) &&
|
|
file_handle->filename == SG(request_info).path_translated) {
|
|
|
|
zend_stat_t *tmpbuf = sapi_module.get_stat();
|
|
|
|
if (tmpbuf) {
|
|
if (size) {
|
|
*size = tmpbuf->st_size;
|
|
}
|
|
return tmpbuf->st_mtime;
|
|
}
|
|
}
|
|
|
|
#ifdef ZEND_WIN32
|
|
res = zend_get_file_handle_timestamp_win(file_handle, size);
|
|
if (res) {
|
|
return res;
|
|
}
|
|
#endif
|
|
|
|
switch (file_handle->type) {
|
|
case ZEND_HANDLE_FD:
|
|
if (zend_fstat(file_handle->handle.fd, &statbuf) == -1) {
|
|
return 0;
|
|
}
|
|
break;
|
|
case ZEND_HANDLE_FP:
|
|
if (zend_fstat(fileno(file_handle->handle.fp), &statbuf) == -1) {
|
|
if (zend_get_stream_timestamp(file_handle->filename, &statbuf) != SUCCESS) {
|
|
return 0;
|
|
}
|
|
}
|
|
break;
|
|
case ZEND_HANDLE_FILENAME:
|
|
case ZEND_HANDLE_MAPPED:
|
|
if (file_handle->opened_path) {
|
|
char *file_path = ZSTR_VAL(file_handle->opened_path);
|
|
|
|
if (is_stream_path(file_path)) {
|
|
if (zend_get_stream_timestamp(file_path, &statbuf) == SUCCESS) {
|
|
break;
|
|
}
|
|
}
|
|
if (VCWD_STAT(file_path, &statbuf) != -1) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (zend_get_stream_timestamp(file_handle->filename, &statbuf) != SUCCESS) {
|
|
return 0;
|
|
}
|
|
break;
|
|
case ZEND_HANDLE_STREAM:
|
|
{
|
|
php_stream *stream = (php_stream *)file_handle->handle.stream.handle;
|
|
php_stream_statbuf sb;
|
|
int ret, er;
|
|
|
|
if (!stream ||
|
|
!stream->ops ||
|
|
!stream->ops->stat) {
|
|
return 0;
|
|
}
|
|
|
|
er = EG(error_reporting);
|
|
EG(error_reporting) = 0;
|
|
zend_try {
|
|
ret = stream->ops->stat(stream, &sb);
|
|
} zend_catch {
|
|
ret = -1;
|
|
} zend_end_try();
|
|
EG(error_reporting) = er;
|
|
if (ret != 0) {
|
|
return 0;
|
|
}
|
|
|
|
statbuf = sb.sb;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
return 0;
|
|
}
|
|
|
|
if (size) {
|
|
*size = statbuf.st_size;
|
|
}
|
|
return statbuf.st_mtime;
|
|
}
|
|
|
|
static inline int do_validate_timestamps(zend_persistent_script *persistent_script, zend_file_handle *file_handle)
|
|
{
|
|
zend_file_handle ps_handle;
|
|
zend_string *full_path_ptr = NULL;
|
|
|
|
/** check that the persistent script is indeed the same file we cached
|
|
* (if part of the path is a symlink than it possible that the user will change it)
|
|
* See bug #15140
|
|
*/
|
|
if (file_handle->opened_path) {
|
|
if (persistent_script->script.filename != file_handle->opened_path &&
|
|
!zend_string_equal_content(persistent_script->script.filename, file_handle->opened_path)) {
|
|
return FAILURE;
|
|
}
|
|
} else {
|
|
full_path_ptr = accelerator_orig_zend_resolve_path(file_handle->filename, strlen(file_handle->filename));
|
|
if (full_path_ptr &&
|
|
persistent_script->script.filename != full_path_ptr &&
|
|
!zend_string_equal_content(persistent_script->script.filename, full_path_ptr)) {
|
|
zend_string_release_ex(full_path_ptr, 0);
|
|
return FAILURE;
|
|
}
|
|
file_handle->opened_path = full_path_ptr;
|
|
}
|
|
|
|
if (persistent_script->timestamp == 0) {
|
|
if (full_path_ptr) {
|
|
zend_string_release_ex(full_path_ptr, 0);
|
|
file_handle->opened_path = NULL;
|
|
}
|
|
return FAILURE;
|
|
}
|
|
|
|
if (zend_get_file_handle_timestamp(file_handle, NULL) == persistent_script->timestamp) {
|
|
if (full_path_ptr) {
|
|
zend_string_release_ex(full_path_ptr, 0);
|
|
file_handle->opened_path = NULL;
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
if (full_path_ptr) {
|
|
zend_string_release_ex(full_path_ptr, 0);
|
|
file_handle->opened_path = NULL;
|
|
}
|
|
|
|
ps_handle.type = ZEND_HANDLE_FILENAME;
|
|
ps_handle.filename = ZSTR_VAL(persistent_script->script.filename);
|
|
ps_handle.opened_path = persistent_script->script.filename;
|
|
|
|
if (zend_get_file_handle_timestamp(&ps_handle, NULL) == persistent_script->timestamp) {
|
|
return SUCCESS;
|
|
}
|
|
|
|
return FAILURE;
|
|
}
|
|
|
|
int validate_timestamp_and_record(zend_persistent_script *persistent_script, zend_file_handle *file_handle)
|
|
{
|
|
if (persistent_script->timestamp == 0) {
|
|
return SUCCESS; /* Don't check timestamps of preloaded scripts */
|
|
} else if (ZCG(accel_directives).revalidate_freq &&
|
|
persistent_script->dynamic_members.revalidate >= ZCG(request_time)) {
|
|
return SUCCESS;
|
|
} else if (do_validate_timestamps(persistent_script, file_handle) == FAILURE) {
|
|
return FAILURE;
|
|
} else {
|
|
persistent_script->dynamic_members.revalidate = ZCG(request_time) + ZCG(accel_directives).revalidate_freq;
|
|
return SUCCESS;
|
|
}
|
|
}
|
|
|
|
int validate_timestamp_and_record_ex(zend_persistent_script *persistent_script, zend_file_handle *file_handle)
|
|
{
|
|
int ret;
|
|
|
|
SHM_UNPROTECT();
|
|
ret = validate_timestamp_and_record(persistent_script, file_handle);
|
|
SHM_PROTECT();
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* Instead of resolving full real path name each time we need to identify file,
|
|
* we create a key that consist from requested file name, current working
|
|
* directory, current include_path, etc */
|
|
char *accel_make_persistent_key(const char *path, size_t path_length, int *key_len)
|
|
{
|
|
int key_length;
|
|
|
|
/* CWD and include_path don't matter for absolute file names and streams */
|
|
if (IS_ABSOLUTE_PATH(path, path_length)) {
|
|
/* pass */
|
|
ZCG(key_len) = 0;
|
|
} else if (UNEXPECTED(is_stream_path(path))) {
|
|
if (!is_cacheable_stream_path(path)) {
|
|
return NULL;
|
|
}
|
|
/* pass */
|
|
ZCG(key_len) = 0;
|
|
} else if (UNEXPECTED(!ZCG(accel_directives).use_cwd)) {
|
|
/* pass */
|
|
ZCG(key_len) = 0;
|
|
} else {
|
|
const char *include_path = NULL, *cwd = NULL;
|
|
int include_path_len = 0, cwd_len = 0;
|
|
zend_string *parent_script = NULL;
|
|
size_t parent_script_len = 0;
|
|
|
|
if (EXPECTED(ZCG(cwd_key_len))) {
|
|
cwd = ZCG(cwd_key);
|
|
cwd_len = ZCG(cwd_key_len);
|
|
} else {
|
|
zend_string *cwd_str = accel_getcwd();
|
|
|
|
if (UNEXPECTED(!cwd_str)) {
|
|
/* we don't handle this well for now. */
|
|
zend_accel_error(ACCEL_LOG_INFO, "getcwd() failed for '%s' (%d), please try to set opcache.use_cwd to 0 in ini file", path, errno);
|
|
return NULL;
|
|
}
|
|
cwd = ZSTR_VAL(cwd_str);
|
|
cwd_len = ZSTR_LEN(cwd_str);
|
|
if (ZCG(cwd_check)) {
|
|
ZCG(cwd_check) = 0;
|
|
if ((ZCG(counted) || ZCSG(accelerator_enabled))) {
|
|
|
|
zend_string *str = accel_find_interned_string(cwd_str);
|
|
if (!str) {
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
|
SHM_UNPROTECT();
|
|
zend_shared_alloc_lock();
|
|
str = accel_new_interned_string(zend_string_copy(cwd_str));
|
|
if (str == cwd_str) {
|
|
zend_string_release_ex(str, 0);
|
|
str = NULL;
|
|
}
|
|
zend_shared_alloc_unlock();
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
}
|
|
if (str) {
|
|
char buf[32];
|
|
char *res = zend_print_long_to_buf(buf + sizeof(buf) - 1, STRTAB_STR_TO_POS(&ZCSG(interned_strings), str));
|
|
|
|
cwd_len = ZCG(cwd_key_len) = buf + sizeof(buf) - 1 - res;
|
|
cwd = ZCG(cwd_key);
|
|
memcpy(ZCG(cwd_key), res, cwd_len + 1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (EXPECTED(ZCG(include_path_key_len))) {
|
|
include_path = ZCG(include_path_key);
|
|
include_path_len = ZCG(include_path_key_len);
|
|
} else if (!ZCG(include_path) || ZSTR_LEN(ZCG(include_path)) == 0) {
|
|
include_path = "";
|
|
include_path_len = 0;
|
|
} else {
|
|
include_path = ZSTR_VAL(ZCG(include_path));
|
|
include_path_len = ZSTR_LEN(ZCG(include_path));
|
|
|
|
if (ZCG(include_path_check)) {
|
|
ZCG(include_path_check) = 0;
|
|
if ((ZCG(counted) || ZCSG(accelerator_enabled))) {
|
|
|
|
zend_string *str = accel_find_interned_string(ZCG(include_path));
|
|
if (!str) {
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
|
SHM_UNPROTECT();
|
|
zend_shared_alloc_lock();
|
|
str = accel_new_interned_string(zend_string_copy(ZCG(include_path)));
|
|
if (str == ZCG(include_path)) {
|
|
str = NULL;
|
|
}
|
|
zend_shared_alloc_unlock();
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
}
|
|
if (str) {
|
|
char buf[32];
|
|
char *res = zend_print_long_to_buf(buf + sizeof(buf) - 1, STRTAB_STR_TO_POS(&ZCSG(interned_strings), str));
|
|
|
|
include_path_len = ZCG(include_path_key_len) = buf + sizeof(buf) - 1 - res;
|
|
include_path = ZCG(include_path_key);
|
|
memcpy(ZCG(include_path_key), res, include_path_len + 1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Calculate key length */
|
|
if (UNEXPECTED((size_t)(cwd_len + path_length + include_path_len + 2) >= sizeof(ZCG(key)))) {
|
|
return NULL;
|
|
}
|
|
|
|
/* Generate key
|
|
* Note - the include_path must be the last element in the key,
|
|
* since in itself, it may include colons (which we use to separate
|
|
* different components of the key)
|
|
*/
|
|
memcpy(ZCG(key), path, path_length);
|
|
ZCG(key)[path_length] = ':';
|
|
key_length = path_length + 1;
|
|
memcpy(ZCG(key) + key_length, cwd, cwd_len);
|
|
key_length += cwd_len;
|
|
|
|
if (include_path_len) {
|
|
ZCG(key)[key_length] = ':';
|
|
key_length += 1;
|
|
memcpy(ZCG(key) + key_length, include_path, include_path_len);
|
|
key_length += include_path_len;
|
|
}
|
|
|
|
/* Here we add to the key the parent script directory,
|
|
* since fopen_wrappers from version 4.0.7 use current script's path
|
|
* in include path too.
|
|
*/
|
|
if (EXPECTED(EG(current_execute_data)) &&
|
|
EXPECTED((parent_script = zend_get_executed_filename_ex()) != NULL)) {
|
|
|
|
parent_script_len = ZSTR_LEN(parent_script);
|
|
while ((--parent_script_len > 0) && !IS_SLASH(ZSTR_VAL(parent_script)[parent_script_len]));
|
|
|
|
if (UNEXPECTED((size_t)(key_length + parent_script_len + 1) >= sizeof(ZCG(key)))) {
|
|
return NULL;
|
|
}
|
|
ZCG(key)[key_length] = ':';
|
|
key_length += 1;
|
|
memcpy(ZCG(key) + key_length, ZSTR_VAL(parent_script), parent_script_len);
|
|
key_length += parent_script_len;
|
|
}
|
|
ZCG(key)[key_length] = '\0';
|
|
*key_len = ZCG(key_len) = key_length;
|
|
return ZCG(key);
|
|
}
|
|
|
|
/* not use_cwd */
|
|
*key_len = path_length;
|
|
return (char*)path;
|
|
}
|
|
|
|
int zend_accel_invalidate(const char *filename, size_t filename_len, zend_bool force)
|
|
{
|
|
zend_string *realpath;
|
|
zend_persistent_script *persistent_script;
|
|
|
|
if (!ZCG(enabled) || !accel_startup_ok || !ZCSG(accelerator_enabled) || accelerator_shm_read_lock() != SUCCESS) {
|
|
return FAILURE;
|
|
}
|
|
|
|
realpath = accelerator_orig_zend_resolve_path(filename, filename_len);
|
|
|
|
if (!realpath) {
|
|
return FAILURE;
|
|
}
|
|
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
if (ZCG(accel_directives).file_cache) {
|
|
zend_file_cache_invalidate(realpath);
|
|
}
|
|
#endif
|
|
|
|
persistent_script = zend_accel_hash_find(&ZCSG(hash), realpath);
|
|
if (persistent_script && !persistent_script->corrupted) {
|
|
zend_file_handle file_handle;
|
|
|
|
file_handle.type = ZEND_HANDLE_FILENAME;
|
|
file_handle.filename = ZSTR_VAL(realpath);
|
|
file_handle.opened_path = realpath;
|
|
|
|
if (force ||
|
|
!ZCG(accel_directives).validate_timestamps ||
|
|
do_validate_timestamps(persistent_script, &file_handle) == FAILURE) {
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
|
SHM_UNPROTECT();
|
|
zend_shared_alloc_lock();
|
|
if (!persistent_script->corrupted) {
|
|
persistent_script->corrupted = 1;
|
|
persistent_script->timestamp = 0;
|
|
ZSMMG(wasted_shared_memory) += persistent_script->dynamic_members.memory_consumption;
|
|
if (ZSMMG(memory_exhausted)) {
|
|
zend_accel_restart_reason reason =
|
|
zend_accel_hash_is_full(&ZCSG(hash)) ? ACCEL_RESTART_HASH : ACCEL_RESTART_OOM;
|
|
zend_accel_schedule_restart_if_necessary(reason);
|
|
}
|
|
}
|
|
zend_shared_alloc_unlock();
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
}
|
|
}
|
|
|
|
accelerator_shm_read_unlock();
|
|
zend_string_release_ex(realpath, 0);
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* Adds another key for existing cached script */
|
|
static void zend_accel_add_key(const char *key, unsigned int key_length, zend_accel_hash_entry *bucket)
|
|
{
|
|
if (!zend_accel_hash_str_find(&ZCSG(hash), key, key_length)) {
|
|
if (zend_accel_hash_is_full(&ZCSG(hash))) {
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "No more entries in hash table!");
|
|
ZSMMG(memory_exhausted) = 1;
|
|
zend_accel_schedule_restart_if_necessary(ACCEL_RESTART_HASH);
|
|
} else {
|
|
char *new_key = zend_shared_alloc(key_length + 1);
|
|
if (new_key) {
|
|
memcpy(new_key, key, key_length + 1);
|
|
if (zend_accel_hash_update(&ZCSG(hash), new_key, key_length, 1, bucket)) {
|
|
zend_accel_error(ACCEL_LOG_INFO, "Added key '%s'", new_key);
|
|
}
|
|
} else {
|
|
zend_accel_schedule_restart_if_necessary(ACCEL_RESTART_OOM);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static zend_always_inline zend_bool is_phar_file(zend_string *filename)
|
|
{
|
|
return filename && ZSTR_LEN(filename) >= sizeof(".phar") &&
|
|
!memcmp(ZSTR_VAL(filename) + ZSTR_LEN(filename) - (sizeof(".phar")-1), ".phar", sizeof(".phar")-1) &&
|
|
!strstr(ZSTR_VAL(filename), "://");
|
|
}
|
|
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
static zend_persistent_script *store_script_in_file_cache(zend_persistent_script *new_persistent_script)
|
|
{
|
|
uint32_t memory_used;
|
|
|
|
zend_shared_alloc_init_xlat_table();
|
|
|
|
/* Calculate the required memory size */
|
|
memory_used = zend_accel_script_persist_calc(new_persistent_script, NULL, 0, 0);
|
|
|
|
/* Allocate memory block */
|
|
#if defined(__AVX__) || defined(__SSE2__)
|
|
/* Align to 64-byte boundary */
|
|
ZCG(mem) = zend_arena_alloc(&CG(arena), memory_used + 64);
|
|
ZCG(mem) = (void*)(((zend_uintptr_t)ZCG(mem) + 63L) & ~63L);
|
|
#else
|
|
ZCG(mem) = zend_arena_alloc(&CG(arena), memory_used);
|
|
#endif
|
|
|
|
zend_shared_alloc_clear_xlat_table();
|
|
|
|
/* Copy into memory block */
|
|
new_persistent_script = zend_accel_script_persist(new_persistent_script, NULL, 0, 0);
|
|
|
|
zend_shared_alloc_destroy_xlat_table();
|
|
|
|
new_persistent_script->is_phar = is_phar_file(new_persistent_script->script.filename);
|
|
|
|
/* Consistency check */
|
|
if ((char*)new_persistent_script->mem + new_persistent_script->size != (char*)ZCG(mem)) {
|
|
zend_accel_error(
|
|
((char*)new_persistent_script->mem + new_persistent_script->size < (char*)ZCG(mem)) ? ACCEL_LOG_ERROR : ACCEL_LOG_WARNING,
|
|
"Internal error: wrong size calculation: %s start=" ZEND_ADDR_FMT ", end=" ZEND_ADDR_FMT ", real=" ZEND_ADDR_FMT "\n",
|
|
ZSTR_VAL(new_persistent_script->script.filename),
|
|
(size_t)new_persistent_script->mem,
|
|
(size_t)((char *)new_persistent_script->mem + new_persistent_script->size),
|
|
(size_t)ZCG(mem));
|
|
}
|
|
|
|
new_persistent_script->dynamic_members.checksum = zend_accel_script_checksum(new_persistent_script);
|
|
|
|
zend_file_cache_script_store(new_persistent_script, 0);
|
|
|
|
return new_persistent_script;
|
|
}
|
|
|
|
static zend_persistent_script *cache_script_in_file_cache(zend_persistent_script *new_persistent_script, int *from_shared_memory)
|
|
{
|
|
uint32_t orig_compiler_options;
|
|
|
|
/* Check if script may be stored in shared memory */
|
|
if (!zend_accel_script_persistable(new_persistent_script)) {
|
|
return new_persistent_script;
|
|
}
|
|
|
|
orig_compiler_options = CG(compiler_options);
|
|
CG(compiler_options) |= ZEND_COMPILE_WITH_FILE_CACHE;
|
|
if (!zend_optimize_script(&new_persistent_script->script, ZCG(accel_directives).optimization_level, ZCG(accel_directives).opt_debug_level)) {
|
|
CG(compiler_options) = orig_compiler_options;
|
|
return new_persistent_script;
|
|
}
|
|
CG(compiler_options) = orig_compiler_options;
|
|
|
|
*from_shared_memory = 1;
|
|
return store_script_in_file_cache(new_persistent_script);
|
|
}
|
|
#endif
|
|
|
|
static zend_persistent_script *cache_script_in_shared_memory(zend_persistent_script *new_persistent_script, const char *key, unsigned int key_length, int *from_shared_memory)
|
|
{
|
|
zend_accel_hash_entry *bucket;
|
|
uint32_t memory_used;
|
|
uint32_t orig_compiler_options;
|
|
|
|
/* Check if script may be stored in shared memory */
|
|
if (!zend_accel_script_persistable(new_persistent_script)) {
|
|
return new_persistent_script;
|
|
}
|
|
|
|
orig_compiler_options = CG(compiler_options);
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
if (ZCG(accel_directives).file_cache) {
|
|
CG(compiler_options) |= ZEND_COMPILE_WITH_FILE_CACHE;
|
|
}
|
|
#endif
|
|
if (!zend_optimize_script(&new_persistent_script->script, ZCG(accel_directives).optimization_level, ZCG(accel_directives).opt_debug_level)) {
|
|
CG(compiler_options) = orig_compiler_options;
|
|
return new_persistent_script;
|
|
}
|
|
CG(compiler_options) = orig_compiler_options;
|
|
|
|
/* exclusive lock */
|
|
zend_shared_alloc_lock();
|
|
|
|
/* Check if we still need to put the file into the cache (may be it was
|
|
* already stored by another process. This final check is done under
|
|
* exclusive lock) */
|
|
bucket = zend_accel_hash_find_entry(&ZCSG(hash), new_persistent_script->script.filename);
|
|
if (bucket) {
|
|
zend_persistent_script *existing_persistent_script = (zend_persistent_script *)bucket->data;
|
|
|
|
if (!existing_persistent_script->corrupted) {
|
|
if (key &&
|
|
(!ZCG(accel_directives).validate_timestamps ||
|
|
(new_persistent_script->timestamp == existing_persistent_script->timestamp))) {
|
|
zend_accel_add_key(key, key_length, bucket);
|
|
}
|
|
zend_shared_alloc_unlock();
|
|
return new_persistent_script;
|
|
}
|
|
}
|
|
|
|
if (zend_accel_hash_is_full(&ZCSG(hash))) {
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "No more entries in hash table!");
|
|
ZSMMG(memory_exhausted) = 1;
|
|
zend_accel_schedule_restart_if_necessary(ACCEL_RESTART_HASH);
|
|
zend_shared_alloc_unlock();
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
if (ZCG(accel_directives).file_cache) {
|
|
new_persistent_script = store_script_in_file_cache(new_persistent_script);
|
|
*from_shared_memory = 1;
|
|
}
|
|
#endif
|
|
return new_persistent_script;
|
|
}
|
|
|
|
zend_shared_alloc_init_xlat_table();
|
|
|
|
/* Calculate the required memory size */
|
|
memory_used = zend_accel_script_persist_calc(new_persistent_script, key, key_length, 1);
|
|
|
|
/* Allocate shared memory */
|
|
#if defined(__AVX__) || defined(__SSE2__)
|
|
/* Align to 64-byte boundary */
|
|
ZCG(mem) = zend_shared_alloc(memory_used + 64);
|
|
if (ZCG(mem)) {
|
|
ZCG(mem) = (void*)(((zend_uintptr_t)ZCG(mem) + 63L) & ~63L);
|
|
#if defined(__x86_64__)
|
|
memset(ZCG(mem), 0, memory_used);
|
|
#elif defined(__AVX__)
|
|
{
|
|
char *p = (char*)ZCG(mem);
|
|
char *end = p + memory_used;
|
|
__m256i ymm0 = _mm256_setzero_si256();
|
|
|
|
while (p < end) {
|
|
_mm256_store_si256((__m256i*)p, ymm0);
|
|
_mm256_store_si256((__m256i*)(p+32), ymm0);
|
|
p += 64;
|
|
}
|
|
}
|
|
#else
|
|
{
|
|
char *p = (char*)ZCG(mem);
|
|
char *end = p + memory_used;
|
|
__m128i xmm0 = _mm_setzero_si128();
|
|
|
|
while (p < end) {
|
|
_mm_store_si128((__m128i*)p, xmm0);
|
|
_mm_store_si128((__m128i*)(p+16), xmm0);
|
|
_mm_store_si128((__m128i*)(p+32), xmm0);
|
|
_mm_store_si128((__m128i*)(p+48), xmm0);
|
|
p += 64;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
#else
|
|
ZCG(mem) = zend_shared_alloc(memory_used);
|
|
if (ZCG(mem)) {
|
|
memset(ZCG(mem), 0, memory_used);
|
|
}
|
|
#endif
|
|
if (!ZCG(mem)) {
|
|
zend_shared_alloc_destroy_xlat_table();
|
|
zend_accel_schedule_restart_if_necessary(ACCEL_RESTART_OOM);
|
|
zend_shared_alloc_unlock();
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
if (ZCG(accel_directives).file_cache) {
|
|
new_persistent_script = store_script_in_file_cache(new_persistent_script);
|
|
*from_shared_memory = 1;
|
|
}
|
|
#endif
|
|
return new_persistent_script;
|
|
}
|
|
|
|
zend_shared_alloc_clear_xlat_table();
|
|
|
|
/* Copy into shared memory */
|
|
new_persistent_script = zend_accel_script_persist(new_persistent_script, &key, key_length, 1);
|
|
|
|
zend_shared_alloc_destroy_xlat_table();
|
|
|
|
new_persistent_script->is_phar = is_phar_file(new_persistent_script->script.filename);
|
|
|
|
/* Consistency check */
|
|
if ((char*)new_persistent_script->mem + new_persistent_script->size != (char*)ZCG(mem)) {
|
|
zend_accel_error(
|
|
((char*)new_persistent_script->mem + new_persistent_script->size < (char*)ZCG(mem)) ? ACCEL_LOG_ERROR : ACCEL_LOG_WARNING,
|
|
"Internal error: wrong size calculation: %s start=" ZEND_ADDR_FMT ", end=" ZEND_ADDR_FMT ", real=" ZEND_ADDR_FMT "\n",
|
|
ZSTR_VAL(new_persistent_script->script.filename),
|
|
(size_t)new_persistent_script->mem,
|
|
(size_t)((char *)new_persistent_script->mem + new_persistent_script->size),
|
|
(size_t)ZCG(mem));
|
|
}
|
|
|
|
new_persistent_script->dynamic_members.checksum = zend_accel_script_checksum(new_persistent_script);
|
|
|
|
/* store script structure in the hash table */
|
|
bucket = zend_accel_hash_update(&ZCSG(hash), ZSTR_VAL(new_persistent_script->script.filename), ZSTR_LEN(new_persistent_script->script.filename), 0, new_persistent_script);
|
|
if (bucket) {
|
|
zend_accel_error(ACCEL_LOG_INFO, "Cached script '%s'", ZSTR_VAL(new_persistent_script->script.filename));
|
|
if (key &&
|
|
/* key may contain non-persistent PHAR aliases (see issues #115 and #149) */
|
|
memcmp(key, "phar://", sizeof("phar://") - 1) != 0 &&
|
|
(ZSTR_LEN(new_persistent_script->script.filename) != key_length ||
|
|
memcmp(ZSTR_VAL(new_persistent_script->script.filename), key, key_length) != 0)) {
|
|
/* link key to the same persistent script in hash table */
|
|
if (zend_accel_hash_update(&ZCSG(hash), key, key_length, 1, bucket)) {
|
|
zend_accel_error(ACCEL_LOG_INFO, "Added key '%s'", key);
|
|
} else {
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "No more entries in hash table!");
|
|
ZSMMG(memory_exhausted) = 1;
|
|
zend_accel_schedule_restart_if_necessary(ACCEL_RESTART_HASH);
|
|
}
|
|
}
|
|
}
|
|
|
|
new_persistent_script->dynamic_members.memory_consumption = ZEND_ALIGNED_SIZE(new_persistent_script->size);
|
|
|
|
zend_shared_alloc_unlock();
|
|
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
if (ZCG(accel_directives).file_cache) {
|
|
SHM_PROTECT();
|
|
zend_file_cache_script_store(new_persistent_script, 1);
|
|
SHM_UNPROTECT();
|
|
}
|
|
#endif
|
|
|
|
*from_shared_memory = 1;
|
|
return new_persistent_script;
|
|
}
|
|
|
|
static const struct jit_auto_global_info
|
|
{
|
|
const char *name;
|
|
size_t len;
|
|
} jit_auto_globals_info[] = {
|
|
{ "_SERVER", sizeof("_SERVER")-1},
|
|
{ "_ENV", sizeof("_ENV")-1},
|
|
{ "_REQUEST", sizeof("_REQUEST")-1},
|
|
{ "GLOBALS", sizeof("GLOBALS")-1},
|
|
};
|
|
|
|
static zend_string *jit_auto_globals_str[4];
|
|
|
|
static int zend_accel_get_auto_globals(void)
|
|
{
|
|
int i, ag_size = (sizeof(jit_auto_globals_info) / sizeof(jit_auto_globals_info[0]));
|
|
int n = 1;
|
|
int mask = 0;
|
|
|
|
for (i = 0; i < ag_size ; i++) {
|
|
if (zend_hash_exists(&EG(symbol_table), jit_auto_globals_str[i])) {
|
|
mask |= n;
|
|
}
|
|
n += n;
|
|
}
|
|
return mask;
|
|
}
|
|
|
|
static int zend_accel_get_auto_globals_no_jit(void)
|
|
{
|
|
if (zend_hash_exists(&EG(symbol_table), jit_auto_globals_str[3])) {
|
|
return 8;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void zend_accel_set_auto_globals(int mask)
|
|
{
|
|
int i, ag_size = (sizeof(jit_auto_globals_info) / sizeof(jit_auto_globals_info[0]));
|
|
int n = 1;
|
|
|
|
for (i = 0; i < ag_size ; i++) {
|
|
if ((mask & n) && !(ZCG(auto_globals_mask) & n)) {
|
|
ZCG(auto_globals_mask) |= n;
|
|
zend_is_auto_global(jit_auto_globals_str[i]);
|
|
}
|
|
n += n;
|
|
}
|
|
}
|
|
|
|
static void zend_accel_init_auto_globals(void)
|
|
{
|
|
int i, ag_size = (sizeof(jit_auto_globals_info) / sizeof(jit_auto_globals_info[0]));
|
|
|
|
for (i = 0; i < ag_size ; i++) {
|
|
jit_auto_globals_str[i] = zend_string_init(jit_auto_globals_info[i].name, jit_auto_globals_info[i].len, 1);
|
|
zend_string_hash_val(jit_auto_globals_str[i]);
|
|
jit_auto_globals_str[i] = accel_new_interned_string(jit_auto_globals_str[i]);
|
|
}
|
|
}
|
|
|
|
static zend_persistent_script *opcache_compile_file(zend_file_handle *file_handle, int type, const char *key, zend_op_array **op_array_p)
|
|
{
|
|
zend_persistent_script *new_persistent_script;
|
|
zend_op_array *orig_active_op_array;
|
|
HashTable *orig_class_table;
|
|
zval orig_user_error_handler;
|
|
zend_op_array *op_array;
|
|
int do_bailout = 0;
|
|
accel_time_t timestamp = 0;
|
|
uint32_t orig_compiler_options = 0;
|
|
|
|
/* Try to open file */
|
|
if (file_handle->type == ZEND_HANDLE_FILENAME) {
|
|
if (accelerator_orig_zend_stream_open_function(file_handle->filename, file_handle) != SUCCESS) {
|
|
*op_array_p = NULL;
|
|
if (type == ZEND_REQUIRE) {
|
|
zend_message_dispatcher(ZMSG_FAILED_REQUIRE_FOPEN, file_handle->filename);
|
|
zend_bailout();
|
|
} else {
|
|
zend_message_dispatcher(ZMSG_FAILED_INCLUDE_FOPEN, file_handle->filename);
|
|
}
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
/* check blacklist right after ensuring that file was opened */
|
|
if (file_handle->opened_path && zend_accel_blacklist_is_blacklisted(&accel_blacklist, ZSTR_VAL(file_handle->opened_path), ZSTR_LEN(file_handle->opened_path))) {
|
|
ZCSG(blacklist_misses)++;
|
|
*op_array_p = accelerator_orig_compile_file(file_handle, type);
|
|
return NULL;
|
|
}
|
|
|
|
if (ZCG(accel_directives).validate_timestamps ||
|
|
ZCG(accel_directives).file_update_protection ||
|
|
ZCG(accel_directives).max_file_size > 0) {
|
|
size_t size = 0;
|
|
|
|
/* Obtain the file timestamps, *before* actually compiling them,
|
|
* otherwise we have a race-condition.
|
|
*/
|
|
timestamp = zend_get_file_handle_timestamp(file_handle, ZCG(accel_directives).max_file_size > 0 ? &size : NULL);
|
|
|
|
/* If we can't obtain a timestamp (that means file is possibly socket)
|
|
* we won't cache it
|
|
*/
|
|
if (timestamp == 0) {
|
|
*op_array_p = accelerator_orig_compile_file(file_handle, type);
|
|
return NULL;
|
|
}
|
|
|
|
/* check if file is too new (may be it's not written completely yet) */
|
|
if (ZCG(accel_directives).file_update_protection &&
|
|
((accel_time_t)(ZCG(request_time) - ZCG(accel_directives).file_update_protection) < timestamp)) {
|
|
*op_array_p = accelerator_orig_compile_file(file_handle, type);
|
|
return NULL;
|
|
}
|
|
|
|
if (ZCG(accel_directives).max_file_size > 0 && size > (size_t)ZCG(accel_directives).max_file_size) {
|
|
ZCSG(blacklist_misses)++;
|
|
*op_array_p = accelerator_orig_compile_file(file_handle, type);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
new_persistent_script = create_persistent_script();
|
|
|
|
/* Save the original values for the op_array, function table and class table */
|
|
orig_active_op_array = CG(active_op_array);
|
|
orig_class_table = CG(class_table);
|
|
ZVAL_COPY_VALUE(&orig_user_error_handler, &EG(user_error_handler));
|
|
|
|
/* Override them with ours */
|
|
EG(class_table) = CG(class_table) = &new_persistent_script->script.class_table;
|
|
ZVAL_UNDEF(&EG(user_error_handler));
|
|
|
|
zend_try {
|
|
orig_compiler_options = CG(compiler_options);
|
|
CG(compiler_options) |= ZEND_COMPILE_HANDLE_OP_ARRAY;
|
|
CG(compiler_options) |= ZEND_COMPILE_IGNORE_INTERNAL_CLASSES;
|
|
CG(compiler_options) |= ZEND_COMPILE_DELAYED_BINDING;
|
|
CG(compiler_options) |= ZEND_COMPILE_NO_CONSTANT_SUBSTITUTION;
|
|
CG(compiler_options) |= ZEND_COMPILE_IGNORE_OTHER_FILES;
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
if (ZCG(accel_directives).file_cache) {
|
|
CG(compiler_options) |= ZEND_COMPILE_WITH_FILE_CACHE;
|
|
}
|
|
#endif
|
|
op_array = *op_array_p = accelerator_orig_compile_file(file_handle, type);
|
|
CG(compiler_options) = orig_compiler_options;
|
|
} zend_catch {
|
|
op_array = NULL;
|
|
do_bailout = 1;
|
|
CG(compiler_options) = orig_compiler_options;
|
|
} zend_end_try();
|
|
|
|
/* Restore originals */
|
|
CG(active_op_array) = orig_active_op_array;
|
|
EG(class_table) = CG(class_table) = orig_class_table;
|
|
EG(user_error_handler) = orig_user_error_handler;
|
|
|
|
if (!op_array) {
|
|
/* compilation failed */
|
|
free_persistent_script(new_persistent_script, 1);
|
|
if (do_bailout) {
|
|
zend_bailout();
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/* Build the persistent_script structure.
|
|
Here we aren't sure we would store it, but we will need it
|
|
further anyway.
|
|
*/
|
|
new_persistent_script->script.main_op_array = *op_array;
|
|
zend_accel_move_user_functions(CG(function_table), &new_persistent_script->script);
|
|
new_persistent_script->script.first_early_binding_opline =
|
|
(op_array->fn_flags & ZEND_ACC_EARLY_BINDING) ?
|
|
zend_build_delayed_early_binding_list(op_array) :
|
|
(uint32_t)-1;
|
|
|
|
efree(op_array); /* we have valid persistent_script, so it's safe to free op_array */
|
|
|
|
/* Fill in the ping_auto_globals_mask for the new script. If jit for auto globals is enabled we
|
|
will have to ping the used auto global variables before execution */
|
|
if (PG(auto_globals_jit)) {
|
|
new_persistent_script->ping_auto_globals_mask = zend_accel_get_auto_globals();
|
|
} else {
|
|
new_persistent_script->ping_auto_globals_mask = zend_accel_get_auto_globals_no_jit();
|
|
}
|
|
|
|
if (ZCG(accel_directives).validate_timestamps) {
|
|
/* Obtain the file timestamps, *before* actually compiling them,
|
|
* otherwise we have a race-condition.
|
|
*/
|
|
new_persistent_script->timestamp = timestamp;
|
|
new_persistent_script->dynamic_members.revalidate = ZCG(request_time) + ZCG(accel_directives).revalidate_freq;
|
|
}
|
|
|
|
if (file_handle->opened_path) {
|
|
new_persistent_script->script.filename = zend_string_copy(file_handle->opened_path);
|
|
} else {
|
|
new_persistent_script->script.filename = zend_string_init(file_handle->filename, strlen(file_handle->filename), 0);
|
|
}
|
|
zend_string_hash_val(new_persistent_script->script.filename);
|
|
|
|
/* Now persistent_script structure is ready in process memory */
|
|
return new_persistent_script;
|
|
}
|
|
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
zend_op_array *file_cache_compile_file(zend_file_handle *file_handle, int type)
|
|
{
|
|
zend_persistent_script *persistent_script;
|
|
zend_op_array *op_array = NULL;
|
|
int from_memory; /* if the script we've got is stored in SHM */
|
|
|
|
if (is_stream_path(file_handle->filename) &&
|
|
!is_cacheable_stream_path(file_handle->filename)) {
|
|
return accelerator_orig_compile_file(file_handle, type);
|
|
}
|
|
|
|
if (!file_handle->opened_path) {
|
|
if (file_handle->type == ZEND_HANDLE_FILENAME &&
|
|
accelerator_orig_zend_stream_open_function(file_handle->filename, file_handle) == FAILURE) {
|
|
if (type == ZEND_REQUIRE) {
|
|
zend_message_dispatcher(ZMSG_FAILED_REQUIRE_FOPEN, file_handle->filename);
|
|
zend_bailout();
|
|
} else {
|
|
zend_message_dispatcher(ZMSG_FAILED_INCLUDE_FOPEN, file_handle->filename);
|
|
}
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
|
SHM_UNPROTECT();
|
|
persistent_script = zend_file_cache_script_load(file_handle);
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
if (persistent_script) {
|
|
/* see bug #15471 (old BTS) */
|
|
if (persistent_script->script.filename) {
|
|
if (!EG(current_execute_data) || !EG(current_execute_data)->opline ||
|
|
!EG(current_execute_data)->func ||
|
|
!ZEND_USER_CODE(EG(current_execute_data)->func->common.type) ||
|
|
EG(current_execute_data)->opline->opcode != ZEND_INCLUDE_OR_EVAL ||
|
|
(EG(current_execute_data)->opline->extended_value != ZEND_INCLUDE_ONCE &&
|
|
EG(current_execute_data)->opline->extended_value != ZEND_REQUIRE_ONCE)) {
|
|
if (zend_hash_add_empty_element(&EG(included_files), persistent_script->script.filename) != NULL) {
|
|
/* ext/phar has to load phar's metadata into memory */
|
|
if (persistent_script->is_phar) {
|
|
php_stream_statbuf ssb;
|
|
char *fname = emalloc(sizeof("phar://") + ZSTR_LEN(persistent_script->script.filename));
|
|
|
|
memcpy(fname, "phar://", sizeof("phar://") - 1);
|
|
memcpy(fname + sizeof("phar://") - 1, ZSTR_VAL(persistent_script->script.filename), ZSTR_LEN(persistent_script->script.filename) + 1);
|
|
php_stream_stat_path(fname, &ssb);
|
|
efree(fname);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
zend_file_handle_dtor(file_handle);
|
|
|
|
if (persistent_script->ping_auto_globals_mask) {
|
|
zend_accel_set_auto_globals(persistent_script->ping_auto_globals_mask);
|
|
}
|
|
|
|
return zend_accel_load_script(persistent_script, 1);
|
|
}
|
|
|
|
persistent_script = opcache_compile_file(file_handle, type, NULL, &op_array);
|
|
|
|
if (persistent_script) {
|
|
from_memory = 0;
|
|
persistent_script = cache_script_in_file_cache(persistent_script, &from_memory);
|
|
return zend_accel_load_script(persistent_script, from_memory);
|
|
}
|
|
|
|
return op_array;
|
|
}
|
|
#endif
|
|
|
|
/* zend_compile() replacement */
|
|
zend_op_array *persistent_compile_file(zend_file_handle *file_handle, int type)
|
|
{
|
|
zend_persistent_script *persistent_script = NULL;
|
|
char *key = NULL;
|
|
int key_length;
|
|
int from_shared_memory; /* if the script we've got is stored in SHM */
|
|
|
|
if (!file_handle->filename || !ZCG(enabled) || !accel_startup_ok) {
|
|
/* The Accelerator is disabled, act as if without the Accelerator */
|
|
return accelerator_orig_compile_file(file_handle, type);
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
} else if (file_cache_only) {
|
|
return file_cache_compile_file(file_handle, type);
|
|
#endif
|
|
} else if ((!ZCG(counted) && !ZCSG(accelerator_enabled)) ||
|
|
(ZCSG(restart_in_progress) && accel_restart_is_active())) {
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
if (ZCG(accel_directives).file_cache) {
|
|
return file_cache_compile_file(file_handle, type);
|
|
}
|
|
#endif
|
|
return accelerator_orig_compile_file(file_handle, type);
|
|
}
|
|
|
|
/* In case this callback is called from include_once, require_once or it's
|
|
* a main FastCGI request, the key must be already calculated, and cached
|
|
* persistent script already found */
|
|
if (ZCG(cache_persistent_script) &&
|
|
((!EG(current_execute_data) &&
|
|
file_handle->filename == SG(request_info).path_translated &&
|
|
ZCG(cache_opline) == NULL) ||
|
|
(EG(current_execute_data) &&
|
|
EG(current_execute_data)->func &&
|
|
ZEND_USER_CODE(EG(current_execute_data)->func->common.type) &&
|
|
ZCG(cache_opline) == EG(current_execute_data)->opline))) {
|
|
|
|
persistent_script = ZCG(cache_persistent_script);
|
|
if (ZCG(key_len)) {
|
|
key = ZCG(key);
|
|
key_length = ZCG(key_len);
|
|
}
|
|
|
|
} else {
|
|
if (!ZCG(accel_directives).revalidate_path) {
|
|
/* try to find cached script by key */
|
|
key = accel_make_persistent_key(file_handle->filename, strlen(file_handle->filename), &key_length);
|
|
if (!key) {
|
|
return accelerator_orig_compile_file(file_handle, type);
|
|
}
|
|
persistent_script = zend_accel_hash_str_find(&ZCSG(hash), key, key_length);
|
|
} else if (UNEXPECTED(is_stream_path(file_handle->filename) && !is_cacheable_stream_path(file_handle->filename))) {
|
|
return accelerator_orig_compile_file(file_handle, type);
|
|
}
|
|
|
|
if (!persistent_script) {
|
|
/* try to find cached script by full real path */
|
|
zend_accel_hash_entry *bucket;
|
|
|
|
/* open file to resolve the path */
|
|
if (file_handle->type == ZEND_HANDLE_FILENAME &&
|
|
accelerator_orig_zend_stream_open_function(file_handle->filename, file_handle) == FAILURE) {
|
|
if (type == ZEND_REQUIRE) {
|
|
zend_message_dispatcher(ZMSG_FAILED_REQUIRE_FOPEN, file_handle->filename);
|
|
zend_bailout();
|
|
} else {
|
|
zend_message_dispatcher(ZMSG_FAILED_INCLUDE_FOPEN, file_handle->filename);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
if (file_handle->opened_path) {
|
|
bucket = zend_accel_hash_find_entry(&ZCSG(hash), file_handle->opened_path);
|
|
|
|
if (bucket) {
|
|
persistent_script = (zend_persistent_script *)bucket->data;
|
|
|
|
if (key && !persistent_script->corrupted) {
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
|
SHM_UNPROTECT();
|
|
zend_shared_alloc_lock();
|
|
zend_accel_add_key(key, key_length, bucket);
|
|
zend_shared_alloc_unlock();
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* clear cache */
|
|
ZCG(cache_opline) = NULL;
|
|
ZCG(cache_persistent_script) = NULL;
|
|
|
|
if (persistent_script && persistent_script->corrupted) {
|
|
persistent_script = NULL;
|
|
}
|
|
|
|
/* Make sure we only increase the currently running processes semaphore
|
|
* once each execution (this function can be called more than once on
|
|
* each execution)
|
|
*/
|
|
if (!ZCG(counted)) {
|
|
if (accel_activate_add() == FAILURE) {
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
if (ZCG(accel_directives).file_cache) {
|
|
return file_cache_compile_file(file_handle, type);
|
|
}
|
|
#endif
|
|
return accelerator_orig_compile_file(file_handle, type);
|
|
}
|
|
ZCG(counted) = 1;
|
|
}
|
|
|
|
/* Revalidate accessibility of cached file */
|
|
if (EXPECTED(persistent_script != NULL) &&
|
|
UNEXPECTED(ZCG(accel_directives).validate_permission) &&
|
|
file_handle->type == ZEND_HANDLE_FILENAME &&
|
|
UNEXPECTED(access(ZSTR_VAL(persistent_script->script.filename), R_OK) != 0)) {
|
|
if (type == ZEND_REQUIRE) {
|
|
zend_message_dispatcher(ZMSG_FAILED_REQUIRE_FOPEN, file_handle->filename);
|
|
zend_bailout();
|
|
} else {
|
|
zend_message_dispatcher(ZMSG_FAILED_INCLUDE_FOPEN, file_handle->filename);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
|
SHM_UNPROTECT();
|
|
|
|
/* If script is found then validate_timestamps if option is enabled */
|
|
if (persistent_script && ZCG(accel_directives).validate_timestamps) {
|
|
if (validate_timestamp_and_record(persistent_script, file_handle) == FAILURE) {
|
|
zend_shared_alloc_lock();
|
|
if (!persistent_script->corrupted) {
|
|
persistent_script->corrupted = 1;
|
|
persistent_script->timestamp = 0;
|
|
ZSMMG(wasted_shared_memory) += persistent_script->dynamic_members.memory_consumption;
|
|
if (ZSMMG(memory_exhausted)) {
|
|
zend_accel_restart_reason reason =
|
|
zend_accel_hash_is_full(&ZCSG(hash)) ? ACCEL_RESTART_HASH : ACCEL_RESTART_OOM;
|
|
zend_accel_schedule_restart_if_necessary(reason);
|
|
}
|
|
}
|
|
zend_shared_alloc_unlock();
|
|
persistent_script = NULL;
|
|
}
|
|
}
|
|
|
|
/* if turned on - check the compiled script ADLER32 checksum */
|
|
if (persistent_script && ZCG(accel_directives).consistency_checks
|
|
&& persistent_script->dynamic_members.hits % ZCG(accel_directives).consistency_checks == 0) {
|
|
|
|
unsigned int checksum = zend_accel_script_checksum(persistent_script);
|
|
if (checksum != persistent_script->dynamic_members.checksum ) {
|
|
/* The checksum is wrong */
|
|
zend_accel_error(ACCEL_LOG_INFO, "Checksum failed for '%s': expected=0x%08x, found=0x%08x",
|
|
ZSTR_VAL(persistent_script->script.filename), persistent_script->dynamic_members.checksum, checksum);
|
|
zend_shared_alloc_lock();
|
|
if (!persistent_script->corrupted) {
|
|
persistent_script->corrupted = 1;
|
|
persistent_script->timestamp = 0;
|
|
ZSMMG(wasted_shared_memory) += persistent_script->dynamic_members.memory_consumption;
|
|
if (ZSMMG(memory_exhausted)) {
|
|
zend_accel_restart_reason reason =
|
|
zend_accel_hash_is_full(&ZCSG(hash)) ? ACCEL_RESTART_HASH : ACCEL_RESTART_OOM;
|
|
zend_accel_schedule_restart_if_necessary(reason);
|
|
}
|
|
}
|
|
zend_shared_alloc_unlock();
|
|
persistent_script = NULL;
|
|
}
|
|
}
|
|
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
/* Check the second level cache */
|
|
if (!persistent_script && ZCG(accel_directives).file_cache) {
|
|
persistent_script = zend_file_cache_script_load(file_handle);
|
|
}
|
|
#endif
|
|
|
|
/* If script was not found or invalidated by validate_timestamps */
|
|
if (!persistent_script) {
|
|
uint32_t old_const_num = zend_hash_next_free_element(EG(zend_constants));
|
|
zend_op_array *op_array;
|
|
|
|
/* Cache miss.. */
|
|
ZCSG(misses)++;
|
|
|
|
/* No memory left. Behave like without the Accelerator */
|
|
if (ZSMMG(memory_exhausted) || ZCSG(restart_pending)) {
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
if (ZCG(accel_directives).file_cache) {
|
|
return file_cache_compile_file(file_handle, type);
|
|
}
|
|
#endif
|
|
return accelerator_orig_compile_file(file_handle, type);
|
|
}
|
|
|
|
/* Try and cache the script and assume that it is returned from_shared_memory.
|
|
* If it isn't compile_and_cache_file() changes the flag to 0
|
|
*/
|
|
from_shared_memory = 0;
|
|
persistent_script = opcache_compile_file(file_handle, type, key, &op_array);
|
|
if (persistent_script) {
|
|
persistent_script = cache_script_in_shared_memory(persistent_script, key, key ? key_length : 0, &from_shared_memory);
|
|
}
|
|
|
|
/* Caching is disabled, returning op_array;
|
|
* or something went wrong during compilation, returning NULL
|
|
*/
|
|
if (!persistent_script) {
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
return op_array;
|
|
}
|
|
if (from_shared_memory) {
|
|
/* Delete immutable arrays moved into SHM */
|
|
uint32_t new_const_num = zend_hash_next_free_element(EG(zend_constants));
|
|
while (new_const_num > old_const_num) {
|
|
new_const_num--;
|
|
zend_hash_index_del(EG(zend_constants), new_const_num);
|
|
}
|
|
}
|
|
} else {
|
|
|
|
#if !ZEND_WIN32
|
|
ZCSG(hits)++; /* TBFixed: may lose one hit */
|
|
persistent_script->dynamic_members.hits++; /* see above */
|
|
#else
|
|
#ifdef _M_X64
|
|
InterlockedIncrement64(&ZCSG(hits));
|
|
#else
|
|
InterlockedIncrement(&ZCSG(hits));
|
|
#endif
|
|
InterlockedIncrement64(&persistent_script->dynamic_members.hits);
|
|
#endif
|
|
|
|
/* see bug #15471 (old BTS) */
|
|
if (persistent_script->script.filename) {
|
|
if (!EG(current_execute_data) || !EG(current_execute_data)->opline ||
|
|
!EG(current_execute_data)->func ||
|
|
!ZEND_USER_CODE(EG(current_execute_data)->func->common.type) ||
|
|
EG(current_execute_data)->opline->opcode != ZEND_INCLUDE_OR_EVAL ||
|
|
(EG(current_execute_data)->opline->extended_value != ZEND_INCLUDE_ONCE &&
|
|
EG(current_execute_data)->opline->extended_value != ZEND_REQUIRE_ONCE)) {
|
|
if (zend_hash_add_empty_element(&EG(included_files), persistent_script->script.filename) != NULL) {
|
|
/* ext/phar has to load phar's metadata into memory */
|
|
if (persistent_script->is_phar) {
|
|
php_stream_statbuf ssb;
|
|
char *fname = emalloc(sizeof("phar://") + ZSTR_LEN(persistent_script->script.filename));
|
|
|
|
memcpy(fname, "phar://", sizeof("phar://") - 1);
|
|
memcpy(fname + sizeof("phar://") - 1, ZSTR_VAL(persistent_script->script.filename), ZSTR_LEN(persistent_script->script.filename) + 1);
|
|
php_stream_stat_path(fname, &ssb);
|
|
efree(fname);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
zend_file_handle_dtor(file_handle);
|
|
from_shared_memory = 1;
|
|
}
|
|
|
|
persistent_script->dynamic_members.last_used = ZCG(request_time);
|
|
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
|
|
/* Fetch jit auto globals used in the script before execution */
|
|
if (persistent_script->ping_auto_globals_mask) {
|
|
zend_accel_set_auto_globals(persistent_script->ping_auto_globals_mask);
|
|
}
|
|
|
|
return zend_accel_load_script(persistent_script, from_shared_memory);
|
|
}
|
|
|
|
/* zend_stream_open_function() replacement for PHP 5.3 and above */
|
|
static int persistent_stream_open_function(const char *filename, zend_file_handle *handle)
|
|
{
|
|
if (ZCG(cache_persistent_script)) {
|
|
/* check if callback is called from include_once or it's a main request */
|
|
if ((!EG(current_execute_data) &&
|
|
filename == SG(request_info).path_translated &&
|
|
ZCG(cache_opline) == NULL) ||
|
|
(EG(current_execute_data) &&
|
|
EG(current_execute_data)->func &&
|
|
ZEND_USER_CODE(EG(current_execute_data)->func->common.type) &&
|
|
ZCG(cache_opline) == EG(current_execute_data)->opline)) {
|
|
|
|
/* we are in include_once or FastCGI request */
|
|
handle->filename = (char*)filename;
|
|
handle->free_filename = 0;
|
|
handle->opened_path = zend_string_copy(ZCG(cache_persistent_script)->script.filename);
|
|
handle->type = ZEND_HANDLE_FILENAME;
|
|
return SUCCESS;
|
|
}
|
|
ZCG(cache_opline) = NULL;
|
|
ZCG(cache_persistent_script) = NULL;
|
|
}
|
|
return accelerator_orig_zend_stream_open_function(filename, handle);
|
|
}
|
|
|
|
/* zend_resolve_path() replacement for PHP 5.3 and above */
|
|
static zend_string* persistent_zend_resolve_path(const char *filename, size_t filename_len)
|
|
{
|
|
if (ZCG(enabled) && accel_startup_ok &&
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
!file_cache_only &&
|
|
#endif
|
|
(ZCG(counted) || ZCSG(accelerator_enabled)) &&
|
|
!ZCSG(restart_in_progress)) {
|
|
|
|
/* check if callback is called from include_once or it's a main request */
|
|
if ((!EG(current_execute_data) &&
|
|
filename == SG(request_info).path_translated) ||
|
|
(EG(current_execute_data) &&
|
|
EG(current_execute_data)->func &&
|
|
ZEND_USER_CODE(EG(current_execute_data)->func->common.type) &&
|
|
EG(current_execute_data)->opline->opcode == ZEND_INCLUDE_OR_EVAL &&
|
|
(EG(current_execute_data)->opline->extended_value == ZEND_INCLUDE_ONCE ||
|
|
EG(current_execute_data)->opline->extended_value == ZEND_REQUIRE_ONCE))) {
|
|
|
|
/* we are in include_once or FastCGI request */
|
|
zend_string *resolved_path;
|
|
int key_length;
|
|
char *key = NULL;
|
|
|
|
if (!ZCG(accel_directives).revalidate_path) {
|
|
/* lookup by "not-real" path */
|
|
key = accel_make_persistent_key(filename, filename_len, &key_length);
|
|
if (key) {
|
|
zend_accel_hash_entry *bucket = zend_accel_hash_str_find_entry(&ZCSG(hash), key, key_length);
|
|
if (bucket != NULL) {
|
|
zend_persistent_script *persistent_script = (zend_persistent_script *)bucket->data;
|
|
if (!persistent_script->corrupted) {
|
|
ZCG(cache_opline) = EG(current_execute_data) ? EG(current_execute_data)->opline : NULL;
|
|
ZCG(cache_persistent_script) = persistent_script;
|
|
return zend_string_copy(persistent_script->script.filename);
|
|
}
|
|
}
|
|
} else {
|
|
ZCG(cache_opline) = NULL;
|
|
ZCG(cache_persistent_script) = NULL;
|
|
return accelerator_orig_zend_resolve_path(filename, filename_len);
|
|
}
|
|
}
|
|
|
|
/* find the full real path */
|
|
resolved_path = accelerator_orig_zend_resolve_path(filename, filename_len);
|
|
|
|
if (resolved_path) {
|
|
/* lookup by real path */
|
|
zend_accel_hash_entry *bucket = zend_accel_hash_find_entry(&ZCSG(hash), resolved_path);
|
|
if (bucket) {
|
|
zend_persistent_script *persistent_script = (zend_persistent_script *)bucket->data;
|
|
if (!persistent_script->corrupted) {
|
|
if (key) {
|
|
/* add another "key" for the same bucket */
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
|
SHM_UNPROTECT();
|
|
zend_shared_alloc_lock();
|
|
zend_accel_add_key(key, key_length, bucket);
|
|
zend_shared_alloc_unlock();
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
} else {
|
|
ZCG(key_len) = 0;
|
|
}
|
|
ZCG(cache_opline) = EG(current_execute_data) ? EG(current_execute_data)->opline : NULL;
|
|
ZCG(cache_persistent_script) = persistent_script;
|
|
return resolved_path;
|
|
}
|
|
}
|
|
}
|
|
|
|
ZCG(cache_opline) = NULL;
|
|
ZCG(cache_persistent_script) = NULL;
|
|
return resolved_path;
|
|
}
|
|
}
|
|
ZCG(cache_opline) = NULL;
|
|
ZCG(cache_persistent_script) = NULL;
|
|
return accelerator_orig_zend_resolve_path(filename, filename_len);
|
|
}
|
|
|
|
static void zend_reset_cache_vars(void)
|
|
{
|
|
ZSMMG(memory_exhausted) = 0;
|
|
ZCSG(hits) = 0;
|
|
ZCSG(misses) = 0;
|
|
ZCSG(blacklist_misses) = 0;
|
|
ZSMMG(wasted_shared_memory) = 0;
|
|
ZCSG(restart_pending) = 0;
|
|
ZCSG(force_restart_time) = 0;
|
|
ZCSG(map_ptr_last) = CG(map_ptr_last);
|
|
}
|
|
|
|
static void accel_reset_pcre_cache(void)
|
|
{
|
|
Bucket *p;
|
|
|
|
ZEND_HASH_FOREACH_BUCKET(&PCRE_G(pcre_cache), p) {
|
|
/* Remove PCRE cache entries with inconsistent keys */
|
|
if (zend_accel_in_shm(p->key)) {
|
|
p->key = NULL;
|
|
zend_hash_del_bucket(&PCRE_G(pcre_cache), p);
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
}
|
|
|
|
int accel_activate(INIT_FUNC_ARGS)
|
|
{
|
|
zend_bool reset_pcre = 0;
|
|
|
|
if (!ZCG(enabled) || !accel_startup_ok) {
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* PHP-5.4 and above return "double", but we use 1 sec precision */
|
|
ZCG(auto_globals_mask) = 0;
|
|
ZCG(request_time) = (time_t)sapi_get_request_time();
|
|
ZCG(cache_opline) = NULL;
|
|
ZCG(cache_persistent_script) = NULL;
|
|
ZCG(include_path_key_len) = 0;
|
|
ZCG(include_path_check) = 1;
|
|
|
|
ZCG(cwd) = NULL;
|
|
ZCG(cwd_key_len) = 0;
|
|
ZCG(cwd_check) = 1;
|
|
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
if (file_cache_only) {
|
|
return SUCCESS;
|
|
}
|
|
#endif
|
|
|
|
#ifndef ZEND_WIN32
|
|
if (ZCG(accel_directives).validate_root) {
|
|
struct stat buf;
|
|
|
|
if (stat("/", &buf) != 0) {
|
|
ZCG(root_hash) = 0;
|
|
} else {
|
|
ZCG(root_hash) = buf.st_ino;
|
|
if (sizeof(buf.st_ino) > sizeof(ZCG(root_hash))) {
|
|
if (ZCG(root_hash) != buf.st_ino) {
|
|
zend_string *key = zend_string_init("opcache.enable", sizeof("opcache.enable")-1, 0);
|
|
zend_alter_ini_entry_chars(key, "0", 1, ZEND_INI_SYSTEM, ZEND_INI_STAGE_RUNTIME);
|
|
zend_string_release_ex(key, 0);
|
|
zend_accel_error(ACCEL_LOG_WARNING, "Can't cache files in chroot() directory with too big inode");
|
|
return SUCCESS;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
ZCG(root_hash) = 0;
|
|
}
|
|
#endif
|
|
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
|
SHM_UNPROTECT();
|
|
|
|
if (ZCG(counted)) {
|
|
#ifdef ZTS
|
|
zend_accel_error(ACCEL_LOG_WARNING, "Stuck count for thread id %lu", (unsigned long) tsrm_thread_id());
|
|
#else
|
|
zend_accel_error(ACCEL_LOG_WARNING, "Stuck count for pid %d", getpid());
|
|
#endif
|
|
accel_unlock_all();
|
|
ZCG(counted) = 0;
|
|
}
|
|
|
|
if (ZCSG(restart_pending)) {
|
|
zend_shared_alloc_lock();
|
|
if (ZCSG(restart_pending) != 0) { /* check again, to ensure that the cache wasn't already cleaned by another process */
|
|
if (accel_is_inactive() == SUCCESS) {
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "Restarting!");
|
|
ZCSG(restart_pending) = 0;
|
|
switch ZCSG(restart_reason) {
|
|
case ACCEL_RESTART_OOM:
|
|
ZCSG(oom_restarts)++;
|
|
break;
|
|
case ACCEL_RESTART_HASH:
|
|
ZCSG(hash_restarts)++;
|
|
break;
|
|
case ACCEL_RESTART_USER:
|
|
ZCSG(manual_restarts)++;
|
|
break;
|
|
}
|
|
accel_restart_enter();
|
|
|
|
zend_map_ptr_reset();
|
|
zend_reset_cache_vars();
|
|
zend_accel_hash_clean(&ZCSG(hash));
|
|
|
|
if (ZCG(accel_directives).interned_strings_buffer) {
|
|
accel_interned_strings_restore_state();
|
|
}
|
|
|
|
zend_shared_alloc_restore_state();
|
|
if (ZCSG(preload_script)) {
|
|
preload_restart();
|
|
}
|
|
ZCSG(accelerator_enabled) = ZCSG(cache_status_before_restart);
|
|
if (ZCSG(last_restart_time) < ZCG(request_time)) {
|
|
ZCSG(last_restart_time) = ZCG(request_time);
|
|
} else {
|
|
ZCSG(last_restart_time)++;
|
|
}
|
|
accel_restart_leave();
|
|
}
|
|
} else {
|
|
reset_pcre = 1;
|
|
}
|
|
zend_shared_alloc_unlock();
|
|
}
|
|
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
|
|
if (ZCSG(last_restart_time) != ZCG(last_restart_time)) {
|
|
/* SHM was reinitialized. */
|
|
ZCG(last_restart_time) = ZCSG(last_restart_time);
|
|
|
|
/* Reset in-process realpath cache */
|
|
realpath_cache_clean();
|
|
|
|
accel_reset_pcre_cache();
|
|
} else if (reset_pcre) {
|
|
accel_reset_pcre_cache();
|
|
}
|
|
|
|
if (ZCSG(preload_script)) {
|
|
preload_activate();
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
int accel_post_deactivate(void)
|
|
{
|
|
if (ZCG(cwd)) {
|
|
zend_string_release_ex(ZCG(cwd), 0);
|
|
ZCG(cwd) = NULL;
|
|
}
|
|
|
|
if (!ZCG(enabled) || !accel_startup_ok) {
|
|
return SUCCESS;
|
|
}
|
|
|
|
zend_shared_alloc_safe_unlock(); /* be sure we didn't leave cache locked */
|
|
accel_unlock_all();
|
|
ZCG(counted) = 0;
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
static int accelerator_remove_cb(zend_extension *element1, zend_extension *element2)
|
|
{
|
|
(void)element2; /* keep the compiler happy */
|
|
|
|
if (!strcmp(element1->name, ACCELERATOR_PRODUCT_NAME )) {
|
|
element1->startup = NULL;
|
|
#if 0
|
|
/* We have to call shutdown callback it to free TS resources */
|
|
element1->shutdown = NULL;
|
|
#endif
|
|
element1->activate = NULL;
|
|
element1->deactivate = NULL;
|
|
element1->op_array_handler = NULL;
|
|
|
|
#ifdef __DEBUG_MESSAGES__
|
|
fprintf(stderr, ACCELERATOR_PRODUCT_NAME " is disabled: %s\n", (zps_failure_reason ? zps_failure_reason : "unknown error"));
|
|
fflush(stderr);
|
|
#endif
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void zps_startup_failure(char *reason, char *api_reason, int (*cb)(zend_extension *, zend_extension *))
|
|
{
|
|
accel_startup_ok = 0;
|
|
zps_failure_reason = reason;
|
|
zps_api_failure_reason = api_reason?api_reason:reason;
|
|
zend_llist_del_element(&zend_extensions, NULL, (int (*)(void *, void *))cb);
|
|
}
|
|
|
|
static inline int accel_find_sapi(void)
|
|
{
|
|
static const char *supported_sapis[] = {
|
|
"apache",
|
|
"fastcgi",
|
|
"cli-server",
|
|
"cgi-fcgi",
|
|
"fpm-fcgi",
|
|
"apache2handler",
|
|
"litespeed",
|
|
"uwsgi",
|
|
NULL
|
|
};
|
|
const char **sapi_name;
|
|
|
|
if (sapi_module.name) {
|
|
for (sapi_name = supported_sapis; *sapi_name; sapi_name++) {
|
|
if (strcmp(sapi_module.name, *sapi_name) == 0) {
|
|
return SUCCESS;
|
|
}
|
|
}
|
|
if (ZCG(accel_directives).enable_cli && (
|
|
strcmp(sapi_module.name, "cli") == 0
|
|
|| strcmp(sapi_module.name, "phpdbg") == 0)) {
|
|
return SUCCESS;
|
|
}
|
|
}
|
|
|
|
return FAILURE;
|
|
}
|
|
|
|
static int zend_accel_init_shm(void)
|
|
{
|
|
int i;
|
|
|
|
zend_shared_alloc_lock();
|
|
|
|
if (ZCG(accel_directives).interned_strings_buffer) {
|
|
accel_shared_globals = zend_shared_alloc((ZCG(accel_directives).interned_strings_buffer * 1024 * 1024));
|
|
} else {
|
|
accel_shared_globals = zend_shared_alloc(sizeof(zend_accel_shared_globals));
|
|
}
|
|
if (!accel_shared_globals) {
|
|
zend_accel_error(ACCEL_LOG_FATAL, "Insufficient shared memory!");
|
|
zend_shared_alloc_unlock();
|
|
return FAILURE;
|
|
}
|
|
memset(accel_shared_globals, 0, sizeof(zend_accel_shared_globals));
|
|
ZSMMG(app_shared_globals) = accel_shared_globals;
|
|
|
|
zend_accel_hash_init(&ZCSG(hash), ZCG(accel_directives).max_accelerated_files);
|
|
|
|
if (ZCG(accel_directives).interned_strings_buffer) {
|
|
uint32_t hash_size;
|
|
|
|
/* must be a power of two */
|
|
hash_size = ZCG(accel_directives).interned_strings_buffer * (32 * 1024);
|
|
hash_size |= (hash_size >> 1);
|
|
hash_size |= (hash_size >> 2);
|
|
hash_size |= (hash_size >> 4);
|
|
hash_size |= (hash_size >> 8);
|
|
hash_size |= (hash_size >> 16);
|
|
|
|
ZCSG(interned_strings).nTableMask = hash_size << 2;
|
|
ZCSG(interned_strings).nNumOfElements = 0;
|
|
ZCSG(interned_strings).start =
|
|
(zend_string*)((char*)&ZCSG(interned_strings) +
|
|
sizeof(zend_string_table) +
|
|
((hash_size + 1) * sizeof(uint32_t))) +
|
|
8;
|
|
ZCSG(interned_strings).top =
|
|
ZCSG(interned_strings).start;
|
|
ZCSG(interned_strings).end =
|
|
(zend_string*)((char*)accel_shared_globals +
|
|
ZCG(accel_directives).interned_strings_buffer * 1024 * 1024);
|
|
ZCSG(interned_strings).saved_top = NULL;
|
|
|
|
memset((char*)&ZCSG(interned_strings) + sizeof(zend_string_table),
|
|
STRTAB_INVALID_POS,
|
|
(char*)ZCSG(interned_strings).start -
|
|
((char*)&ZCSG(interned_strings) + sizeof(zend_string_table)));
|
|
}
|
|
|
|
zend_interned_strings_set_request_storage_handlers(accel_new_interned_string_for_php, accel_init_interned_string_for_php);
|
|
|
|
zend_reset_cache_vars();
|
|
|
|
ZCSG(oom_restarts) = 0;
|
|
ZCSG(hash_restarts) = 0;
|
|
ZCSG(manual_restarts) = 0;
|
|
|
|
ZCSG(accelerator_enabled) = 1;
|
|
ZCSG(start_time) = zend_accel_get_time();
|
|
ZCSG(last_restart_time) = 0;
|
|
ZCSG(restart_in_progress) = 0;
|
|
|
|
for (i = 0; i < -HT_MIN_MASK; i++) {
|
|
ZCSG(uninitialized_bucket)[i] = HT_INVALID_IDX;
|
|
}
|
|
|
|
zend_shared_alloc_unlock();
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
static void accel_globals_ctor(zend_accel_globals *accel_globals)
|
|
{
|
|
#if defined(COMPILE_DL_OPCACHE) && defined(ZTS)
|
|
ZEND_TSRMLS_CACHE_UPDATE();
|
|
#endif
|
|
memset(accel_globals, 0, sizeof(zend_accel_globals));
|
|
|
|
/* TODO refactor to init this just once. */
|
|
accel_gen_system_id();
|
|
}
|
|
|
|
#define ZEND_BIN_ID "BIN_" ZEND_TOSTR(SIZEOF_CHAR) ZEND_TOSTR(SIZEOF_INT) ZEND_TOSTR(SIZEOF_LONG) ZEND_TOSTR(SIZEOF_SIZE_T) ZEND_TOSTR(SIZEOF_ZEND_LONG) ZEND_TOSTR(ZEND_MM_ALIGNMENT)
|
|
|
|
static void accel_gen_system_id(void)
|
|
{
|
|
PHP_MD5_CTX context;
|
|
unsigned char digest[16], c;
|
|
char *md5str = ZCG(system_id);
|
|
int i;
|
|
|
|
PHP_MD5Init(&context);
|
|
PHP_MD5Update(&context, PHP_VERSION, sizeof(PHP_VERSION)-1);
|
|
PHP_MD5Update(&context, ZEND_EXTENSION_BUILD_ID, sizeof(ZEND_EXTENSION_BUILD_ID)-1);
|
|
PHP_MD5Update(&context, ZEND_BIN_ID, sizeof(ZEND_BIN_ID)-1);
|
|
if (strstr(PHP_VERSION, "-dev") != 0) {
|
|
/* Development versions may be changed from build to build */
|
|
PHP_MD5Update(&context, __DATE__, sizeof(__DATE__)-1);
|
|
PHP_MD5Update(&context, __TIME__, sizeof(__TIME__)-1);
|
|
}
|
|
PHP_MD5Final(digest, &context);
|
|
for (i = 0; i < 16; i++) {
|
|
c = digest[i] >> 4;
|
|
c = (c <= 9) ? c + '0' : c - 10 + 'a';
|
|
md5str[i * 2] = c;
|
|
c = digest[i] & 0x0f;
|
|
c = (c <= 9) ? c + '0' : c - 10 + 'a';
|
|
md5str[(i * 2) + 1] = c;
|
|
}
|
|
}
|
|
|
|
#ifdef HAVE_HUGE_CODE_PAGES
|
|
# ifndef _WIN32
|
|
# include <sys/mman.h>
|
|
# ifndef MAP_ANON
|
|
# ifdef MAP_ANONYMOUS
|
|
# define MAP_ANON MAP_ANONYMOUS
|
|
# endif
|
|
# endif
|
|
# ifndef MAP_FAILED
|
|
# define MAP_FAILED ((void*)-1)
|
|
# endif
|
|
# endif
|
|
|
|
# if defined(MAP_HUGETLB) || defined(MADV_HUGEPAGE)
|
|
static int accel_remap_huge_pages(void *start, size_t size, const char *name, size_t offset)
|
|
{
|
|
void *ret = MAP_FAILED;
|
|
void *mem;
|
|
|
|
mem = mmap(NULL, size,
|
|
PROT_READ | PROT_WRITE,
|
|
MAP_PRIVATE | MAP_ANONYMOUS,
|
|
-1, 0);
|
|
if (mem == MAP_FAILED) {
|
|
zend_error(E_WARNING,
|
|
ACCELERATOR_PRODUCT_NAME " huge_code_pages: mmap failed: %s (%d)",
|
|
strerror(errno), errno);
|
|
return -1;
|
|
}
|
|
memcpy(mem, start, size);
|
|
|
|
# ifdef MAP_HUGETLB
|
|
ret = mmap(start, size,
|
|
PROT_READ | PROT_WRITE | PROT_EXEC,
|
|
MAP_PRIVATE | MAP_ANONYMOUS | MAP_FIXED | MAP_HUGETLB,
|
|
-1, 0);
|
|
# endif
|
|
if (ret == MAP_FAILED) {
|
|
ret = mmap(start, size,
|
|
PROT_READ | PROT_WRITE | PROT_EXEC,
|
|
MAP_PRIVATE | MAP_ANONYMOUS | MAP_FIXED,
|
|
-1, 0);
|
|
/* this should never happen? */
|
|
ZEND_ASSERT(ret != MAP_FAILED);
|
|
# ifdef MADV_HUGEPAGE
|
|
if (-1 == madvise(start, size, MADV_HUGEPAGE)) {
|
|
memcpy(start, mem, size);
|
|
mprotect(start, size, PROT_READ | PROT_EXEC);
|
|
munmap(mem, size);
|
|
zend_error(E_WARNING,
|
|
ACCELERATOR_PRODUCT_NAME " huge_code_pages: madvise(HUGEPAGE) failed: %s (%d)",
|
|
strerror(errno), errno);
|
|
return -1;
|
|
}
|
|
# else
|
|
memcpy(start, mem, size);
|
|
mprotect(start, size, PROT_READ | PROT_EXEC);
|
|
munmap(mem, size);
|
|
zend_error(E_WARNING,
|
|
ACCELERATOR_PRODUCT_NAME " huge_code_pages: mmap(HUGETLB) failed: %s (%d)",
|
|
strerror(errno), errno);
|
|
return -1;
|
|
# endif
|
|
}
|
|
|
|
if (ret == start) {
|
|
memcpy(start, mem, size);
|
|
mprotect(start, size, PROT_READ | PROT_EXEC);
|
|
}
|
|
munmap(mem, size);
|
|
|
|
return (ret == start) ? 0 : -1;
|
|
}
|
|
|
|
static void accel_move_code_to_huge_pages(void)
|
|
{
|
|
FILE *f;
|
|
long unsigned int huge_page_size = 2 * 1024 * 1024;
|
|
|
|
f = fopen("/proc/self/maps", "r");
|
|
if (f) {
|
|
long unsigned int start, end, offset, inode;
|
|
char perm[5], dev[6], name[MAXPATHLEN];
|
|
int ret;
|
|
|
|
ret = fscanf(f, "%lx-%lx %4s %lx %5s %ld %s\n", &start, &end, perm, &offset, dev, &inode, name);
|
|
if (ret == 7 && perm[0] == 'r' && perm[1] == '-' && perm[2] == 'x' && name[0] == '/') {
|
|
long unsigned int seg_start = ZEND_MM_ALIGNED_SIZE_EX(start, huge_page_size);
|
|
long unsigned int seg_end = (end & ~(huge_page_size-1L));
|
|
|
|
if (seg_end > seg_start) {
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "remap to huge page %lx-%lx %s \n", seg_start, seg_end, name);
|
|
accel_remap_huge_pages((void*)seg_start, seg_end - seg_start, name, offset + seg_start - start);
|
|
}
|
|
}
|
|
fclose(f);
|
|
}
|
|
}
|
|
# else
|
|
static void accel_move_code_to_huge_pages(void)
|
|
{
|
|
zend_error(E_WARNING, ACCELERATOR_PRODUCT_NAME ": opcache.huge_code_pages has no affect as huge page is not supported");
|
|
return;
|
|
}
|
|
# endif /* defined(MAP_HUGETLB) || defined(MADV_HUGEPAGE) */
|
|
#endif /* HAVE_HUGE_CODE_PAGES */
|
|
|
|
static int accel_startup(zend_extension *extension)
|
|
{
|
|
#ifdef ZTS
|
|
accel_globals_id = ts_allocate_id(&accel_globals_id, sizeof(zend_accel_globals), (ts_allocate_ctor) accel_globals_ctor, NULL);
|
|
#else
|
|
accel_globals_ctor(&accel_globals);
|
|
#endif
|
|
|
|
#ifdef ZEND_WIN32
|
|
# if !defined(__has_feature) || !__has_feature(address_sanitizer)
|
|
_setmaxstdio(2048); /* The default configuration is limited to 512 stdio files */
|
|
# endif
|
|
#endif
|
|
|
|
if (start_accel_module() == FAILURE) {
|
|
accel_startup_ok = 0;
|
|
zend_error(E_WARNING, ACCELERATOR_PRODUCT_NAME ": module registration failed!");
|
|
return FAILURE;
|
|
}
|
|
|
|
accel_gen_system_id();
|
|
|
|
#ifdef HAVE_HUGE_CODE_PAGES
|
|
if (ZCG(accel_directives).huge_code_pages &&
|
|
(strcmp(sapi_module.name, "cli") == 0 ||
|
|
strcmp(sapi_module.name, "cli-server") == 0 ||
|
|
strcmp(sapi_module.name, "cgi-fcgi") == 0 ||
|
|
strcmp(sapi_module.name, "fpm-fcgi") == 0)) {
|
|
accel_move_code_to_huge_pages();
|
|
}
|
|
#endif
|
|
|
|
/* no supported SAPI found - disable acceleration and stop initialization */
|
|
if (accel_find_sapi() == FAILURE) {
|
|
accel_startup_ok = 0;
|
|
if (!ZCG(accel_directives).enable_cli &&
|
|
strcmp(sapi_module.name, "cli") == 0) {
|
|
zps_startup_failure("Opcode Caching is disabled for CLI", NULL, accelerator_remove_cb);
|
|
} else {
|
|
zps_startup_failure("Opcode Caching is only supported in Apache, FPM, FastCGI and LiteSpeed SAPIs", NULL, accelerator_remove_cb);
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
if (ZCG(enabled) == 0) {
|
|
return SUCCESS ;
|
|
}
|
|
|
|
orig_post_startup_cb = zend_post_startup_cb;
|
|
zend_post_startup_cb = accel_post_startup;
|
|
|
|
/* Prevent unloadig */
|
|
extension->handle = 0;
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
static int accel_post_startup(void)
|
|
{
|
|
zend_function *func;
|
|
zend_ini_entry *ini_entry;
|
|
|
|
if (orig_post_startup_cb) {
|
|
int (*cb)(void) = orig_post_startup_cb;
|
|
|
|
orig_post_startup_cb = NULL;
|
|
if (cb() != SUCCESS) {
|
|
return FAILURE;
|
|
}
|
|
}
|
|
|
|
/********************************************/
|
|
/* End of non-SHM dependent initializations */
|
|
/********************************************/
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
file_cache_only = ZCG(accel_directives).file_cache_only;
|
|
if (!file_cache_only) {
|
|
#else
|
|
if (1) {
|
|
#endif
|
|
switch (zend_shared_alloc_startup(ZCG(accel_directives).memory_consumption)) {
|
|
case ALLOC_SUCCESS:
|
|
if (zend_accel_init_shm() == FAILURE) {
|
|
accel_startup_ok = 0;
|
|
return FAILURE;
|
|
}
|
|
break;
|
|
case ALLOC_FAILURE:
|
|
accel_startup_ok = 0;
|
|
zend_accel_error(ACCEL_LOG_FATAL, "Failure to initialize shared memory structures - probably not enough shared memory.");
|
|
return SUCCESS;
|
|
case SUCCESSFULLY_REATTACHED:
|
|
zend_shared_alloc_lock();
|
|
accel_shared_globals = (zend_accel_shared_globals *) ZSMMG(app_shared_globals);
|
|
zend_interned_strings_set_request_storage_handlers(accel_new_interned_string_for_php, accel_init_interned_string_for_php);
|
|
zend_shared_alloc_unlock();
|
|
break;
|
|
case FAILED_REATTACHED:
|
|
accel_startup_ok = 0;
|
|
zend_accel_error(ACCEL_LOG_FATAL, "Failure to initialize shared memory structures - can not reattach to exiting shared memory.");
|
|
return SUCCESS;
|
|
break;
|
|
#if ENABLE_FILE_CACHE_FALLBACK
|
|
case ALLOC_FALLBACK:
|
|
zend_shared_alloc_lock();
|
|
file_cache_only = 1;
|
|
fallback_process = 1;
|
|
zend_accel_init_auto_globals();
|
|
zend_shared_alloc_unlock();
|
|
goto file_cache_fallback;
|
|
break;
|
|
#endif
|
|
}
|
|
|
|
/* from this point further, shared memory is supposed to be OK */
|
|
|
|
/* remember the last restart time in the process memory */
|
|
ZCG(last_restart_time) = ZCSG(last_restart_time);
|
|
|
|
/* Init auto-global strings */
|
|
zend_accel_init_auto_globals();
|
|
|
|
zend_shared_alloc_lock();
|
|
zend_shared_alloc_save_state();
|
|
zend_shared_alloc_unlock();
|
|
|
|
SHM_PROTECT();
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
} else if (!ZCG(accel_directives).file_cache) {
|
|
accel_startup_ok = 0;
|
|
zend_accel_error(ACCEL_LOG_FATAL, "opcache.file_cache_only is set without a proper setting of opcache.file_cache");
|
|
return SUCCESS;
|
|
} else {
|
|
accel_shared_globals = calloc(1, sizeof(zend_accel_shared_globals));
|
|
|
|
/* Init auto-global strings */
|
|
zend_accel_init_auto_globals();
|
|
#endif
|
|
}
|
|
#if ENABLE_FILE_CACHE_FALLBACK
|
|
file_cache_fallback:
|
|
#endif
|
|
|
|
/* Override compiler */
|
|
accelerator_orig_compile_file = zend_compile_file;
|
|
zend_compile_file = persistent_compile_file;
|
|
|
|
/* Override stream opener function (to eliminate open() call caused by
|
|
* include/require statements ) */
|
|
accelerator_orig_zend_stream_open_function = zend_stream_open_function;
|
|
zend_stream_open_function = persistent_stream_open_function;
|
|
|
|
/* Override path resolver function (to eliminate stat() calls caused by
|
|
* include_once/require_once statements */
|
|
accelerator_orig_zend_resolve_path = zend_resolve_path;
|
|
zend_resolve_path = persistent_zend_resolve_path;
|
|
|
|
/* Override chdir() function */
|
|
if ((func = zend_hash_str_find_ptr(CG(function_table), "chdir", sizeof("chdir")-1)) != NULL &&
|
|
func->type == ZEND_INTERNAL_FUNCTION) {
|
|
orig_chdir = func->internal_function.handler;
|
|
func->internal_function.handler = ZEND_FN(accel_chdir);
|
|
}
|
|
ZCG(cwd) = NULL;
|
|
ZCG(include_path) = NULL;
|
|
|
|
/* Override "include_path" modifier callback */
|
|
if ((ini_entry = zend_hash_str_find_ptr(EG(ini_directives), "include_path", sizeof("include_path")-1)) != NULL) {
|
|
ZCG(include_path) = ini_entry->value;
|
|
orig_include_path_on_modify = ini_entry->on_modify;
|
|
ini_entry->on_modify = accel_include_path_on_modify;
|
|
}
|
|
|
|
accel_startup_ok = 1;
|
|
|
|
/* Override file_exists(), is_file() and is_readable() */
|
|
zend_accel_override_file_functions();
|
|
|
|
/* Load black list */
|
|
accel_blacklist.entries = NULL;
|
|
if (ZCG(enabled) && accel_startup_ok &&
|
|
ZCG(accel_directives).user_blacklist_filename &&
|
|
*ZCG(accel_directives.user_blacklist_filename)) {
|
|
zend_accel_blacklist_init(&accel_blacklist);
|
|
zend_accel_blacklist_load(&accel_blacklist, ZCG(accel_directives.user_blacklist_filename));
|
|
}
|
|
|
|
zend_optimizer_startup();
|
|
|
|
if (!file_cache_only && ZCG(accel_directives).interned_strings_buffer) {
|
|
accel_use_shm_interned_strings();
|
|
}
|
|
|
|
return accel_finish_startup();
|
|
}
|
|
|
|
static void (*orig_post_shutdown_cb)(void);
|
|
|
|
static void accel_post_shutdown(void)
|
|
{
|
|
zend_shared_alloc_shutdown();
|
|
}
|
|
|
|
void accel_shutdown(void)
|
|
{
|
|
zend_ini_entry *ini_entry;
|
|
zend_bool _file_cache_only = 0;
|
|
|
|
zend_optimizer_shutdown();
|
|
|
|
zend_accel_blacklist_shutdown(&accel_blacklist);
|
|
|
|
if (!ZCG(enabled) || !accel_startup_ok) {
|
|
#ifdef ZTS
|
|
ts_free_id(accel_globals_id);
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
if (ZCSG(preload_script)) {
|
|
preload_shutdown();
|
|
}
|
|
|
|
#ifdef HAVE_OPCACHE_FILE_CACHE
|
|
_file_cache_only = file_cache_only;
|
|
#endif
|
|
|
|
accel_reset_pcre_cache();
|
|
|
|
#ifdef ZTS
|
|
ts_free_id(accel_globals_id);
|
|
#endif
|
|
|
|
if (!_file_cache_only) {
|
|
/* Delay SHM dettach */
|
|
orig_post_shutdown_cb = zend_post_shutdown_cb;
|
|
zend_post_shutdown_cb = accel_post_shutdown;
|
|
}
|
|
|
|
zend_compile_file = accelerator_orig_compile_file;
|
|
|
|
if ((ini_entry = zend_hash_str_find_ptr(EG(ini_directives), "include_path", sizeof("include_path")-1)) != NULL) {
|
|
ini_entry->on_modify = orig_include_path_on_modify;
|
|
}
|
|
}
|
|
|
|
void zend_accel_schedule_restart(zend_accel_restart_reason reason)
|
|
{
|
|
const char *zend_accel_restart_reason_text[ACCEL_RESTART_USER + 1] = {
|
|
"out of memory",
|
|
"hash overflow",
|
|
"user",
|
|
};
|
|
|
|
if (ZCSG(restart_pending)) {
|
|
/* don't schedule twice */
|
|
return;
|
|
}
|
|
zend_accel_error(ACCEL_LOG_DEBUG, "Restart Scheduled! Reason: %s",
|
|
zend_accel_restart_reason_text[reason]);
|
|
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
|
SHM_UNPROTECT();
|
|
ZCSG(restart_pending) = 1;
|
|
ZCSG(restart_reason) = reason;
|
|
ZCSG(cache_status_before_restart) = ZCSG(accelerator_enabled);
|
|
ZCSG(accelerator_enabled) = 0;
|
|
|
|
if (ZCG(accel_directives).force_restart_timeout) {
|
|
ZCSG(force_restart_time) = zend_accel_get_time() + ZCG(accel_directives).force_restart_timeout;
|
|
} else {
|
|
ZCSG(force_restart_time) = 0;
|
|
}
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
}
|
|
|
|
/* this is needed because on WIN32 lock is not decreased unless ZCG(counted) is set */
|
|
#ifdef ZEND_WIN32
|
|
#define accel_deactivate_now() ZCG(counted) = 1; accel_deactivate_sub()
|
|
#else
|
|
#define accel_deactivate_now() accel_deactivate_sub()
|
|
#endif
|
|
|
|
/* ensures it is OK to read SHM
|
|
if it's not OK (restart in progress) returns FAILURE
|
|
if OK returns SUCCESS
|
|
MUST call accelerator_shm_read_unlock after done lock operations
|
|
*/
|
|
int accelerator_shm_read_lock(void)
|
|
{
|
|
if (ZCG(counted)) {
|
|
/* counted means we are holding read lock for SHM, so that nothing bad can happen */
|
|
return SUCCESS;
|
|
} else {
|
|
/* here accelerator is active but we do not hold SHM lock. This means restart was scheduled
|
|
or is in progress now */
|
|
if (accel_activate_add() == FAILURE) { /* acquire usage lock */
|
|
return FAILURE;
|
|
}
|
|
/* Now if we weren't inside restart, restart would not begin until we remove usage lock */
|
|
if (ZCSG(restart_in_progress)) {
|
|
/* we already were inside restart this means it's not safe to touch shm */
|
|
accel_deactivate_now(); /* drop usage lock */
|
|
return FAILURE;
|
|
}
|
|
ZCG(counted) = 1;
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* must be called ONLY after SUCCESSFUL accelerator_shm_read_lock */
|
|
void accelerator_shm_read_unlock(void)
|
|
{
|
|
if (!ZCG(counted)) {
|
|
/* counted is 0 - meaning we had to readlock manually, release readlock now */
|
|
accel_deactivate_now();
|
|
}
|
|
}
|
|
|
|
/* Preloading */
|
|
static HashTable *preload_scripts = NULL;
|
|
static zend_op_array *(*preload_orig_compile_file)(zend_file_handle *file_handle, int type);
|
|
|
|
static void preload_shutdown(void)
|
|
{
|
|
zval *zv;
|
|
|
|
#if 0
|
|
if (EG(zend_constants)) {
|
|
ZEND_HASH_REVERSE_FOREACH_VAL(EG(zend_constants), zv) {
|
|
zend_constant *c = Z_PTR_P(zv);
|
|
if (ZEND_CONSTANT_FLAGS(c) & CONST_PERSISTENT) {
|
|
break;
|
|
}
|
|
} ZEND_HASH_FOREACH_END_DEL();
|
|
}
|
|
#endif
|
|
|
|
if (EG(function_table)) {
|
|
ZEND_HASH_REVERSE_FOREACH_VAL(EG(function_table), zv) {
|
|
zend_function *func = Z_PTR_P(zv);
|
|
if (func->type == ZEND_INTERNAL_FUNCTION) {
|
|
break;
|
|
}
|
|
} ZEND_HASH_FOREACH_END_DEL();
|
|
}
|
|
|
|
if (EG(class_table)) {
|
|
ZEND_HASH_REVERSE_FOREACH_VAL(EG(class_table), zv) {
|
|
zend_class_entry *ce = Z_PTR_P(zv);
|
|
if (ce->type == ZEND_INTERNAL_CLASS) {
|
|
break;
|
|
}
|
|
} ZEND_HASH_FOREACH_END_DEL();
|
|
}
|
|
}
|
|
|
|
static void preload_activate(void)
|
|
{
|
|
if (ZCSG(preload_script)->ping_auto_globals_mask) {
|
|
zend_accel_set_auto_globals(ZCSG(preload_script)->ping_auto_globals_mask);
|
|
}
|
|
}
|
|
|
|
static void preload_restart(void)
|
|
{
|
|
zend_accel_hash_update(&ZCSG(hash), ZSTR_VAL(ZCSG(preload_script)->script.filename), ZSTR_LEN(ZCSG(preload_script)->script.filename), 0, ZCSG(preload_script));
|
|
if (ZCSG(saved_scripts)) {
|
|
zend_persistent_script **p = ZCSG(saved_scripts);
|
|
while (*p) {
|
|
zend_accel_hash_update(&ZCSG(hash), ZSTR_VAL((*p)->script.filename), ZSTR_LEN((*p)->script.filename), 0, *p);
|
|
p++;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void preload_move_user_functions(HashTable *src, HashTable *dst)
|
|
{
|
|
Bucket *p;
|
|
dtor_func_t orig_dtor = src->pDestructor;
|
|
|
|
src->pDestructor = NULL;
|
|
zend_hash_extend(dst, dst->nNumUsed + src->nNumUsed, 0);
|
|
ZEND_HASH_REVERSE_FOREACH_BUCKET(src, p) {
|
|
zend_function *function = Z_PTR(p->val);
|
|
|
|
if (EXPECTED(function->type == ZEND_USER_FUNCTION)) {
|
|
_zend_hash_append_ptr(dst, p->key, function);
|
|
zend_hash_del_bucket(src, p);
|
|
} else {
|
|
break;
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
src->pDestructor = orig_dtor;
|
|
}
|
|
|
|
static void preload_move_user_classes(HashTable *src, HashTable *dst)
|
|
{
|
|
Bucket *p;
|
|
dtor_func_t orig_dtor = src->pDestructor;
|
|
|
|
src->pDestructor = NULL;
|
|
zend_hash_extend(dst, dst->nNumUsed + src->nNumUsed, 0);
|
|
ZEND_HASH_REVERSE_FOREACH_BUCKET(src, p) {
|
|
zend_class_entry *ce = Z_PTR(p->val);
|
|
|
|
if (EXPECTED(ce->type == ZEND_USER_CLASS)) {
|
|
_zend_hash_append_ptr(dst, p->key, ce);
|
|
zend_hash_del_bucket(src, p);
|
|
} else {
|
|
break;
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
src->pDestructor = orig_dtor;
|
|
}
|
|
|
|
static zend_op_array *preload_compile_file(zend_file_handle *file_handle, int type)
|
|
{
|
|
zend_op_array *op_array = preload_orig_compile_file(file_handle, type);
|
|
|
|
if (op_array && op_array->refcount) {
|
|
zend_persistent_script *script;
|
|
|
|
script = create_persistent_script();
|
|
script->script.first_early_binding_opline = (uint32_t)-1;
|
|
script->script.filename = zend_string_copy(op_array->filename);
|
|
zend_string_hash_val(script->script.filename);
|
|
script->script.main_op_array = *op_array;
|
|
|
|
//??? efree(op_array->refcount);
|
|
op_array->refcount = NULL;
|
|
|
|
if (op_array->static_variables &&
|
|
!(GC_FLAGS(op_array->static_variables) & IS_ARRAY_IMMUTABLE)) {
|
|
GC_ADDREF(op_array->static_variables);
|
|
}
|
|
|
|
zend_hash_add_ptr(preload_scripts, script->script.filename, script);
|
|
}
|
|
|
|
return op_array;
|
|
}
|
|
|
|
static void preload_sort_classes(void *base, size_t count, size_t siz, compare_func_t compare, swap_func_t swp) /* {{{ */
|
|
{
|
|
Bucket *b1 = base;
|
|
Bucket *b2;
|
|
Bucket *end = b1 + count;
|
|
Bucket tmp;
|
|
zend_class_entry *ce, *p;
|
|
|
|
while (b1 < end) {
|
|
try_again:
|
|
ce = (zend_class_entry*)Z_PTR(b1->val);
|
|
if (ce->parent && (ce->ce_flags & ZEND_ACC_LINKED)) {
|
|
p = ce->parent;
|
|
if (p->type == ZEND_USER_CLASS) {
|
|
b2 = b1 + 1;
|
|
while (b2 < end) {
|
|
if (p == Z_PTR(b2->val)) {
|
|
tmp = *b1;
|
|
*b1 = *b2;
|
|
*b2 = tmp;
|
|
goto try_again;
|
|
}
|
|
b2++;
|
|
}
|
|
}
|
|
}
|
|
if (ce->num_interfaces && (ce->ce_flags & ZEND_ACC_LINKED)) {
|
|
uint32_t i = 0;
|
|
for (i = 0; i < ce->num_interfaces; i++) {
|
|
p = ce->interfaces[i];
|
|
if (p->type == ZEND_USER_CLASS) {
|
|
b2 = b1 + 1;
|
|
while (b2 < end) {
|
|
if (p == Z_PTR(b2->val)) {
|
|
tmp = *b1;
|
|
*b1 = *b2;
|
|
*b2 = tmp;
|
|
goto try_again;
|
|
}
|
|
b2++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
b1++;
|
|
}
|
|
}
|
|
|
|
static void preload_link(void)
|
|
{
|
|
zval *zv;
|
|
zend_persistent_script *script;
|
|
zend_class_entry *ce, *parent, *p;
|
|
zend_string *key;
|
|
zend_bool found, changed;
|
|
uint32_t i;
|
|
zend_op_array *op_array;
|
|
dtor_func_t orig_dtor;
|
|
|
|
/* Resolve class dependencies */
|
|
do {
|
|
changed = 0;
|
|
|
|
ZEND_HASH_REVERSE_FOREACH_VAL(EG(class_table), zv) {
|
|
ce = Z_PTR_P(zv);
|
|
if (ce->type == ZEND_INTERNAL_CLASS) {
|
|
break;
|
|
}
|
|
if ((ce->ce_flags & ZEND_ACC_TOP_LEVEL)
|
|
&& !(ce->ce_flags & ZEND_ACC_LINKED)) {
|
|
|
|
parent = NULL;
|
|
|
|
if (ce->parent_name) {
|
|
key = zend_string_tolower(ce->parent_name);
|
|
parent = zend_hash_find_ptr(EG(class_table), key);
|
|
zend_string_release(key);
|
|
if (!parent) continue;
|
|
#ifdef ZEND_WIN32
|
|
/* On Windows we can't link with internal class, because of ASLR */
|
|
if (parent->type == ZEND_INTERNAL_CLASS) continue;
|
|
#endif
|
|
}
|
|
|
|
if (ce->num_interfaces) {
|
|
found = 1;
|
|
for (i = 0; i < ce->num_interfaces; i++) {
|
|
p = zend_hash_find_ptr(EG(class_table), ce->interface_names[i].lc_name);
|
|
if (!p) {
|
|
found = 0;
|
|
break;
|
|
}
|
|
#ifdef ZEND_WIN32
|
|
/* On Windows we can't link with internal class, because of ASLR */
|
|
if (p->type == ZEND_INTERNAL_CLASS) {
|
|
found = 0;
|
|
break;
|
|
}
|
|
#endif
|
|
}
|
|
if (!found) continue;
|
|
}
|
|
|
|
if (ce->num_traits) {
|
|
found = 1;
|
|
for (i = 0; i < ce->num_traits; i++) {
|
|
p = zend_hash_find_ptr(EG(class_table), ce->trait_names[i].lc_name);
|
|
if (!p) {
|
|
found = 0;
|
|
break;
|
|
}
|
|
#ifdef ZEND_WIN32
|
|
/* On Windows we can't link with internal class, because of ASLR */
|
|
if (p->type == ZEND_INTERNAL_CLASS) {
|
|
found = 0;
|
|
break;
|
|
}
|
|
#endif
|
|
}
|
|
if (!found) continue;
|
|
}
|
|
|
|
key = zend_string_tolower(ce->name);
|
|
zv = zend_hash_set_bucket_key(EG(class_table), (Bucket*)zv, key);
|
|
zend_string_release(key);
|
|
if (EXPECTED(zv)) {
|
|
zend_do_link_class(ce, parent);
|
|
changed = 1;
|
|
}
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
} while (changed);
|
|
|
|
/* Resolve class constants */
|
|
EG(exception) = (void*)(uintptr_t)-1; /* prevent error reporting */
|
|
do {
|
|
changed = 0;
|
|
ZEND_HASH_REVERSE_FOREACH_VAL(EG(class_table), zv) {
|
|
ce = Z_PTR_P(zv);
|
|
if (ce->type == ZEND_INTERNAL_CLASS) {
|
|
break;
|
|
}
|
|
if ((ce->ce_flags & ZEND_ACC_LINKED)
|
|
&& !(ce->ce_flags & ZEND_ACC_CONSTANTS_UPDATED)) {
|
|
zend_bool ok = 1;
|
|
zend_class_constant *c;
|
|
zval *val;
|
|
|
|
ZEND_HASH_FOREACH_PTR(&ce->constants_table, c) {
|
|
val = &c->value;
|
|
if (Z_TYPE_P(val) == IS_CONSTANT_AST) {
|
|
if (EXPECTED(zval_update_constant_ex(val, c->ce) == SUCCESS)) {
|
|
changed = 1;
|
|
} else {
|
|
ok = 0;
|
|
}
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
if (ce->default_properties_count) {
|
|
zend_class_entry *pce = ce;
|
|
|
|
val = ce->default_properties_table + ce->default_properties_count - 1;
|
|
do {
|
|
uint32_t count = pce->parent ? pce->default_properties_count - pce->parent->default_properties_count : pce->default_properties_count;
|
|
|
|
while (count) {
|
|
if (Z_TYPE_P(val) == IS_CONSTANT_AST) {
|
|
if (UNEXPECTED(zval_update_constant_ex(val, pce) != SUCCESS)) {
|
|
ok = 0;
|
|
}
|
|
}
|
|
val--;
|
|
count--;
|
|
}
|
|
pce = pce->parent;
|
|
} while (pce && pce-> default_properties_count);
|
|
}
|
|
if (ce->default_static_members_count) {
|
|
uint32_t count = ce->parent ? ce->default_static_members_count - ce->parent->default_static_members_count : ce->default_static_members_count;
|
|
|
|
val = ce->default_static_members_table + ce->default_static_members_count - 1;
|
|
while (count) {
|
|
if (Z_TYPE_P(val) == IS_CONSTANT_AST) {
|
|
if (UNEXPECTED(zval_update_constant_ex(val, ce) != SUCCESS)) {
|
|
ok = 0;
|
|
}
|
|
}
|
|
val--;
|
|
count--;
|
|
}
|
|
}
|
|
if (ok) {
|
|
ce->ce_flags |= ZEND_ACC_CONSTANTS_UPDATED;
|
|
}
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
} while (changed);
|
|
EG(exception) = NULL;
|
|
|
|
/* Move unlinked clases (and with unresilved constants) back to scripts */
|
|
orig_dtor = EG(class_table)->pDestructor;
|
|
EG(class_table)->pDestructor = NULL;
|
|
ZEND_HASH_REVERSE_FOREACH_STR_KEY_VAL(EG(class_table), key, zv) {
|
|
ce = Z_PTR_P(zv);
|
|
if (ce->type == ZEND_INTERNAL_CLASS) {
|
|
break;
|
|
}
|
|
if (!(ce->ce_flags & ZEND_ACC_LINKED)) {
|
|
zend_error(E_WARNING, "Can't preload unlinked class %s at %s:%d\n", ZSTR_VAL(ce->name), ZSTR_VAL(ce->info.user.filename), ce->info.user.line_start);
|
|
} else if (!(ce->ce_flags & ZEND_ACC_CONSTANTS_UPDATED)) {
|
|
zend_error(E_WARNING, "Can't preload class %s with unresolved constants at %s:%d\n", ZSTR_VAL(ce->name), ZSTR_VAL(ce->info.user.filename), ce->info.user.line_start);
|
|
} else {
|
|
continue;
|
|
}
|
|
script = zend_hash_find_ptr(preload_scripts, ce->info.user.filename);
|
|
ZEND_ASSERT(script);
|
|
zend_hash_add(&script->script.class_table, key, zv);
|
|
ZVAL_UNDEF(zv);
|
|
zend_string_release(key);
|
|
EG(class_table)->nNumOfElements--;
|
|
} ZEND_HASH_FOREACH_END();
|
|
EG(class_table)->pDestructor = orig_dtor;
|
|
zend_hash_rehash(EG(class_table));
|
|
|
|
/* Move run-time declared functions back to scripts */
|
|
orig_dtor = EG(function_table)->pDestructor;
|
|
EG(function_table)->pDestructor = NULL;
|
|
ZEND_HASH_REVERSE_FOREACH_STR_KEY_VAL(EG(function_table), key, zv) {
|
|
op_array = Z_PTR_P(zv);
|
|
if (op_array->type == ZEND_INTERNAL_FUNCTION) {
|
|
break;
|
|
}
|
|
if (op_array->fn_flags & (ZEND_ACC_TOP_LEVEL|ZEND_ACC_CLOSURE)) {
|
|
continue;
|
|
}
|
|
script = zend_hash_find_ptr(preload_scripts, op_array->filename);
|
|
ZEND_ASSERT(script);
|
|
zend_hash_add(&script->script.function_table, key, zv);
|
|
ZVAL_UNDEF(zv);
|
|
zend_string_release(key);
|
|
EG(function_table)->nNumOfElements--;
|
|
} ZEND_HASH_FOREACH_END();
|
|
EG(function_table)->pDestructor = orig_dtor;
|
|
zend_hash_rehash(EG(function_table));
|
|
|
|
|
|
/* Remove DECLARE opcodes */
|
|
ZEND_HASH_FOREACH_PTR(preload_scripts, script) {
|
|
zend_op_array *op_array = &script->script.main_op_array;
|
|
zend_op *opline = op_array->opcodes;
|
|
zend_op *end = opline + op_array->last;
|
|
|
|
while (opline != end) {
|
|
switch (opline->opcode) {
|
|
case ZEND_DECLARE_CLASS:
|
|
case ZEND_DECLARE_INHERITED_CLASS:
|
|
case ZEND_DECLARE_INHERITED_CLASS_DELAYED:
|
|
key = Z_STR_P(RT_CONSTANT(opline, opline->op1) + 1);
|
|
zv = zend_hash_find_ex(EG(class_table), key, 1);
|
|
if (EXPECTED(!zv)) {
|
|
MAKE_NOP(opline);
|
|
}
|
|
break;
|
|
}
|
|
opline++;
|
|
}
|
|
|
|
if (op_array->fn_flags & ZEND_ACC_EARLY_BINDING) {
|
|
script->script.first_early_binding_opline = zend_build_delayed_early_binding_list(op_array);
|
|
if (script->script.first_early_binding_opline == (uint32_t)-1) {
|
|
op_array->fn_flags &= ~ZEND_ACC_EARLY_BINDING;
|
|
}
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
}
|
|
|
|
static zend_string *preload_resolve_path(zend_string *filename)
|
|
{
|
|
return zend_resolve_path(ZSTR_VAL(filename), ZSTR_LEN(filename));
|
|
}
|
|
|
|
static void preload_remove_empty_includes(void)
|
|
{
|
|
zend_persistent_script *script;
|
|
zend_bool changed;
|
|
|
|
/* mark all as empty */
|
|
ZEND_HASH_FOREACH_PTR(preload_scripts, script) {
|
|
script->empty = 1;
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
/* find non empty scripts */
|
|
do {
|
|
changed = 0;
|
|
ZEND_HASH_FOREACH_PTR(preload_scripts, script) {
|
|
if (script->empty) {
|
|
int empty = 1;
|
|
zend_op *opline = script->script.main_op_array.opcodes;
|
|
zend_op *end = opline + script->script.main_op_array.last;
|
|
|
|
while (opline < end) {
|
|
if (opline->opcode == ZEND_INCLUDE_OR_EVAL &&
|
|
opline->extended_value != ZEND_EVAL &&
|
|
opline->op1_type == IS_CONST &&
|
|
Z_TYPE_P(RT_CONSTANT(opline, opline->op1)) == IS_STRING) {
|
|
|
|
zend_string *resolved_path = preload_resolve_path(Z_STR_P(RT_CONSTANT(opline, opline->op1)));
|
|
|
|
if (resolved_path) {
|
|
zend_persistent_script *incl = zend_hash_find_ptr(preload_scripts, resolved_path);
|
|
zend_string_release(resolved_path);
|
|
if (!incl->empty) {
|
|
empty = 0;
|
|
break;
|
|
}
|
|
} else {
|
|
empty = 0;
|
|
break;
|
|
}
|
|
} else if (opline->opcode != ZEND_NOP &&
|
|
opline->opcode != ZEND_RETURN &&
|
|
opline->opcode != ZEND_HANDLE_EXCEPTION) {
|
|
empty = 0;
|
|
break;
|
|
}
|
|
opline++;
|
|
}
|
|
if (!empty) {
|
|
script->empty = 0;
|
|
changed = 1;
|
|
}
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
} while (changed);
|
|
|
|
/* remove empty includes */
|
|
ZEND_HASH_FOREACH_PTR(preload_scripts, script) {
|
|
zend_op *opline = script->script.main_op_array.opcodes;
|
|
zend_op *end = opline + script->script.main_op_array.last;
|
|
|
|
while (opline < end) {
|
|
if (opline->opcode == ZEND_INCLUDE_OR_EVAL &&
|
|
opline->extended_value != ZEND_EVAL &&
|
|
opline->op1_type == IS_CONST &&
|
|
Z_TYPE_P(RT_CONSTANT(opline, opline->op1)) == IS_STRING) {
|
|
|
|
zend_string *resolved_path = preload_resolve_path(Z_STR_P(RT_CONSTANT(opline, opline->op1)));
|
|
|
|
if (resolved_path) {
|
|
zend_persistent_script *incl = zend_hash_find_ptr(preload_scripts, resolved_path);
|
|
if (incl->empty) {
|
|
MAKE_NOP(opline);
|
|
} else {
|
|
if (!IS_ABSOLUTE_PATH(Z_STRVAL_P(RT_CONSTANT(opline, opline->op1)), Z_STRLEN_P(RT_CONSTANT(opline, opline->op1)))) {
|
|
/* replace relative patch with absolute one */
|
|
zend_string_release(Z_STR_P(RT_CONSTANT(opline, opline->op1)));
|
|
ZVAL_STR_COPY(RT_CONSTANT(opline, opline->op1), resolved_path);
|
|
}
|
|
}
|
|
zend_string_release(resolved_path);
|
|
}
|
|
}
|
|
opline++;
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
}
|
|
|
|
static int preload_optimize(zend_persistent_script *script)
|
|
{
|
|
zend_class_entry *ce;
|
|
zend_op_array *op_array;
|
|
|
|
zend_shared_alloc_init_xlat_table();
|
|
|
|
ZEND_HASH_FOREACH_PTR(&script->script.class_table, ce) {
|
|
if (ce->ce_flags & ZEND_ACC_TRAIT) {
|
|
ZEND_HASH_FOREACH_PTR(&ce->function_table, op_array) {
|
|
if (!(op_array->fn_flags & ZEND_ACC_TRAIT_CLONE)) {
|
|
zend_shared_alloc_register_xlat_entry(op_array->opcodes, op_array);
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
if (!zend_optimize_script(&script->script, ZCG(accel_directives).optimization_level, ZCG(accel_directives).opt_debug_level)) {
|
|
return FAILURE;
|
|
}
|
|
|
|
ZEND_HASH_FOREACH_PTR(&script->script.class_table, ce) {
|
|
if (ce->ce_flags & ZEND_ACC_IMPLEMENT_TRAITS) {
|
|
ZEND_HASH_FOREACH_PTR(&ce->function_table, op_array) {
|
|
if (op_array->fn_flags & ZEND_ACC_TRAIT_CLONE) {
|
|
zend_op_array *orig_op_array = zend_shared_alloc_get_xlat_entry(op_array->opcodes);
|
|
if (orig_op_array) {
|
|
zend_class_entry *scope = op_array->scope;
|
|
uint32_t fn_flags = op_array->fn_flags;
|
|
zend_function *prototype = op_array->prototype;
|
|
HashTable *ht = op_array->static_variables;
|
|
*op_array = *orig_op_array;
|
|
op_array->scope = scope;
|
|
op_array->fn_flags = fn_flags;
|
|
op_array->prototype = prototype;
|
|
op_array->static_variables = ht;
|
|
}
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
zend_shared_alloc_destroy_xlat_table();
|
|
|
|
ZEND_HASH_FOREACH_PTR(preload_scripts, script) {
|
|
if (!zend_optimize_script(&script->script, ZCG(accel_directives).optimization_level, ZCG(accel_directives).opt_debug_level)) {
|
|
return FAILURE;
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
return SUCCESS;
|
|
}
|
|
|
|
static zend_persistent_script* preload_script_in_shared_memory(zend_persistent_script *new_persistent_script)
|
|
{
|
|
zend_accel_hash_entry *bucket;
|
|
uint32_t memory_used;
|
|
uint32_t checkpoint;
|
|
|
|
if (zend_accel_hash_is_full(&ZCSG(hash))) {
|
|
zend_accel_error(ACCEL_LOG_FATAL, "Not enough entries in hash table for preloading!");
|
|
return NULL;
|
|
}
|
|
|
|
checkpoint = zend_shared_alloc_checkpoint_xlat_table();
|
|
|
|
/* Calculate the required memory size */
|
|
memory_used = zend_accel_script_persist_calc(new_persistent_script, NULL, 0, 1);
|
|
|
|
/* Allocate shared memory */
|
|
#if defined(__AVX__) || defined(__SSE2__)
|
|
/* Align to 64-byte boundary */
|
|
ZCG(mem) = zend_shared_alloc(memory_used + 64);
|
|
if (ZCG(mem)) {
|
|
ZCG(mem) = (void*)(((zend_uintptr_t)ZCG(mem) + 63L) & ~63L);
|
|
#if defined(__x86_64__)
|
|
memset(ZCG(mem), 0, memory_used);
|
|
#elif defined(__AVX__)
|
|
{
|
|
char *p = (char*)ZCG(mem);
|
|
char *end = p + memory_used;
|
|
__m256i ymm0 = _mm256_setzero_si256();
|
|
|
|
while (p < end) {
|
|
_mm256_store_si256((__m256i*)p, ymm0);
|
|
_mm256_store_si256((__m256i*)(p+32), ymm0);
|
|
p += 64;
|
|
}
|
|
}
|
|
#else
|
|
{
|
|
char *p = (char*)ZCG(mem);
|
|
char *end = p + memory_used;
|
|
__m128i xmm0 = _mm_setzero_si128();
|
|
|
|
while (p < end) {
|
|
_mm_store_si128((__m128i*)p, xmm0);
|
|
_mm_store_si128((__m128i*)(p+16), xmm0);
|
|
_mm_store_si128((__m128i*)(p+32), xmm0);
|
|
_mm_store_si128((__m128i*)(p+48), xmm0);
|
|
p += 64;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
#else
|
|
ZCG(mem) = zend_shared_alloc(memory_used);
|
|
if (ZCG(mem)) {
|
|
memset(ZCG(mem), 0, memory_used);
|
|
}
|
|
#endif
|
|
if (!ZCG(mem)) {
|
|
zend_accel_error(ACCEL_LOG_FATAL, "Not enough shared memory for preloading!");
|
|
return NULL;
|
|
}
|
|
|
|
zend_shared_alloc_restore_xlat_table(checkpoint);
|
|
|
|
/* Copy into shared memory */
|
|
new_persistent_script = zend_accel_script_persist(new_persistent_script, NULL, 0, 1);
|
|
|
|
new_persistent_script->is_phar = is_phar_file(new_persistent_script->script.filename);
|
|
|
|
/* Consistency check */
|
|
if ((char*)new_persistent_script->mem + new_persistent_script->size != (char*)ZCG(mem)) {
|
|
zend_accel_error(
|
|
((char*)new_persistent_script->mem + new_persistent_script->size < (char*)ZCG(mem)) ? ACCEL_LOG_ERROR : ACCEL_LOG_WARNING,
|
|
"Internal error: wrong size calculation: %s start=" ZEND_ADDR_FMT ", end=" ZEND_ADDR_FMT ", real=" ZEND_ADDR_FMT "\n",
|
|
ZSTR_VAL(new_persistent_script->script.filename),
|
|
(size_t)new_persistent_script->mem,
|
|
(size_t)((char *)new_persistent_script->mem + new_persistent_script->size),
|
|
(size_t)ZCG(mem));
|
|
}
|
|
|
|
new_persistent_script->dynamic_members.checksum = zend_accel_script_checksum(new_persistent_script);
|
|
|
|
/* store script structure in the hash table */
|
|
bucket = zend_accel_hash_update(&ZCSG(hash), ZSTR_VAL(new_persistent_script->script.filename), ZSTR_LEN(new_persistent_script->script.filename), 0, new_persistent_script);
|
|
if (bucket) {
|
|
zend_accel_error(ACCEL_LOG_INFO, "Cached script '%s'", ZSTR_VAL(new_persistent_script->script.filename));
|
|
}
|
|
|
|
new_persistent_script->dynamic_members.memory_consumption = ZEND_ALIGNED_SIZE(new_persistent_script->size);
|
|
|
|
return new_persistent_script;
|
|
}
|
|
|
|
static void preload_load(void)
|
|
{
|
|
/* Load into process tables */
|
|
if (zend_hash_num_elements(&ZCSG(preload_script)->script.function_table)) {
|
|
Bucket *p = ZCSG(preload_script)->script.function_table.arData;
|
|
Bucket *end = p + ZCSG(preload_script)->script.function_table.nNumUsed;
|
|
|
|
for (; p != end; p++) {
|
|
_zend_hash_append_ptr_ex(CG(function_table), p->key, Z_PTR(p->val), 1);
|
|
}
|
|
}
|
|
|
|
if (zend_hash_num_elements(&ZCSG(preload_script)->script.class_table)) {
|
|
Bucket *p = ZCSG(preload_script)->script.class_table.arData;
|
|
Bucket *end = p + ZCSG(preload_script)->script.class_table.nNumUsed;
|
|
|
|
for (; p != end; p++) {
|
|
_zend_hash_append_ptr_ex(CG(class_table), p->key, Z_PTR(p->val), 1);
|
|
}
|
|
}
|
|
|
|
if (EG(zend_constants)) {
|
|
EG(persistent_constants_count) = EG(zend_constants)->nNumUsed;
|
|
}
|
|
if (EG(function_table)) {
|
|
EG(persistent_functions_count) = EG(function_table)->nNumUsed;
|
|
}
|
|
if (EG(class_table)) {
|
|
EG(persistent_classes_count) = EG(class_table)->nNumUsed;
|
|
}
|
|
CG(map_ptr_last) = ZCSG(map_ptr_last);
|
|
}
|
|
|
|
static int accel_preload(const char *config)
|
|
{
|
|
zend_file_handle file_handle;
|
|
int ret;
|
|
uint32_t orig_compiler_options;
|
|
|
|
ZCG(enabled) = 0;
|
|
preload_orig_compile_file = accelerator_orig_compile_file;
|
|
accelerator_orig_compile_file = preload_compile_file;
|
|
|
|
orig_compiler_options = CG(compiler_options);
|
|
CG(compiler_options) |= ZEND_COMPILE_HANDLE_OP_ARRAY;
|
|
// CG(compiler_options) |= ZEND_COMPILE_IGNORE_INTERNAL_CLASSES;
|
|
CG(compiler_options) |= ZEND_COMPILE_DELAYED_BINDING;
|
|
CG(compiler_options) |= ZEND_COMPILE_NO_CONSTANT_SUBSTITUTION;
|
|
// CG(compiler_options) |= ZEND_COMPILE_IGNORE_OTHER_FILES;
|
|
|
|
/* Compile and execute proloading script */
|
|
memset(&file_handle, 0, sizeof(file_handle));
|
|
file_handle.filename = (char*)config;
|
|
file_handle.free_filename = 0;
|
|
file_handle.type = ZEND_HANDLE_FILENAME;
|
|
file_handle.opened_path = NULL;
|
|
file_handle.handle.fp = NULL;
|
|
|
|
preload_scripts = emalloc(sizeof(HashTable));
|
|
zend_hash_init(preload_scripts, 0, NULL, NULL, 0);
|
|
|
|
zend_try {
|
|
ret = zend_execute_scripts(ZEND_REQUIRE, NULL, 1, &file_handle);
|
|
} zend_catch {
|
|
ret = FAILURE;
|
|
} zend_end_try();
|
|
|
|
CG(compiler_options) = orig_compiler_options;
|
|
accelerator_orig_compile_file = preload_orig_compile_file;
|
|
ZCG(enabled) = 1;
|
|
|
|
zend_destroy_file_handle(&file_handle);
|
|
|
|
if (ret == SUCCESS) {
|
|
zend_persistent_script *script;
|
|
zend_string *filename;
|
|
int i;
|
|
|
|
/* Release stored values to avoid dangling pointers */
|
|
zend_hash_graceful_reverse_destroy(&EG(symbol_table));
|
|
zend_hash_init(&EG(symbol_table), 0, NULL, ZVAL_PTR_DTOR, 0);
|
|
|
|
preload_link();
|
|
preload_remove_empty_includes();
|
|
|
|
/* Don't preload constants */
|
|
if (EG(zend_constants)) {
|
|
zval *zv;
|
|
ZEND_HASH_REVERSE_FOREACH_VAL(EG(zend_constants), zv) {
|
|
zend_constant *c = Z_PTR_P(zv);
|
|
if (ZEND_CONSTANT_FLAGS(c) & CONST_PERSISTENT) {
|
|
break;
|
|
}
|
|
EG(zend_constants)->pDestructor(zv);
|
|
} ZEND_HASH_FOREACH_END_DEL();
|
|
}
|
|
|
|
script = create_persistent_script();
|
|
|
|
/* Fill in the ping_auto_globals_mask for the new script. If jit for auto globals is enabled we
|
|
will have to ping the used auto global variables before execution */
|
|
if (PG(auto_globals_jit)) {
|
|
script->ping_auto_globals_mask = zend_accel_get_auto_globals();
|
|
} else {
|
|
script->ping_auto_globals_mask = zend_accel_get_auto_globals_no_jit();
|
|
}
|
|
|
|
/* Store all functions and classes in a single pseudo-file */
|
|
filename = zend_string_init("$PRELOAD$", strlen("$PRELOAD$"), 0);
|
|
#if ZEND_USE_ABS_CONST_ADDR
|
|
init_op_array(&script->script.main_op_array, ZEND_USER_FUNCTION, 1);
|
|
#else
|
|
init_op_array(&script->script.main_op_array, ZEND_USER_FUNCTION, 2);
|
|
#endif
|
|
script->script.main_op_array.last = 1;
|
|
script->script.main_op_array.last_literal = 1;
|
|
#if ZEND_USE_ABS_CONST_ADDR
|
|
script->script.main_op_array.literals = (zval*)emalloc(sizeof(zval));
|
|
#else
|
|
script->script.main_op_array.literals = (zval*)(script->script.main_op_array.opcodes + 1);
|
|
#endif
|
|
ZVAL_NULL(script->script.main_op_array.literals);
|
|
memset(script->script.main_op_array.opcodes, 0, sizeof(zend_op));
|
|
script->script.main_op_array.opcodes[0].opcode = ZEND_RETURN;
|
|
script->script.main_op_array.opcodes[0].op1_type = IS_CONST;
|
|
script->script.main_op_array.opcodes[0].op1.constant = 0;
|
|
ZEND_PASS_TWO_UPDATE_CONSTANT(&script->script.main_op_array, script->script.main_op_array.opcodes, script->script.main_op_array.opcodes[0].op1);
|
|
|
|
script->script.main_op_array.filename = filename;
|
|
script->script.filename = zend_string_copy(filename);
|
|
|
|
script->script.first_early_binding_opline = (uint32_t)-1;
|
|
|
|
preload_move_user_functions(CG(function_table), &script->script.function_table);
|
|
preload_move_user_classes(CG(class_table), &script->script.class_table);
|
|
|
|
zend_hash_sort_ex(&script->script.class_table, preload_sort_classes, NULL, 0);
|
|
|
|
if (preload_optimize(script) != SUCCESS) {
|
|
zend_accel_error(ACCEL_LOG_FATAL, "Optimization error during preloading!");
|
|
return FAILURE;
|
|
}
|
|
|
|
zend_shared_alloc_init_xlat_table();
|
|
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
|
SHM_UNPROTECT();
|
|
|
|
ZCSG(preload_script) = preload_script_in_shared_memory(script);
|
|
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
|
|
zend_string_release(filename);
|
|
|
|
ZEND_ASSERT(ZCSG(preload_script)->arena_size == 0);
|
|
|
|
preload_load();
|
|
|
|
/* Store individual scripts with unlinked classes */
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
|
SHM_UNPROTECT();
|
|
|
|
i = 0;
|
|
ZCSG(saved_scripts) = zend_shared_alloc((zend_hash_num_elements(preload_scripts) + 1) * sizeof(void*));
|
|
ZEND_HASH_FOREACH_PTR(preload_scripts, script) {
|
|
ZCSG(saved_scripts)[i++] = preload_script_in_shared_memory(script);
|
|
} ZEND_HASH_FOREACH_END();
|
|
ZCSG(saved_scripts)[i] = NULL;
|
|
|
|
zend_shared_alloc_save_state();
|
|
accel_interned_strings_save_state();
|
|
|
|
SHM_PROTECT();
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
|
|
zend_shared_alloc_destroy_xlat_table();
|
|
}
|
|
|
|
zend_hash_destroy(preload_scripts);
|
|
efree(preload_scripts);
|
|
preload_scripts = NULL;
|
|
|
|
return ret;
|
|
}
|
|
|
|
size_t preload_ub_write(const char *str, size_t str_length)
|
|
{
|
|
return fwrite(str, 1, str_length, stdout);
|
|
}
|
|
|
|
void preload_flush(void *server_context)
|
|
{
|
|
fflush(stdout);
|
|
}
|
|
|
|
static int accel_finish_startup(void)
|
|
{
|
|
if (!ZCG(enabled) || !accel_startup_ok) {
|
|
return SUCCESS;
|
|
}
|
|
|
|
if (ZCG(accel_directives).preload && *ZCG(accel_directives).preload) {
|
|
int ret = SUCCESS;
|
|
|
|
int (*orig_activate)(TSRMLS_D) = sapi_module.activate;
|
|
int (*orig_deactivate)(TSRMLS_D) = sapi_module.deactivate;
|
|
void (*orig_register_server_variables)(zval *track_vars_array TSRMLS_DC) = sapi_module.register_server_variables;
|
|
int (*orig_header_handler)(sapi_header_struct *sapi_header, sapi_header_op_enum op, sapi_headers_struct *sapi_headers TSRMLS_DC) = sapi_module.header_handler;
|
|
int (*orig_send_headers)(sapi_headers_struct *sapi_headers TSRMLS_DC) = sapi_module.send_headers;
|
|
void (*orig_send_header)(sapi_header_struct *sapi_header, void *server_context TSRMLS_DC)= sapi_module.send_header;
|
|
char *(*orig_getenv)(char *name, size_t name_len TSRMLS_DC) = sapi_module.getenv;
|
|
size_t (*orig_ub_write)(const char *str, size_t str_length) = sapi_module.ub_write;
|
|
void (*orig_flush)(void *server_context) = sapi_module.flush;
|
|
#ifdef ZEND_SIGNALS
|
|
zend_bool old_reset_signals = SIGG(reset);
|
|
#endif
|
|
|
|
if (UNEXPECTED(file_cache_only)) {
|
|
zend_accel_error(ACCEL_LOG_WARNING, "Preloading doesn't work in \"file_cache_only\" mode");
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* exclusive lock */
|
|
zend_shared_alloc_lock();
|
|
|
|
if (ZCSG(preload_script)) {
|
|
/* Preloading was done in another process */
|
|
preload_load();
|
|
zend_shared_alloc_unlock();
|
|
return SUCCESS;
|
|
}
|
|
|
|
sapi_module.activate = NULL;
|
|
sapi_module.deactivate = NULL;
|
|
sapi_module.register_server_variables = NULL;
|
|
sapi_module.header_handler = NULL;
|
|
sapi_module.send_headers = NULL;
|
|
sapi_module.send_header = NULL;
|
|
sapi_module.getenv = NULL;
|
|
sapi_module.ub_write = preload_ub_write;
|
|
sapi_module.flush = preload_flush;
|
|
|
|
zend_interned_strings_switch_storage(1);
|
|
|
|
#ifdef ZEND_SIGNALS
|
|
SIGG(reset) = 0;
|
|
#endif
|
|
if (php_request_startup() == SUCCESS) {
|
|
|
|
/* don't send headers */
|
|
SG(headers_sent) = 1;
|
|
SG(request_info).no_headers = 1;
|
|
php_output_set_status(0);
|
|
|
|
ZCG(auto_globals_mask) = 0;
|
|
ZCG(request_time) = (time_t)sapi_get_request_time();
|
|
ZCG(cache_opline) = NULL;
|
|
ZCG(cache_persistent_script) = NULL;
|
|
ZCG(include_path_key_len) = 0;
|
|
ZCG(include_path_check) = 1;
|
|
|
|
ZCG(cwd) = NULL;
|
|
ZCG(cwd_key_len) = 0;
|
|
ZCG(cwd_check) = 1;
|
|
|
|
if (accel_preload(ZCG(accel_directives).preload) != SUCCESS) {
|
|
ret = FAILURE;
|
|
}
|
|
|
|
php_request_shutdown(NULL);
|
|
} else {
|
|
ret = FAILURE;
|
|
}
|
|
#ifdef ZEND_SIGNALS
|
|
SIGG(reset) = old_reset_signals;
|
|
#endif
|
|
|
|
sapi_module.activate = orig_activate;
|
|
sapi_module.deactivate = orig_deactivate;
|
|
sapi_module.register_server_variables = orig_register_server_variables;
|
|
sapi_module.header_handler = orig_header_handler;
|
|
sapi_module.send_headers = orig_send_headers;
|
|
sapi_module.send_header = orig_send_header;
|
|
sapi_module.getenv = orig_getenv;
|
|
sapi_module.ub_write = orig_ub_write;
|
|
sapi_module.flush = orig_flush;
|
|
|
|
zend_shared_alloc_unlock();
|
|
|
|
sapi_activate();
|
|
|
|
return ret;
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
ZEND_EXT_API zend_extension zend_extension_entry = {
|
|
ACCELERATOR_PRODUCT_NAME, /* name */
|
|
PHP_VERSION, /* version */
|
|
"Zend Technologies", /* author */
|
|
"http://www.zend.com/", /* URL */
|
|
"Copyright (c) 1999-2018", /* copyright */
|
|
accel_startup, /* startup */
|
|
NULL, /* shutdown */
|
|
NULL, /* per-script activation */
|
|
NULL, /* per-script deactivation */
|
|
NULL, /* message handler */
|
|
NULL, /* op_array handler */
|
|
NULL, /* extended statement handler */
|
|
NULL, /* extended fcall begin handler */
|
|
NULL, /* extended fcall end handler */
|
|
NULL, /* op_array ctor */
|
|
NULL, /* op_array dtor */
|
|
STANDARD_ZEND_EXTENSION_PROPERTIES
|
|
};
|