shared_alloc_shm.c 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  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 "zend_shared_alloc.h"
  22. #ifdef USE_SHM
  23. #if defined(__FreeBSD__)
  24. # include <machine/param.h>
  25. #endif
  26. #include <sys/types.h>
  27. #include <sys/shm.h>
  28. #include <sys/ipc.h>
  29. #include <dirent.h>
  30. #include <signal.h>
  31. #include <stdio.h>
  32. #include <stdlib.h>
  33. #include <unistd.h>
  34. #include <errno.h>
  35. #include <sys/stat.h>
  36. #include <fcntl.h>
  37. #ifndef MIN
  38. # define MIN(x, y) ((x) > (y)? (y) : (x))
  39. #endif
  40. #define SEG_ALLOC_SIZE_MAX 32*1024*1024
  41. #define SEG_ALLOC_SIZE_MIN 2*1024*1024
  42. typedef struct {
  43. zend_shared_segment common;
  44. int shm_id;
  45. } zend_shared_segment_shm;
  46. static int create_segments(size_t requested_size, zend_shared_segment_shm ***shared_segments_p, int *shared_segments_count, char **error_in)
  47. {
  48. int i;
  49. size_t allocate_size = 0, remaining_bytes = requested_size, seg_allocate_size;
  50. int first_segment_id = -1;
  51. key_t first_segment_key = -1;
  52. struct shmid_ds sds;
  53. int shmget_flags;
  54. zend_shared_segment_shm *shared_segments;
  55. seg_allocate_size = SEG_ALLOC_SIZE_MAX;
  56. /* determine segment size we _really_ need:
  57. * no more than to include requested_size
  58. */
  59. while (requested_size * 2 <= seg_allocate_size && seg_allocate_size > SEG_ALLOC_SIZE_MIN) {
  60. seg_allocate_size >>= 1;
  61. }
  62. shmget_flags = IPC_CREAT|SHM_R|SHM_W|IPC_EXCL;
  63. /* try allocating this much, if not - try shrinking */
  64. while (seg_allocate_size >= SEG_ALLOC_SIZE_MIN) {
  65. allocate_size = MIN(requested_size, seg_allocate_size);
  66. first_segment_id = shmget(first_segment_key, allocate_size, shmget_flags);
  67. if (first_segment_id != -1) {
  68. break;
  69. }
  70. seg_allocate_size >>= 1; /* shrink the allocated block */
  71. }
  72. if (first_segment_id == -1) {
  73. *error_in = "shmget";
  74. return ALLOC_FAILURE;
  75. }
  76. *shared_segments_count = ((requested_size - 1) / seg_allocate_size) + 1;
  77. *shared_segments_p = (zend_shared_segment_shm **) calloc(1, (*shared_segments_count) * sizeof(zend_shared_segment_shm) + sizeof(void *) * (*shared_segments_count));
  78. if (!*shared_segments_p) {
  79. *error_in = "calloc";
  80. return ALLOC_FAILURE;
  81. }
  82. shared_segments = (zend_shared_segment_shm *)((char *)(*shared_segments_p) + sizeof(void *) * (*shared_segments_count));
  83. for (i = 0; i < *shared_segments_count; i++) {
  84. (*shared_segments_p)[i] = shared_segments + i;
  85. }
  86. remaining_bytes = requested_size;
  87. for (i = 0; i < *shared_segments_count; i++) {
  88. allocate_size = MIN(remaining_bytes, seg_allocate_size);
  89. if (i != 0) {
  90. shared_segments[i].shm_id = shmget(IPC_PRIVATE, allocate_size, shmget_flags);
  91. } else {
  92. shared_segments[i].shm_id = first_segment_id;
  93. }
  94. if (shared_segments[i].shm_id == -1) {
  95. return ALLOC_FAILURE;
  96. }
  97. shared_segments[i].common.p = shmat(shared_segments[i].shm_id, NULL, 0);
  98. if (shared_segments[i].common.p == (void *)-1) {
  99. *error_in = "shmat";
  100. shmctl(shared_segments[i].shm_id, IPC_RMID, &sds);
  101. return ALLOC_FAILURE;
  102. }
  103. shmctl(shared_segments[i].shm_id, IPC_RMID, &sds);
  104. shared_segments[i].common.pos = 0;
  105. shared_segments[i].common.size = allocate_size;
  106. remaining_bytes -= allocate_size;
  107. }
  108. return ALLOC_SUCCESS;
  109. }
  110. static int detach_segment(zend_shared_segment_shm *shared_segment)
  111. {
  112. shmdt(shared_segment->common.p);
  113. return 0;
  114. }
  115. static size_t segment_type_size(void)
  116. {
  117. return sizeof(zend_shared_segment_shm);
  118. }
  119. zend_shared_memory_handlers zend_alloc_shm_handlers = {
  120. (create_segments_t)create_segments,
  121. (detach_segment_t)detach_segment,
  122. segment_type_size
  123. };
  124. #endif /* USE_SHM */