zend_accelerator_module.c 33 KB

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