zend_shared_alloc.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545
  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 <errno.h>
  22. #include "ZendAccelerator.h"
  23. #include "zend_shared_alloc.h"
  24. #ifdef HAVE_UNISTD_H
  25. # include <unistd.h>
  26. #endif
  27. #include <fcntl.h>
  28. #ifndef ZEND_WIN32
  29. # include <sys/types.h>
  30. # include <dirent.h>
  31. # include <signal.h>
  32. # include <sys/stat.h>
  33. # include <stdio.h>
  34. #endif
  35. #ifdef HAVE_MPROTECT
  36. # include "sys/mman.h"
  37. #endif
  38. #define SEM_FILENAME_PREFIX ".ZendSem."
  39. #define S_H(s) g_shared_alloc_handler->s
  40. /* True globals */
  41. /* old/new mapping. We can use true global even for ZTS because its usage
  42. is wrapped with exclusive lock anyway */
  43. static const zend_shared_memory_handlers *g_shared_alloc_handler = NULL;
  44. static const char *g_shared_model;
  45. /* pointer to globals allocated in SHM and shared across processes */
  46. zend_smm_shared_globals *smm_shared_globals;
  47. #ifndef ZEND_WIN32
  48. #ifdef ZTS
  49. static MUTEX_T zts_lock;
  50. #endif
  51. int lock_file;
  52. static char lockfile_name[MAXPATHLEN];
  53. #endif
  54. static const zend_shared_memory_handler_entry handler_table[] = {
  55. #ifdef USE_MMAP
  56. { "mmap", &zend_alloc_mmap_handlers },
  57. #endif
  58. #ifdef USE_SHM
  59. { "shm", &zend_alloc_shm_handlers },
  60. #endif
  61. #ifdef USE_SHM_OPEN
  62. { "posix", &zend_alloc_posix_handlers },
  63. #endif
  64. #ifdef ZEND_WIN32
  65. { "win32", &zend_alloc_win32_handlers },
  66. #endif
  67. { NULL, NULL}
  68. };
  69. #ifndef ZEND_WIN32
  70. void zend_shared_alloc_create_lock(char *lockfile_path)
  71. {
  72. int val;
  73. #ifdef ZTS
  74. zts_lock = tsrm_mutex_alloc();
  75. #endif
  76. snprintf(lockfile_name, sizeof(lockfile_name), "%s/%sXXXXXX", lockfile_path, SEM_FILENAME_PREFIX);
  77. lock_file = mkstemp(lockfile_name);
  78. fchmod(lock_file, 0666);
  79. if (lock_file == -1) {
  80. zend_accel_error(ACCEL_LOG_FATAL, "Unable to create lock file: %s (%d)", strerror(errno), errno);
  81. }
  82. val = fcntl(lock_file, F_GETFD, 0);
  83. val |= FD_CLOEXEC;
  84. fcntl(lock_file, F_SETFD, val);
  85. unlink(lockfile_name);
  86. }
  87. #endif
  88. static void no_memory_bailout(size_t allocate_size, char *error)
  89. {
  90. zend_accel_error(ACCEL_LOG_FATAL, "Unable to allocate shared memory segment of %zu bytes: %s: %s (%d)", allocate_size, error?error:"unknown", strerror(errno), errno );
  91. }
  92. static void copy_shared_segments(void *to, void *from, int count, int size)
  93. {
  94. zend_shared_segment **shared_segments_v = (zend_shared_segment **)to;
  95. void *shared_segments_to_p = ((char *)to + count*(sizeof(void *)));
  96. void *shared_segments_from_p = from;
  97. int i;
  98. for (i = 0; i < count; i++) {
  99. shared_segments_v[i] = shared_segments_to_p;
  100. memcpy(shared_segments_to_p, shared_segments_from_p, size);
  101. shared_segments_to_p = ((char *)shared_segments_to_p + size);
  102. shared_segments_from_p = ((char *)shared_segments_from_p + size);
  103. }
  104. }
  105. static int zend_shared_alloc_try(const zend_shared_memory_handler_entry *he, size_t requested_size, zend_shared_segment ***shared_segments_p, int *shared_segments_count, char **error_in)
  106. {
  107. int res;
  108. g_shared_alloc_handler = he->handler;
  109. g_shared_model = he->name;
  110. ZSMMG(shared_segments) = NULL;
  111. ZSMMG(shared_segments_count) = 0;
  112. res = S_H(create_segments)(requested_size, shared_segments_p, shared_segments_count, error_in);
  113. if (res) {
  114. /* this model works! */
  115. return res;
  116. }
  117. if (*shared_segments_p) {
  118. int i;
  119. /* cleanup */
  120. for (i = 0; i < *shared_segments_count; i++) {
  121. if ((*shared_segments_p)[i]->p && (*shared_segments_p)[i]->p != (void *)-1) {
  122. S_H(detach_segment)((*shared_segments_p)[i]);
  123. }
  124. }
  125. free(*shared_segments_p);
  126. *shared_segments_p = NULL;
  127. }
  128. g_shared_alloc_handler = NULL;
  129. return ALLOC_FAILURE;
  130. }
  131. int zend_shared_alloc_startup(size_t requested_size)
  132. {
  133. zend_shared_segment **tmp_shared_segments;
  134. size_t shared_segments_array_size;
  135. zend_smm_shared_globals tmp_shared_globals, *p_tmp_shared_globals;
  136. char *error_in = NULL;
  137. const zend_shared_memory_handler_entry *he;
  138. int res = ALLOC_FAILURE;
  139. /* shared_free must be valid before we call zend_shared_alloc()
  140. * - make it temporarily point to a local variable
  141. */
  142. smm_shared_globals = &tmp_shared_globals;
  143. ZSMMG(shared_free) = requested_size; /* goes to tmp_shared_globals.shared_free */
  144. #ifndef ZEND_WIN32
  145. zend_shared_alloc_create_lock(ZCG(accel_directives).lockfile_path);
  146. #else
  147. zend_shared_alloc_create_lock();
  148. #endif
  149. if (ZCG(accel_directives).memory_model && ZCG(accel_directives).memory_model[0]) {
  150. char *model = ZCG(accel_directives).memory_model;
  151. /* "cgi" is really "shm"... */
  152. if (strncmp(ZCG(accel_directives).memory_model, "cgi", sizeof("cgi")) == 0) {
  153. model = "shm";
  154. }
  155. for (he = handler_table; he->name; he++) {
  156. if (strcmp(model, he->name) == 0) {
  157. res = zend_shared_alloc_try(he, requested_size, &ZSMMG(shared_segments), &ZSMMG(shared_segments_count), &error_in);
  158. if (res) {
  159. /* this model works! */
  160. }
  161. break;
  162. }
  163. }
  164. }
  165. if (res == FAILED_REATTACHED) {
  166. smm_shared_globals = NULL;
  167. return res;
  168. }
  169. #if ENABLE_FILE_CACHE_FALLBACK
  170. if (ALLOC_FALLBACK == res) {
  171. return ALLOC_FALLBACK;
  172. }
  173. #endif
  174. if (!g_shared_alloc_handler) {
  175. /* try memory handlers in order */
  176. for (he = handler_table; he->name; he++) {
  177. res = zend_shared_alloc_try(he, requested_size, &ZSMMG(shared_segments), &ZSMMG(shared_segments_count), &error_in);
  178. if (res) {
  179. /* this model works! */
  180. break;
  181. }
  182. }
  183. }
  184. if (!g_shared_alloc_handler) {
  185. no_memory_bailout(requested_size, error_in);
  186. return ALLOC_FAILURE;
  187. }
  188. if (res == SUCCESSFULLY_REATTACHED) {
  189. return res;
  190. }
  191. #if ENABLE_FILE_CACHE_FALLBACK
  192. if (ALLOC_FALLBACK == res) {
  193. return ALLOC_FALLBACK;
  194. }
  195. #endif
  196. shared_segments_array_size = ZSMMG(shared_segments_count) * S_H(segment_type_size)();
  197. /* move shared_segments and shared_free to shared memory */
  198. ZCG(locked) = 1; /* no need to perform a real lock at this point */
  199. p_tmp_shared_globals = (zend_smm_shared_globals *) zend_shared_alloc(sizeof(zend_smm_shared_globals));
  200. if (!p_tmp_shared_globals) {
  201. zend_accel_error(ACCEL_LOG_FATAL, "Insufficient shared memory!");
  202. return ALLOC_FAILURE;
  203. }
  204. memset(p_tmp_shared_globals, 0, sizeof(zend_smm_shared_globals));
  205. tmp_shared_segments = zend_shared_alloc(shared_segments_array_size + ZSMMG(shared_segments_count) * sizeof(void *));
  206. if (!tmp_shared_segments) {
  207. zend_accel_error(ACCEL_LOG_FATAL, "Insufficient shared memory!");
  208. return ALLOC_FAILURE;
  209. }
  210. copy_shared_segments(tmp_shared_segments, ZSMMG(shared_segments)[0], ZSMMG(shared_segments_count), S_H(segment_type_size)());
  211. *p_tmp_shared_globals = tmp_shared_globals;
  212. smm_shared_globals = p_tmp_shared_globals;
  213. free(ZSMMG(shared_segments));
  214. ZSMMG(shared_segments) = tmp_shared_segments;
  215. ZSMMG(shared_memory_state).positions = (int *)zend_shared_alloc(sizeof(int) * ZSMMG(shared_segments_count));
  216. if (!ZSMMG(shared_memory_state).positions) {
  217. zend_accel_error(ACCEL_LOG_FATAL, "Insufficient shared memory!");
  218. return ALLOC_FAILURE;
  219. }
  220. ZCG(locked) = 0;
  221. return res;
  222. }
  223. void zend_shared_alloc_shutdown(void)
  224. {
  225. zend_shared_segment **tmp_shared_segments;
  226. size_t shared_segments_array_size;
  227. zend_smm_shared_globals tmp_shared_globals;
  228. int i;
  229. tmp_shared_globals = *smm_shared_globals;
  230. smm_shared_globals = &tmp_shared_globals;
  231. shared_segments_array_size = ZSMMG(shared_segments_count) * (S_H(segment_type_size)() + sizeof(void *));
  232. tmp_shared_segments = emalloc(shared_segments_array_size);
  233. copy_shared_segments(tmp_shared_segments, ZSMMG(shared_segments)[0], ZSMMG(shared_segments_count), S_H(segment_type_size)());
  234. ZSMMG(shared_segments) = tmp_shared_segments;
  235. for (i = 0; i < ZSMMG(shared_segments_count); i++) {
  236. S_H(detach_segment)(ZSMMG(shared_segments)[i]);
  237. }
  238. efree(ZSMMG(shared_segments));
  239. ZSMMG(shared_segments) = NULL;
  240. g_shared_alloc_handler = NULL;
  241. #ifndef ZEND_WIN32
  242. close(lock_file);
  243. #endif
  244. }
  245. static size_t zend_shared_alloc_get_largest_free_block(void)
  246. {
  247. int i;
  248. size_t largest_block_size = 0;
  249. for (i = 0; i < ZSMMG(shared_segments_count); i++) {
  250. size_t block_size = ZSMMG(shared_segments)[i]->size - ZSMMG(shared_segments)[i]->pos;
  251. if (block_size>largest_block_size) {
  252. largest_block_size = block_size;
  253. }
  254. }
  255. return largest_block_size;
  256. }
  257. #define MIN_FREE_MEMORY 64*1024
  258. #define SHARED_ALLOC_FAILED() do { \
  259. zend_accel_error(ACCEL_LOG_WARNING, "Not enough free shared space to allocate "ZEND_LONG_FMT" bytes ("ZEND_LONG_FMT" bytes free)", (zend_long)size, (zend_long)ZSMMG(shared_free)); \
  260. if (zend_shared_alloc_get_largest_free_block() < MIN_FREE_MEMORY) { \
  261. ZSMMG(memory_exhausted) = 1; \
  262. } \
  263. } while (0)
  264. void *zend_shared_alloc(size_t size)
  265. {
  266. int i;
  267. unsigned int block_size = ZEND_ALIGNED_SIZE(size);
  268. #if 1
  269. if (!ZCG(locked)) {
  270. zend_accel_error(ACCEL_LOG_ERROR, "Shared memory lock not obtained");
  271. }
  272. #endif
  273. if (block_size > ZSMMG(shared_free)) { /* No hope to find a big-enough block */
  274. SHARED_ALLOC_FAILED();
  275. return NULL;
  276. }
  277. for (i = 0; i < ZSMMG(shared_segments_count); i++) {
  278. if (ZSMMG(shared_segments)[i]->size - ZSMMG(shared_segments)[i]->pos >= block_size) { /* found a valid block */
  279. void *retval = (void *) (((char *) ZSMMG(shared_segments)[i]->p) + ZSMMG(shared_segments)[i]->pos);
  280. ZSMMG(shared_segments)[i]->pos += block_size;
  281. ZSMMG(shared_free) -= block_size;
  282. ZEND_ASSERT(((zend_uintptr_t)retval & 0x7) == 0); /* should be 8 byte aligned */
  283. return retval;
  284. }
  285. }
  286. SHARED_ALLOC_FAILED();
  287. return NULL;
  288. }
  289. int zend_shared_memdup_size(void *source, size_t size)
  290. {
  291. void *old_p;
  292. zend_ulong key = (zend_ulong)source;
  293. key = (key >> 3) | (key << ((sizeof(key) * 8) - 3)); /* key = _rotr(key, 3);*/
  294. if ((old_p = zend_hash_index_find_ptr(&ZCG(xlat_table), key)) != NULL) {
  295. /* we already duplicated this pointer */
  296. return 0;
  297. }
  298. zend_shared_alloc_register_xlat_entry(source, source);
  299. return ZEND_ALIGNED_SIZE(size);
  300. }
  301. void *_zend_shared_memdup(void *source, size_t size, zend_bool free_source)
  302. {
  303. void *old_p, *retval;
  304. zend_ulong key = (zend_ulong)source;
  305. key = (key >> 3) | (key << ((sizeof(key) * 8) - 3)); /* key = _rotr(key, 3);*/
  306. if ((old_p = zend_hash_index_find_ptr(&ZCG(xlat_table), key)) != NULL) {
  307. /* we already duplicated this pointer */
  308. return old_p;
  309. }
  310. retval = ZCG(mem);
  311. ZCG(mem) = (void*)(((char*)ZCG(mem)) + ZEND_ALIGNED_SIZE(size));
  312. memcpy(retval, source, size);
  313. zend_shared_alloc_register_xlat_entry(source, retval);
  314. if (free_source) {
  315. efree(source);
  316. }
  317. return retval;
  318. }
  319. void zend_shared_alloc_safe_unlock(void)
  320. {
  321. if (ZCG(locked)) {
  322. zend_shared_alloc_unlock();
  323. }
  324. }
  325. void zend_shared_alloc_lock(void)
  326. {
  327. #ifndef ZEND_WIN32
  328. struct flock mem_write_lock;
  329. mem_write_lock.l_type = F_WRLCK;
  330. mem_write_lock.l_whence = SEEK_SET;
  331. mem_write_lock.l_start = 0;
  332. mem_write_lock.l_len = 1;
  333. #ifdef ZTS
  334. tsrm_mutex_lock(zts_lock);
  335. #endif
  336. #if 0
  337. /* this will happen once per process, and will un-globalize mem_write_lock */
  338. if (mem_write_lock.l_pid == -1) {
  339. mem_write_lock.l_pid = getpid();
  340. }
  341. #endif
  342. while (1) {
  343. if (fcntl(lock_file, F_SETLKW, &mem_write_lock) == -1) {
  344. if (errno == EINTR) {
  345. continue;
  346. }
  347. zend_accel_error(ACCEL_LOG_ERROR, "Cannot create lock - %s (%d)", strerror(errno), errno);
  348. }
  349. break;
  350. }
  351. #else
  352. zend_shared_alloc_lock_win32();
  353. #endif
  354. ZCG(locked) = 1;
  355. }
  356. void zend_shared_alloc_unlock(void)
  357. {
  358. #ifndef ZEND_WIN32
  359. struct flock mem_write_unlock;
  360. mem_write_unlock.l_type = F_UNLCK;
  361. mem_write_unlock.l_whence = SEEK_SET;
  362. mem_write_unlock.l_start = 0;
  363. mem_write_unlock.l_len = 1;
  364. #endif
  365. ZCG(locked) = 0;
  366. #ifndef ZEND_WIN32
  367. if (fcntl(lock_file, F_SETLK, &mem_write_unlock) == -1) {
  368. zend_accel_error(ACCEL_LOG_ERROR, "Cannot remove lock - %s (%d)", strerror(errno), errno);
  369. }
  370. #ifdef ZTS
  371. tsrm_mutex_unlock(zts_lock);
  372. #endif
  373. #else
  374. zend_shared_alloc_unlock_win32();
  375. #endif
  376. }
  377. void zend_shared_alloc_init_xlat_table(void)
  378. {
  379. /* Prepare translation table */
  380. zend_hash_init(&ZCG(xlat_table), 128, NULL, NULL, 0);
  381. }
  382. void zend_shared_alloc_destroy_xlat_table(void)
  383. {
  384. /* Destroy translation table */
  385. zend_hash_destroy(&ZCG(xlat_table));
  386. }
  387. void zend_shared_alloc_clear_xlat_table(void)
  388. {
  389. zend_hash_clean(&ZCG(xlat_table));
  390. }
  391. void zend_shared_alloc_register_xlat_entry(const void *old, const void *new)
  392. {
  393. zend_ulong key = (zend_ulong)old;
  394. key = (key >> 3) | (key << ((sizeof(key) * 8) - 3)); /* key = _rotr(key, 3);*/
  395. zend_hash_index_add_new_ptr(&ZCG(xlat_table), key, (void*)new);
  396. }
  397. void *zend_shared_alloc_get_xlat_entry(const void *old)
  398. {
  399. void *retval;
  400. zend_ulong key = (zend_ulong)old;
  401. key = (key >> 3) | (key << ((sizeof(key) * 8) - 3)); /* key = _rotr(key, 3);*/
  402. if ((retval = zend_hash_index_find_ptr(&ZCG(xlat_table), key)) == NULL) {
  403. return NULL;
  404. }
  405. return retval;
  406. }
  407. size_t zend_shared_alloc_get_free_memory(void)
  408. {
  409. return ZSMMG(shared_free);
  410. }
  411. void zend_shared_alloc_save_state(void)
  412. {
  413. int i;
  414. for (i = 0; i < ZSMMG(shared_segments_count); i++) {
  415. ZSMMG(shared_memory_state).positions[i] = ZSMMG(shared_segments)[i]->pos;
  416. }
  417. ZSMMG(shared_memory_state).shared_free = ZSMMG(shared_free);
  418. }
  419. void zend_shared_alloc_restore_state(void)
  420. {
  421. int i;
  422. for (i = 0; i < ZSMMG(shared_segments_count); i++) {
  423. ZSMMG(shared_segments)[i]->pos = ZSMMG(shared_memory_state).positions[i];
  424. }
  425. ZSMMG(shared_free) = ZSMMG(shared_memory_state).shared_free;
  426. ZSMMG(memory_exhausted) = 0;
  427. ZSMMG(wasted_shared_memory) = 0;
  428. }
  429. const char *zend_accel_get_shared_model(void)
  430. {
  431. return g_shared_model;
  432. }
  433. void zend_accel_shared_protect(int mode)
  434. {
  435. #ifdef HAVE_MPROTECT
  436. int i;
  437. if (!smm_shared_globals) {
  438. return;
  439. }
  440. if (mode) {
  441. mode = PROT_READ;
  442. } else {
  443. mode = PROT_READ|PROT_WRITE;
  444. }
  445. for (i = 0; i < ZSMMG(shared_segments_count); i++) {
  446. mprotect(ZSMMG(shared_segments)[i]->p, ZSMMG(shared_segments)[i]->size, mode);
  447. }
  448. #endif
  449. }
  450. int zend_accel_in_shm(void *ptr)
  451. {
  452. int i;
  453. if (!smm_shared_globals) {
  454. return 0;
  455. }
  456. for (i = 0; i < ZSMMG(shared_segments_count); i++) {
  457. if ((char*)ptr >= (char*)ZSMMG(shared_segments)[i]->p &&
  458. (char*)ptr < (char*)ZSMMG(shared_segments)[i]->p + ZSMMG(shared_segments)[i]->size) {
  459. return 1;
  460. }
  461. }
  462. return 0;
  463. }