zend_accelerator_module.c 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872
  1. /*
  2. +----------------------------------------------------------------------+
  3. | Zend OPcache |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1998-2018 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Authors: Andi Gutmans <andi@php.net> |
  16. | Zeev Suraski <zeev@php.net> |
  17. | Stanislav Malyshev <stas@zend.com> |
  18. | Dmitry Stogov <dmitry@php.net> |
  19. +----------------------------------------------------------------------+
  20. */
  21. #include <time.h>
  22. #include "php.h"
  23. #include "ZendAccelerator.h"
  24. #include "zend_API.h"
  25. #include "zend_shared_alloc.h"
  26. #include "zend_accelerator_blacklist.h"
  27. #include "php_ini.h"
  28. #include "SAPI.h"
  29. #include "zend_virtual_cwd.h"
  30. #include "ext/standard/info.h"
  31. #include "ext/standard/php_filestat.h"
  32. #define STRING_NOT_NULL(s) (NULL == (s)?"":s)
  33. #define MIN_ACCEL_FILES 200
  34. #define MAX_ACCEL_FILES 1000000
  35. #define TOKENTOSTR(X) #X
  36. static zif_handler orig_file_exists = NULL;
  37. static zif_handler orig_is_file = NULL;
  38. static zif_handler orig_is_readable = NULL;
  39. ZEND_BEGIN_ARG_INFO(arginfo_opcache_none, 0)
  40. ZEND_END_ARG_INFO()
  41. ZEND_BEGIN_ARG_INFO_EX(arginfo_opcache_get_status, 0, 0, 0)
  42. ZEND_ARG_INFO(0, fetch_scripts)
  43. ZEND_END_ARG_INFO()
  44. ZEND_BEGIN_ARG_INFO_EX(arginfo_opcache_compile_file, 0, 0, 1)
  45. ZEND_ARG_INFO(0, file)
  46. ZEND_END_ARG_INFO()
  47. ZEND_BEGIN_ARG_INFO_EX(arginfo_opcache_invalidate, 0, 0, 1)
  48. ZEND_ARG_INFO(0, script)
  49. ZEND_ARG_INFO(0, force)
  50. ZEND_END_ARG_INFO()
  51. ZEND_BEGIN_ARG_INFO_EX(arginfo_opcache_is_script_cached, 0, 0, 1)
  52. ZEND_ARG_INFO(0, script)
  53. ZEND_END_ARG_INFO()
  54. /* User functions */
  55. static ZEND_FUNCTION(opcache_reset);
  56. static ZEND_FUNCTION(opcache_invalidate);
  57. static ZEND_FUNCTION(opcache_is_script_cached);
  58. /* Private functions */
  59. static ZEND_FUNCTION(opcache_get_status);
  60. static ZEND_FUNCTION(opcache_compile_file);
  61. static ZEND_FUNCTION(opcache_get_configuration);
  62. static const zend_function_entry accel_functions[] = {
  63. /* User functions */
  64. ZEND_FE(opcache_reset, arginfo_opcache_none)
  65. ZEND_FE(opcache_invalidate, arginfo_opcache_invalidate)
  66. ZEND_FE(opcache_compile_file, arginfo_opcache_compile_file)
  67. ZEND_FE(opcache_is_script_cached, arginfo_opcache_is_script_cached)
  68. /* Private functions */
  69. ZEND_FE(opcache_get_configuration, arginfo_opcache_none)
  70. ZEND_FE(opcache_get_status, arginfo_opcache_get_status)
  71. ZEND_FE_END
  72. };
  73. static int validate_api_restriction(void)
  74. {
  75. if (ZCG(accel_directives).restrict_api && *ZCG(accel_directives).restrict_api) {
  76. size_t len = strlen(ZCG(accel_directives).restrict_api);
  77. if (!SG(request_info).path_translated ||
  78. strlen(SG(request_info).path_translated) < len ||
  79. memcmp(SG(request_info).path_translated, ZCG(accel_directives).restrict_api, len) != 0) {
  80. zend_error(E_WARNING, ACCELERATOR_PRODUCT_NAME " API is restricted by \"restrict_api\" configuration directive");
  81. return 0;
  82. }
  83. }
  84. return 1;
  85. }
  86. static ZEND_INI_MH(OnUpdateMemoryConsumption)
  87. {
  88. zend_long *p;
  89. zend_long memsize;
  90. #ifndef ZTS
  91. char *base = (char *) mh_arg2;
  92. #else
  93. char *base = (char *) ts_resource(*((int *) mh_arg2));
  94. #endif
  95. /* keep the compiler happy */
  96. (void)entry; (void)mh_arg2; (void)mh_arg3; (void)stage;
  97. p = (zend_long *) (base + (size_t)mh_arg1);
  98. memsize = atoi(ZSTR_VAL(new_value));
  99. /* sanity check we must use at least 8 MB */
  100. if (memsize < 8) {
  101. const char *new_new_value = "8";
  102. zend_ini_entry *ini_entry;
  103. memsize = 8;
  104. zend_accel_error(ACCEL_LOG_WARNING, "opcache.memory_consumption is set below the required 8MB.\n");
  105. zend_accel_error(ACCEL_LOG_WARNING, ACCELERATOR_PRODUCT_NAME " will use the minimal 8MB configuration.\n");
  106. if ((ini_entry = zend_hash_str_find_ptr(EG(ini_directives),
  107. "opcache.memory_consumption",
  108. sizeof("opcache.memory_consumption")-1)) == NULL) {
  109. return FAILURE;
  110. }
  111. ini_entry->value = zend_string_init_interned(new_new_value, 1, 1);
  112. }
  113. if (UNEXPECTED(memsize > ZEND_ULONG_MAX / (1024 * 1024))) {
  114. *p = ZEND_ULONG_MAX;
  115. } else {
  116. *p = memsize * (1024 * 1024);
  117. }
  118. return SUCCESS;
  119. }
  120. static ZEND_INI_MH(OnUpdateMaxAcceleratedFiles)
  121. {
  122. zend_long *p;
  123. zend_long size;
  124. #ifndef ZTS
  125. char *base = (char *) mh_arg2;
  126. #else
  127. char *base = (char *) ts_resource(*((int *) mh_arg2));
  128. #endif
  129. /* keep the compiler happy */
  130. (void)entry; (void)mh_arg2; (void)mh_arg3; (void)stage;
  131. p = (zend_long *) (base + (size_t)mh_arg1);
  132. size = atoi(ZSTR_VAL(new_value));
  133. /* sanity check we must use a value between MIN_ACCEL_FILES and MAX_ACCEL_FILES */
  134. if (size < MIN_ACCEL_FILES || size > MAX_ACCEL_FILES) {
  135. const char *new_new_value;
  136. zend_ini_entry *ini_entry;
  137. if (size < MIN_ACCEL_FILES) {
  138. size = MIN_ACCEL_FILES;
  139. new_new_value = TOKENTOSTR(MIN_ACCEL_FILES);
  140. zend_accel_error(ACCEL_LOG_WARNING, "opcache.max_accelerated_files is set below the required minimum (%d).\n", MIN_ACCEL_FILES);
  141. zend_accel_error(ACCEL_LOG_WARNING, ACCELERATOR_PRODUCT_NAME " will use the minimal configuration.\n");
  142. }
  143. if (size > MAX_ACCEL_FILES) {
  144. size = MAX_ACCEL_FILES;
  145. new_new_value = TOKENTOSTR(MAX_ACCEL_FILES);
  146. zend_accel_error(ACCEL_LOG_WARNING, "opcache.max_accelerated_files is set above the limit (%d).\n", MAX_ACCEL_FILES);
  147. zend_accel_error(ACCEL_LOG_WARNING, ACCELERATOR_PRODUCT_NAME " will use the maximal configuration.\n");
  148. }
  149. if ((ini_entry = zend_hash_str_find_ptr(EG(ini_directives),
  150. "opcache.max_accelerated_files",
  151. sizeof("opcache.max_accelerated_files")-1)) == NULL) {
  152. return FAILURE;
  153. }
  154. ini_entry->value = zend_string_init_interned(new_new_value, strlen(new_new_value), 1);
  155. }
  156. *p = size;
  157. return SUCCESS;
  158. }
  159. static ZEND_INI_MH(OnUpdateMaxWastedPercentage)
  160. {
  161. double *p;
  162. zend_long percentage;
  163. #ifndef ZTS
  164. char *base = (char *) mh_arg2;
  165. #else
  166. char *base = (char *) ts_resource(*((int *) mh_arg2));
  167. #endif
  168. /* keep the compiler happy */
  169. (void)entry; (void)mh_arg2; (void)mh_arg3; (void)stage;
  170. p = (double *) (base + (size_t)mh_arg1);
  171. percentage = atoi(ZSTR_VAL(new_value));
  172. if (percentage <= 0 || percentage > 50) {
  173. const char *new_new_value = "5";
  174. zend_ini_entry *ini_entry;
  175. percentage = 5;
  176. zend_accel_error(ACCEL_LOG_WARNING, "opcache.max_wasted_percentage must be set between 1 and 50.\n");
  177. zend_accel_error(ACCEL_LOG_WARNING, ACCELERATOR_PRODUCT_NAME " will use 5%%.\n");
  178. if ((ini_entry = zend_hash_str_find_ptr(EG(ini_directives),
  179. "opcache.max_wasted_percentage",
  180. sizeof("opcache.max_wasted_percentage")-1)) == NULL) {
  181. return FAILURE;
  182. }
  183. ini_entry->value = zend_string_init_interned(new_new_value, strlen(new_new_value), 1);
  184. }
  185. *p = (double)percentage / 100.0;
  186. return SUCCESS;
  187. }
  188. static ZEND_INI_MH(OnEnable)
  189. {
  190. if (stage == ZEND_INI_STAGE_STARTUP ||
  191. stage == ZEND_INI_STAGE_SHUTDOWN ||
  192. stage == ZEND_INI_STAGE_DEACTIVATE) {
  193. return OnUpdateBool(entry, new_value, mh_arg1, mh_arg2, mh_arg3, stage);
  194. } else {
  195. /* It may be only temporary disabled */
  196. zend_bool *p;
  197. #ifndef ZTS
  198. char *base = (char *) mh_arg2;
  199. #else
  200. char *base = (char *) ts_resource(*((int *) mh_arg2));
  201. #endif
  202. p = (zend_bool *) (base+(size_t) mh_arg1);
  203. if ((ZSTR_LEN(new_value) == 2 && strcasecmp("on", ZSTR_VAL(new_value)) == 0) ||
  204. (ZSTR_LEN(new_value) == 3 && strcasecmp("yes", ZSTR_VAL(new_value)) == 0) ||
  205. (ZSTR_LEN(new_value) == 4 && strcasecmp("true", ZSTR_VAL(new_value)) == 0) ||
  206. atoi(ZSTR_VAL(new_value)) != 0) {
  207. zend_error(E_WARNING, ACCELERATOR_PRODUCT_NAME " can't be temporary enabled (it may be only disabled till the end of request)");
  208. return FAILURE;
  209. } else {
  210. *p = 0;
  211. ZCG(accelerator_enabled) = 0;
  212. return SUCCESS;
  213. }
  214. }
  215. }
  216. #ifdef HAVE_OPCACHE_FILE_CACHE
  217. static ZEND_INI_MH(OnUpdateFileCache)
  218. {
  219. if (new_value) {
  220. if (!ZSTR_LEN(new_value)) {
  221. new_value = NULL;
  222. } else {
  223. zend_stat_t buf;
  224. if (!IS_ABSOLUTE_PATH(ZSTR_VAL(new_value), ZSTR_LEN(new_value)) ||
  225. zend_stat(ZSTR_VAL(new_value), &buf) != 0 ||
  226. !S_ISDIR(buf.st_mode) ||
  227. #ifndef ZEND_WIN32
  228. access(ZSTR_VAL(new_value), R_OK | W_OK | X_OK) != 0) {
  229. #else
  230. _access(ZSTR_VAL(new_value), 06) != 0) {
  231. #endif
  232. zend_accel_error(ACCEL_LOG_WARNING, "opcache.file_cache must be a full path of accessible directory.\n");
  233. new_value = NULL;
  234. }
  235. }
  236. }
  237. OnUpdateString(entry, new_value, mh_arg1, mh_arg2, mh_arg3, stage);
  238. return SUCCESS;
  239. }
  240. #endif
  241. ZEND_INI_BEGIN()
  242. STD_PHP_INI_BOOLEAN("opcache.enable" , "1", PHP_INI_ALL, OnEnable, enabled , zend_accel_globals, accel_globals)
  243. STD_PHP_INI_BOOLEAN("opcache.use_cwd" , "1", PHP_INI_SYSTEM, OnUpdateBool, accel_directives.use_cwd , zend_accel_globals, accel_globals)
  244. STD_PHP_INI_BOOLEAN("opcache.validate_timestamps", "1", PHP_INI_ALL , OnUpdateBool, accel_directives.validate_timestamps, zend_accel_globals, accel_globals)
  245. STD_PHP_INI_BOOLEAN("opcache.validate_permission", "0", PHP_INI_SYSTEM, OnUpdateBool, accel_directives.validate_permission, zend_accel_globals, accel_globals)
  246. #ifndef ZEND_WIN32
  247. STD_PHP_INI_BOOLEAN("opcache.validate_root" , "0", PHP_INI_SYSTEM, OnUpdateBool, accel_directives.validate_root , zend_accel_globals, accel_globals)
  248. #endif
  249. STD_PHP_INI_BOOLEAN("opcache.dups_fix" , "0", PHP_INI_ALL , OnUpdateBool, accel_directives.ignore_dups , zend_accel_globals, accel_globals)
  250. STD_PHP_INI_BOOLEAN("opcache.revalidate_path" , "0", PHP_INI_ALL , OnUpdateBool, accel_directives.revalidate_path , zend_accel_globals, accel_globals)
  251. STD_PHP_INI_ENTRY("opcache.log_verbosity_level" , "1" , PHP_INI_SYSTEM, OnUpdateLong, accel_directives.log_verbosity_level, zend_accel_globals, accel_globals)
  252. STD_PHP_INI_ENTRY("opcache.memory_consumption" , "128" , PHP_INI_SYSTEM, OnUpdateMemoryConsumption, accel_directives.memory_consumption, zend_accel_globals, accel_globals)
  253. STD_PHP_INI_ENTRY("opcache.interned_strings_buffer", "8" , PHP_INI_SYSTEM, OnUpdateLong, accel_directives.interned_strings_buffer, zend_accel_globals, accel_globals)
  254. STD_PHP_INI_ENTRY("opcache.max_accelerated_files" , "10000", PHP_INI_SYSTEM, OnUpdateMaxAcceleratedFiles, accel_directives.max_accelerated_files, zend_accel_globals, accel_globals)
  255. STD_PHP_INI_ENTRY("opcache.max_wasted_percentage" , "5" , PHP_INI_SYSTEM, OnUpdateMaxWastedPercentage, accel_directives.max_wasted_percentage, zend_accel_globals, accel_globals)
  256. STD_PHP_INI_ENTRY("opcache.consistency_checks" , "0" , PHP_INI_ALL , OnUpdateLong, accel_directives.consistency_checks, zend_accel_globals, accel_globals)
  257. STD_PHP_INI_ENTRY("opcache.force_restart_timeout" , "180" , PHP_INI_SYSTEM, OnUpdateLong, accel_directives.force_restart_timeout, zend_accel_globals, accel_globals)
  258. STD_PHP_INI_ENTRY("opcache.revalidate_freq" , "2" , PHP_INI_ALL , OnUpdateLong, accel_directives.revalidate_freq, zend_accel_globals, accel_globals)
  259. STD_PHP_INI_ENTRY("opcache.file_update_protection", "2" , PHP_INI_ALL , OnUpdateLong, accel_directives.file_update_protection, zend_accel_globals, accel_globals)
  260. STD_PHP_INI_ENTRY("opcache.preferred_memory_model", "" , PHP_INI_SYSTEM, OnUpdateStringUnempty, accel_directives.memory_model, zend_accel_globals, accel_globals)
  261. STD_PHP_INI_ENTRY("opcache.blacklist_filename" , "" , PHP_INI_SYSTEM, OnUpdateString, accel_directives.user_blacklist_filename, zend_accel_globals, accel_globals)
  262. STD_PHP_INI_ENTRY("opcache.max_file_size" , "0" , PHP_INI_SYSTEM, OnUpdateLong, accel_directives.max_file_size, zend_accel_globals, accel_globals)
  263. STD_PHP_INI_BOOLEAN("opcache.protect_memory" , "0" , PHP_INI_SYSTEM, OnUpdateBool, accel_directives.protect_memory, zend_accel_globals, accel_globals)
  264. STD_PHP_INI_BOOLEAN("opcache.save_comments" , "1" , PHP_INI_SYSTEM, OnUpdateBool, accel_directives.save_comments, zend_accel_globals, accel_globals)
  265. STD_PHP_INI_ENTRY("opcache.optimization_level" , DEFAULT_OPTIMIZATION_LEVEL , PHP_INI_SYSTEM, OnUpdateLong, accel_directives.optimization_level, zend_accel_globals, accel_globals)
  266. STD_PHP_INI_ENTRY("opcache.opt_debug_level" , "0" , PHP_INI_SYSTEM, OnUpdateLong, accel_directives.opt_debug_level, zend_accel_globals, accel_globals)
  267. STD_PHP_INI_BOOLEAN("opcache.enable_file_override" , "0" , PHP_INI_SYSTEM, OnUpdateBool, accel_directives.file_override_enabled, zend_accel_globals, accel_globals)
  268. STD_PHP_INI_BOOLEAN("opcache.enable_cli" , "0" , PHP_INI_SYSTEM, OnUpdateBool, accel_directives.enable_cli, zend_accel_globals, accel_globals)
  269. STD_PHP_INI_ENTRY("opcache.error_log" , "" , PHP_INI_SYSTEM, OnUpdateString, accel_directives.error_log, zend_accel_globals, accel_globals)
  270. STD_PHP_INI_ENTRY("opcache.restrict_api" , "" , PHP_INI_SYSTEM, OnUpdateString, accel_directives.restrict_api, zend_accel_globals, accel_globals)
  271. #ifndef ZEND_WIN32
  272. STD_PHP_INI_ENTRY("opcache.lockfile_path" , "/tmp" , PHP_INI_SYSTEM, OnUpdateString, accel_directives.lockfile_path, zend_accel_globals, accel_globals)
  273. #else
  274. STD_PHP_INI_ENTRY("opcache.mmap_base", NULL, PHP_INI_SYSTEM, OnUpdateString, accel_directives.mmap_base, zend_accel_globals, accel_globals)
  275. #endif
  276. #ifdef HAVE_OPCACHE_FILE_CACHE
  277. STD_PHP_INI_ENTRY("opcache.file_cache" , NULL , PHP_INI_SYSTEM, OnUpdateFileCache, accel_directives.file_cache, zend_accel_globals, accel_globals)
  278. STD_PHP_INI_BOOLEAN("opcache.file_cache_only" , "0" , PHP_INI_SYSTEM, OnUpdateBool, accel_directives.file_cache_only, zend_accel_globals, accel_globals)
  279. STD_PHP_INI_BOOLEAN("opcache.file_cache_consistency_checks" , "1" , PHP_INI_SYSTEM, OnUpdateBool, accel_directives.file_cache_consistency_checks, zend_accel_globals, accel_globals)
  280. #endif
  281. #if ENABLE_FILE_CACHE_FALLBACK
  282. STD_PHP_INI_BOOLEAN("opcache.file_cache_fallback" , "1" , PHP_INI_SYSTEM, OnUpdateBool, accel_directives.file_cache_fallback, zend_accel_globals, accel_globals)
  283. #endif
  284. #ifdef HAVE_HUGE_CODE_PAGES
  285. STD_PHP_INI_BOOLEAN("opcache.huge_code_pages" , "0" , PHP_INI_SYSTEM, OnUpdateBool, accel_directives.huge_code_pages, zend_accel_globals, accel_globals)
  286. #endif
  287. ZEND_INI_END()
  288. static int filename_is_in_cache(zend_string *filename)
  289. {
  290. char *key;
  291. int key_length;
  292. key = accel_make_persistent_key(ZSTR_VAL(filename), ZSTR_LEN(filename), &key_length);
  293. if (key != NULL) {
  294. zend_persistent_script *persistent_script = zend_accel_hash_str_find(&ZCSG(hash), key, key_length);
  295. if (persistent_script && !persistent_script->corrupted) {
  296. zend_file_handle handle = {{0}, NULL, NULL, 0, 0};
  297. handle.filename = ZSTR_VAL(filename);
  298. handle.type = ZEND_HANDLE_FILENAME;
  299. if (ZCG(accel_directives).validate_timestamps) {
  300. return validate_timestamp_and_record_ex(persistent_script, &handle) == SUCCESS;
  301. }
  302. return 1;
  303. }
  304. }
  305. return 0;
  306. }
  307. static int accel_file_in_cache(INTERNAL_FUNCTION_PARAMETERS)
  308. {
  309. zval zfilename;
  310. if (ZEND_NUM_ARGS() != 1 ||
  311. zend_get_parameters_array_ex(1, &zfilename) == FAILURE ||
  312. Z_TYPE(zfilename) != IS_STRING ||
  313. Z_STRLEN(zfilename) == 0) {
  314. return 0;
  315. }
  316. return filename_is_in_cache(Z_STR(zfilename));
  317. }
  318. static ZEND_NAMED_FUNCTION(accel_file_exists)
  319. {
  320. if (accel_file_in_cache(INTERNAL_FUNCTION_PARAM_PASSTHRU)) {
  321. RETURN_TRUE;
  322. } else {
  323. orig_file_exists(INTERNAL_FUNCTION_PARAM_PASSTHRU);
  324. }
  325. }
  326. static ZEND_NAMED_FUNCTION(accel_is_file)
  327. {
  328. if (accel_file_in_cache(INTERNAL_FUNCTION_PARAM_PASSTHRU)) {
  329. RETURN_TRUE;
  330. } else {
  331. orig_is_file(INTERNAL_FUNCTION_PARAM_PASSTHRU);
  332. }
  333. }
  334. static ZEND_NAMED_FUNCTION(accel_is_readable)
  335. {
  336. if (accel_file_in_cache(INTERNAL_FUNCTION_PARAM_PASSTHRU)) {
  337. RETURN_TRUE;
  338. } else {
  339. orig_is_readable(INTERNAL_FUNCTION_PARAM_PASSTHRU);
  340. }
  341. }
  342. static ZEND_MINIT_FUNCTION(zend_accelerator)
  343. {
  344. (void)type; /* keep the compiler happy */
  345. REGISTER_INI_ENTRIES();
  346. return SUCCESS;
  347. }
  348. void zend_accel_override_file_functions(void)
  349. {
  350. zend_function *old_function;
  351. if (ZCG(enabled) && accel_startup_ok && ZCG(accel_directives).file_override_enabled) {
  352. #ifdef HAVE_OPCACHE_FILE_CACHE
  353. if (file_cache_only) {
  354. zend_accel_error(ACCEL_LOG_WARNING, "file_override_enabled has no effect when file_cache_only is set");
  355. return;
  356. }
  357. #endif
  358. /* override file_exists */
  359. if ((old_function = zend_hash_str_find_ptr(CG(function_table), "file_exists", sizeof("file_exists")-1)) != NULL) {
  360. orig_file_exists = old_function->internal_function.handler;
  361. old_function->internal_function.handler = accel_file_exists;
  362. }
  363. if ((old_function = zend_hash_str_find_ptr(CG(function_table), "is_file", sizeof("is_file")-1)) != NULL) {
  364. orig_is_file = old_function->internal_function.handler;
  365. old_function->internal_function.handler = accel_is_file;
  366. }
  367. if ((old_function = zend_hash_str_find_ptr(CG(function_table), "is_readable", sizeof("is_readable")-1)) != NULL) {
  368. orig_is_readable = old_function->internal_function.handler;
  369. old_function->internal_function.handler = accel_is_readable;
  370. }
  371. }
  372. }
  373. static ZEND_MSHUTDOWN_FUNCTION(zend_accelerator)
  374. {
  375. (void)type; /* keep the compiler happy */
  376. UNREGISTER_INI_ENTRIES();
  377. accel_shutdown();
  378. return SUCCESS;
  379. }
  380. void zend_accel_info(ZEND_MODULE_INFO_FUNC_ARGS)
  381. {
  382. php_info_print_table_start();
  383. if (
  384. #ifdef HAVE_OPCACHE_FILE_CACHE
  385. (ZCG(accelerator_enabled) || file_cache_only)
  386. #else
  387. (ZCG(accelerator_enabled))
  388. #endif
  389. ) {
  390. php_info_print_table_row(2, "Opcode Caching", "Up and Running");
  391. } else {
  392. php_info_print_table_row(2, "Opcode Caching", "Disabled");
  393. }
  394. if (ZCG(enabled) && accel_startup_ok && ZCG(accel_directives).optimization_level) {
  395. php_info_print_table_row(2, "Optimization", "Enabled");
  396. } else {
  397. php_info_print_table_row(2, "Optimization", "Disabled");
  398. }
  399. #ifdef HAVE_OPCACHE_FILE_CACHE
  400. if (!file_cache_only) {
  401. php_info_print_table_row(2, "SHM Cache", "Enabled");
  402. } else {
  403. php_info_print_table_row(2, "SHM Cache", "Disabled");
  404. }
  405. if (ZCG(accel_directives).file_cache) {
  406. php_info_print_table_row(2, "File Cache", "Enabled");
  407. } else {
  408. php_info_print_table_row(2, "File Cache", "Disabled");
  409. }
  410. if (file_cache_only) {
  411. if (!accel_startup_ok || zps_api_failure_reason) {
  412. php_info_print_table_row(2, "Startup Failed", zps_api_failure_reason);
  413. } else {
  414. php_info_print_table_row(2, "Startup", "OK");
  415. }
  416. } else
  417. #endif
  418. if (ZCG(enabled)) {
  419. if (!accel_startup_ok || zps_api_failure_reason) {
  420. php_info_print_table_row(2, "Startup Failed", zps_api_failure_reason);
  421. } else {
  422. char buf[32];
  423. php_info_print_table_row(2, "Startup", "OK");
  424. php_info_print_table_row(2, "Shared memory model", zend_accel_get_shared_model());
  425. snprintf(buf, sizeof(buf), ZEND_ULONG_FMT, ZCSG(hits));
  426. php_info_print_table_row(2, "Cache hits", buf);
  427. snprintf(buf, sizeof(buf), ZEND_ULONG_FMT, ZSMMG(memory_exhausted)?ZCSG(misses):ZCSG(misses)-ZCSG(blacklist_misses));
  428. php_info_print_table_row(2, "Cache misses", buf);
  429. snprintf(buf, sizeof(buf), ZEND_LONG_FMT, ZCG(accel_directives).memory_consumption-zend_shared_alloc_get_free_memory()-ZSMMG(wasted_shared_memory));
  430. php_info_print_table_row(2, "Used memory", buf);
  431. snprintf(buf, sizeof(buf), "%zu", zend_shared_alloc_get_free_memory());
  432. php_info_print_table_row(2, "Free memory", buf);
  433. snprintf(buf, sizeof(buf), "%zu", ZSMMG(wasted_shared_memory));
  434. php_info_print_table_row(2, "Wasted memory", buf);
  435. if (ZCSG(interned_strings).start && ZCSG(interned_strings).end) {
  436. snprintf(buf, sizeof(buf), "%zu", (size_t)((char*)ZCSG(interned_strings).top - (char*)ZCSG(interned_strings).start));
  437. php_info_print_table_row(2, "Interned Strings Used memory", buf);
  438. snprintf(buf, sizeof(buf), "%zu", (size_t)((char*)ZCSG(interned_strings).end - (char*)ZCSG(interned_strings).top));
  439. php_info_print_table_row(2, "Interned Strings Free memory", buf);
  440. }
  441. snprintf(buf, sizeof(buf), "%" PRIu32, ZCSG(hash).num_direct_entries);
  442. php_info_print_table_row(2, "Cached scripts", buf);
  443. snprintf(buf, sizeof(buf), "%" PRIu32, ZCSG(hash).num_entries);
  444. php_info_print_table_row(2, "Cached keys", buf);
  445. snprintf(buf, sizeof(buf), "%" PRIu32, ZCSG(hash).max_num_entries);
  446. php_info_print_table_row(2, "Max keys", buf);
  447. snprintf(buf, sizeof(buf), ZEND_ULONG_FMT, ZCSG(oom_restarts));
  448. php_info_print_table_row(2, "OOM restarts", buf);
  449. snprintf(buf, sizeof(buf), ZEND_ULONG_FMT, ZCSG(hash_restarts));
  450. php_info_print_table_row(2, "Hash keys restarts", buf);
  451. snprintf(buf, sizeof(buf), ZEND_ULONG_FMT, ZCSG(manual_restarts));
  452. php_info_print_table_row(2, "Manual restarts", buf);
  453. }
  454. }
  455. php_info_print_table_end();
  456. DISPLAY_INI_ENTRIES();
  457. }
  458. static zend_module_entry accel_module_entry = {
  459. STANDARD_MODULE_HEADER,
  460. ACCELERATOR_PRODUCT_NAME,
  461. accel_functions,
  462. ZEND_MINIT(zend_accelerator),
  463. ZEND_MSHUTDOWN(zend_accelerator),
  464. NULL,
  465. NULL,
  466. zend_accel_info,
  467. PHP_VERSION,
  468. NO_MODULE_GLOBALS,
  469. accel_post_deactivate,
  470. STANDARD_MODULE_PROPERTIES_EX
  471. };
  472. int start_accel_module(void)
  473. {
  474. return zend_startup_module(&accel_module_entry);
  475. }
  476. /* {{{ proto array accelerator_get_scripts()
  477. Get the scripts which are accelerated by ZendAccelerator */
  478. static int accelerator_get_scripts(zval *return_value)
  479. {
  480. uint32_t i;
  481. zval persistent_script_report;
  482. zend_accel_hash_entry *cache_entry;
  483. struct tm *ta;
  484. struct timeval exec_time;
  485. struct timeval fetch_time;
  486. if (!ZCG(accelerator_enabled) || accelerator_shm_read_lock() != SUCCESS) {
  487. return 0;
  488. }
  489. array_init(return_value);
  490. for (i = 0; i<ZCSG(hash).max_num_entries; i++) {
  491. for (cache_entry = ZCSG(hash).hash_table[i]; cache_entry; cache_entry = cache_entry->next) {
  492. zend_persistent_script *script;
  493. char *str;
  494. size_t len;
  495. if (cache_entry->indirect) continue;
  496. script = (zend_persistent_script *)cache_entry->data;
  497. array_init(&persistent_script_report);
  498. add_assoc_str(&persistent_script_report, "full_path", zend_string_dup(script->script.filename, 0));
  499. add_assoc_long(&persistent_script_report, "hits", (zend_long)script->dynamic_members.hits);
  500. add_assoc_long(&persistent_script_report, "memory_consumption", script->dynamic_members.memory_consumption);
  501. ta = localtime(&script->dynamic_members.last_used);
  502. str = asctime(ta);
  503. len = strlen(str);
  504. if (len > 0 && str[len - 1] == '\n') len--;
  505. add_assoc_stringl(&persistent_script_report, "last_used", str, len);
  506. add_assoc_long(&persistent_script_report, "last_used_timestamp", script->dynamic_members.last_used);
  507. if (ZCG(accel_directives).validate_timestamps) {
  508. add_assoc_long(&persistent_script_report, "timestamp", (zend_long)script->timestamp);
  509. }
  510. timerclear(&exec_time);
  511. timerclear(&fetch_time);
  512. zend_hash_str_update(Z_ARRVAL_P(return_value), cache_entry->key, cache_entry->key_length, &persistent_script_report);
  513. }
  514. }
  515. accelerator_shm_read_unlock();
  516. return 1;
  517. }
  518. /* {{{ proto array accelerator_get_status([bool fetch_scripts])
  519. Obtain statistics information regarding code acceleration */
  520. static ZEND_FUNCTION(opcache_get_status)
  521. {
  522. zend_long reqs;
  523. zval memory_usage, statistics, scripts;
  524. zend_bool fetch_scripts = 1;
  525. if (zend_parse_parameters(ZEND_NUM_ARGS(), "|b", &fetch_scripts) == FAILURE) {
  526. return;
  527. }
  528. if (!validate_api_restriction()) {
  529. RETURN_FALSE;
  530. }
  531. if (!accel_startup_ok) {
  532. RETURN_FALSE;
  533. }
  534. array_init(return_value);
  535. /* Trivia */
  536. add_assoc_bool(return_value, "opcache_enabled", ZCG(accelerator_enabled));
  537. #ifdef HAVE_OPCACHE_FILE_CACHE
  538. if (ZCG(accel_directives).file_cache) {
  539. add_assoc_string(return_value, "file_cache", ZCG(accel_directives).file_cache);
  540. }
  541. if (file_cache_only) {
  542. add_assoc_bool(return_value, "file_cache_only", 1);
  543. return;
  544. }
  545. #endif
  546. add_assoc_bool(return_value, "cache_full", ZSMMG(memory_exhausted));
  547. add_assoc_bool(return_value, "restart_pending", ZCSG(restart_pending));
  548. add_assoc_bool(return_value, "restart_in_progress", ZCSG(restart_in_progress));
  549. /* Memory usage statistics */
  550. array_init(&memory_usage);
  551. add_assoc_long(&memory_usage, "used_memory", ZCG(accel_directives).memory_consumption-zend_shared_alloc_get_free_memory()-ZSMMG(wasted_shared_memory));
  552. add_assoc_long(&memory_usage, "free_memory", zend_shared_alloc_get_free_memory());
  553. add_assoc_long(&memory_usage, "wasted_memory", ZSMMG(wasted_shared_memory));
  554. add_assoc_double(&memory_usage, "current_wasted_percentage", (((double) ZSMMG(wasted_shared_memory))/ZCG(accel_directives).memory_consumption)*100.0);
  555. add_assoc_zval(return_value, "memory_usage", &memory_usage);
  556. if (ZCSG(interned_strings).start && ZCSG(interned_strings).end) {
  557. zval interned_strings_usage;
  558. array_init(&interned_strings_usage);
  559. add_assoc_long(&interned_strings_usage, "buffer_size", (char*)ZCSG(interned_strings).end - (char*)ZCSG(interned_strings).start);
  560. add_assoc_long(&interned_strings_usage, "used_memory", (char*)ZCSG(interned_strings).top - (char*)ZCSG(interned_strings).start);
  561. add_assoc_long(&interned_strings_usage, "free_memory", (char*)ZCSG(interned_strings).end - (char*)ZCSG(interned_strings).top);
  562. add_assoc_long(&interned_strings_usage, "number_of_strings", ZCSG(interned_strings).nNumOfElements);
  563. add_assoc_zval(return_value, "interned_strings_usage", &interned_strings_usage);
  564. }
  565. /* Accelerator statistics */
  566. array_init(&statistics);
  567. add_assoc_long(&statistics, "num_cached_scripts", ZCSG(hash).num_direct_entries);
  568. add_assoc_long(&statistics, "num_cached_keys", ZCSG(hash).num_entries);
  569. add_assoc_long(&statistics, "max_cached_keys", ZCSG(hash).max_num_entries);
  570. add_assoc_long(&statistics, "hits", (zend_long)ZCSG(hits));
  571. add_assoc_long(&statistics, "start_time", ZCSG(start_time));
  572. add_assoc_long(&statistics, "last_restart_time", ZCSG(last_restart_time));
  573. add_assoc_long(&statistics, "oom_restarts", ZCSG(oom_restarts));
  574. add_assoc_long(&statistics, "hash_restarts", ZCSG(hash_restarts));
  575. add_assoc_long(&statistics, "manual_restarts", ZCSG(manual_restarts));
  576. add_assoc_long(&statistics, "misses", ZSMMG(memory_exhausted)?ZCSG(misses):ZCSG(misses)-ZCSG(blacklist_misses));
  577. add_assoc_long(&statistics, "blacklist_misses", ZCSG(blacklist_misses));
  578. reqs = ZCSG(hits)+ZCSG(misses);
  579. add_assoc_double(&statistics, "blacklist_miss_ratio", reqs?(((double) ZCSG(blacklist_misses))/reqs)*100.0:0);
  580. add_assoc_double(&statistics, "opcache_hit_rate", reqs?(((double) ZCSG(hits))/reqs)*100.0:0);
  581. add_assoc_zval(return_value, "opcache_statistics", &statistics);
  582. if (fetch_scripts) {
  583. /* accelerated scripts */
  584. if (accelerator_get_scripts(&scripts)) {
  585. add_assoc_zval(return_value, "scripts", &scripts);
  586. }
  587. }
  588. }
  589. static int add_blacklist_path(zend_blacklist_entry *p, zval *return_value)
  590. {
  591. add_next_index_stringl(return_value, p->path, p->path_length);
  592. return 0;
  593. }
  594. /* {{{ proto array accelerator_get_configuration()
  595. Obtain configuration information */
  596. static ZEND_FUNCTION(opcache_get_configuration)
  597. {
  598. zval directives, version, blacklist;
  599. if (zend_parse_parameters_none() == FAILURE) {
  600. RETURN_FALSE;
  601. }
  602. if (!validate_api_restriction()) {
  603. RETURN_FALSE;
  604. }
  605. array_init(return_value);
  606. /* directives */
  607. array_init(&directives);
  608. add_assoc_bool(&directives, "opcache.enable", ZCG(enabled));
  609. add_assoc_bool(&directives, "opcache.enable_cli", ZCG(accel_directives).enable_cli);
  610. add_assoc_bool(&directives, "opcache.use_cwd", ZCG(accel_directives).use_cwd);
  611. add_assoc_bool(&directives, "opcache.validate_timestamps", ZCG(accel_directives).validate_timestamps);
  612. add_assoc_bool(&directives, "opcache.validate_permission", ZCG(accel_directives).validate_permission);
  613. #ifndef ZEND_WIN32
  614. add_assoc_bool(&directives, "opcache.validate_root", ZCG(accel_directives).validate_root);
  615. #endif
  616. add_assoc_bool(&directives, "opcache.dups_fix", ZCG(accel_directives).ignore_dups);
  617. add_assoc_bool(&directives, "opcache.revalidate_path", ZCG(accel_directives).revalidate_path);
  618. add_assoc_long(&directives, "opcache.log_verbosity_level", ZCG(accel_directives).log_verbosity_level);
  619. add_assoc_long(&directives, "opcache.memory_consumption", ZCG(accel_directives).memory_consumption);
  620. add_assoc_long(&directives, "opcache.interned_strings_buffer",ZCG(accel_directives).interned_strings_buffer);
  621. add_assoc_long(&directives, "opcache.max_accelerated_files", ZCG(accel_directives).max_accelerated_files);
  622. add_assoc_double(&directives, "opcache.max_wasted_percentage", ZCG(accel_directives).max_wasted_percentage);
  623. add_assoc_long(&directives, "opcache.consistency_checks", ZCG(accel_directives).consistency_checks);
  624. add_assoc_long(&directives, "opcache.force_restart_timeout", ZCG(accel_directives).force_restart_timeout);
  625. add_assoc_long(&directives, "opcache.revalidate_freq", ZCG(accel_directives).revalidate_freq);
  626. add_assoc_string(&directives, "opcache.preferred_memory_model", STRING_NOT_NULL(ZCG(accel_directives).memory_model));
  627. add_assoc_string(&directives, "opcache.blacklist_filename", STRING_NOT_NULL(ZCG(accel_directives).user_blacklist_filename));
  628. add_assoc_long(&directives, "opcache.max_file_size", ZCG(accel_directives).max_file_size);
  629. add_assoc_string(&directives, "opcache.error_log", STRING_NOT_NULL(ZCG(accel_directives).error_log));
  630. add_assoc_bool(&directives, "opcache.protect_memory", ZCG(accel_directives).protect_memory);
  631. add_assoc_bool(&directives, "opcache.save_comments", ZCG(accel_directives).save_comments);
  632. add_assoc_bool(&directives, "opcache.enable_file_override", ZCG(accel_directives).file_override_enabled);
  633. add_assoc_long(&directives, "opcache.optimization_level", ZCG(accel_directives).optimization_level);
  634. #ifndef ZEND_WIN32
  635. add_assoc_string(&directives, "opcache.lockfile_path", STRING_NOT_NULL(ZCG(accel_directives).lockfile_path));
  636. #else
  637. add_assoc_string(&directives, "opcache.mmap_base", STRING_NOT_NULL(ZCG(accel_directives).mmap_base));
  638. #endif
  639. #ifdef HAVE_OPCACHE_FILE_CACHE
  640. add_assoc_string(&directives, "opcache.file_cache", ZCG(accel_directives).file_cache ? ZCG(accel_directives).file_cache : "");
  641. add_assoc_bool(&directives, "opcache.file_cache_only", ZCG(accel_directives).file_cache_only);
  642. add_assoc_bool(&directives, "opcache.file_cache_consistency_checks", ZCG(accel_directives).file_cache_consistency_checks);
  643. #endif
  644. #if ENABLE_FILE_CACHE_FALLBACK
  645. add_assoc_bool(&directives, "opcache.file_cache_fallback", ZCG(accel_directives).file_cache_fallback);
  646. #endif
  647. add_assoc_long(&directives, "opcache.file_update_protection", ZCG(accel_directives).file_update_protection);
  648. add_assoc_long(&directives, "opcache.opt_debug_level", ZCG(accel_directives).opt_debug_level);
  649. add_assoc_string(&directives, "opcache.restrict_api", STRING_NOT_NULL(ZCG(accel_directives).restrict_api));
  650. #ifdef HAVE_HUGE_CODE_PAGES
  651. add_assoc_bool(&directives, "opcache.huge_code_pages", ZCG(accel_directives).huge_code_pages);
  652. #endif
  653. add_assoc_zval(return_value, "directives", &directives);
  654. /*version */
  655. array_init(&version);
  656. add_assoc_string(&version, "version", PHP_VERSION);
  657. add_assoc_string(&version, "opcache_product_name", ACCELERATOR_PRODUCT_NAME);
  658. add_assoc_zval(return_value, "version", &version);
  659. /* blacklist */
  660. array_init(&blacklist);
  661. zend_accel_blacklist_apply(&accel_blacklist, add_blacklist_path, &blacklist);
  662. add_assoc_zval(return_value, "blacklist", &blacklist);
  663. }
  664. /* {{{ proto void accelerator_reset()
  665. Request that the contents of the opcode cache to be reset */
  666. static ZEND_FUNCTION(opcache_reset)
  667. {
  668. if (zend_parse_parameters_none() == FAILURE) {
  669. RETURN_FALSE;
  670. }
  671. if (!validate_api_restriction()) {
  672. RETURN_FALSE;
  673. }
  674. if ((!ZCG(enabled) || !accel_startup_ok || !ZCSG(accelerator_enabled))
  675. #if ENABLE_FILE_CACHE_FALLBACK
  676. && !fallback_process
  677. #endif
  678. ) {
  679. RETURN_FALSE;
  680. }
  681. /* exclusive lock */
  682. zend_shared_alloc_lock();
  683. zend_accel_schedule_restart(ACCEL_RESTART_USER);
  684. zend_shared_alloc_unlock();
  685. RETURN_TRUE;
  686. }
  687. /* {{{ proto void opcache_invalidate(string $script [, bool $force = false])
  688. Invalidates cached script (in necessary or forced) */
  689. static ZEND_FUNCTION(opcache_invalidate)
  690. {
  691. char *script_name;
  692. size_t script_name_len;
  693. zend_bool force = 0;
  694. if (zend_parse_parameters(ZEND_NUM_ARGS(), "s|b", &script_name, &script_name_len, &force) == FAILURE) {
  695. return;
  696. }
  697. if (!validate_api_restriction()) {
  698. RETURN_FALSE;
  699. }
  700. if (zend_accel_invalidate(script_name, script_name_len, force) == SUCCESS) {
  701. RETURN_TRUE;
  702. } else {
  703. RETURN_FALSE;
  704. }
  705. }
  706. static ZEND_FUNCTION(opcache_compile_file)
  707. {
  708. char *script_name;
  709. size_t script_name_len;
  710. zend_file_handle handle;
  711. zend_op_array *op_array = NULL;
  712. zend_execute_data *orig_execute_data = NULL;
  713. if (zend_parse_parameters(ZEND_NUM_ARGS(), "s", &script_name, &script_name_len) == FAILURE) {
  714. return;
  715. }
  716. if (!ZCG(accelerator_enabled)) {
  717. zend_error(E_NOTICE, ACCELERATOR_PRODUCT_NAME " seems to be disabled, can't compile file");
  718. RETURN_FALSE;
  719. }
  720. handle.filename = script_name;
  721. handle.free_filename = 0;
  722. handle.opened_path = NULL;
  723. handle.type = ZEND_HANDLE_FILENAME;
  724. orig_execute_data = EG(current_execute_data);
  725. zend_try {
  726. op_array = persistent_compile_file(&handle, ZEND_INCLUDE);
  727. } zend_catch {
  728. EG(current_execute_data) = orig_execute_data;
  729. zend_error(E_WARNING, ACCELERATOR_PRODUCT_NAME " could not compile file %s", handle.filename);
  730. } zend_end_try();
  731. if(op_array != NULL) {
  732. destroy_op_array(op_array);
  733. efree(op_array);
  734. RETVAL_TRUE;
  735. } else {
  736. RETVAL_FALSE;
  737. }
  738. zend_destroy_file_handle(&handle);
  739. }
  740. /* {{{ proto bool opcache_is_script_cached(string $script)
  741. Return true if the script is cached in OPCache, false if it is not cached or if OPCache is not running. */
  742. static ZEND_FUNCTION(opcache_is_script_cached)
  743. {
  744. zend_string *script_name;
  745. if (!validate_api_restriction()) {
  746. RETURN_FALSE;
  747. }
  748. if (!ZCG(accelerator_enabled)) {
  749. RETURN_FALSE;
  750. }
  751. if (zend_parse_parameters(ZEND_NUM_ARGS(), "S", &script_name) == FAILURE) {
  752. return;
  753. }
  754. RETURN_BOOL(filename_is_in_cache(script_name));
  755. }