shared_alloc_mmap.c 3.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  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 "zend_shared_alloc.h"
  22. #ifdef USE_MMAP
  23. #include <sys/types.h>
  24. #include <sys/stat.h>
  25. #include <stdio.h>
  26. #include <stdlib.h>
  27. #include <sys/mman.h>
  28. #if defined(MAP_ANON) && !defined(MAP_ANONYMOUS)
  29. # define MAP_ANONYMOUS MAP_ANON
  30. #endif
  31. static int create_segments(size_t requested_size, zend_shared_segment ***shared_segments_p, int *shared_segments_count, char **error_in)
  32. {
  33. zend_shared_segment *shared_segment;
  34. *shared_segments_count = 1;
  35. *shared_segments_p = (zend_shared_segment **) calloc(1, sizeof(zend_shared_segment) + sizeof(void *));
  36. if (!*shared_segments_p) {
  37. *error_in = "calloc";
  38. return ALLOC_FAILURE;
  39. }
  40. shared_segment = (zend_shared_segment *)((char *)(*shared_segments_p) + sizeof(void *));
  41. (*shared_segments_p)[0] = shared_segment;
  42. #ifdef MAP_HUGETLB
  43. /* Try to allocate huge pages first to reduce dTLB misses.
  44. * OS has to be configured properly
  45. * (e.g. https://wiki.debian.org/Hugepages#Enabling_HugeTlbPage)
  46. * You may verify huge page usage with the following command:
  47. * `grep "Huge" /proc/meminfo`
  48. */
  49. shared_segment->p = mmap(0, requested_size, PROT_READ | PROT_WRITE, MAP_SHARED|MAP_ANONYMOUS|MAP_HUGETLB, -1, 0);
  50. if (shared_segment->p != MAP_FAILED) {
  51. shared_segment->pos = 0;
  52. shared_segment->size = requested_size;
  53. return ALLOC_SUCCESS;
  54. }
  55. #endif
  56. shared_segment->p = mmap(0, requested_size, PROT_READ | PROT_WRITE, MAP_SHARED|MAP_ANONYMOUS, -1, 0);
  57. if (shared_segment->p == MAP_FAILED) {
  58. *error_in = "mmap";
  59. return ALLOC_FAILURE;
  60. }
  61. shared_segment->pos = 0;
  62. shared_segment->size = requested_size;
  63. return ALLOC_SUCCESS;
  64. }
  65. static int detach_segment(zend_shared_segment *shared_segment)
  66. {
  67. munmap(shared_segment->p, shared_segment->size);
  68. return 0;
  69. }
  70. static size_t segment_type_size(void)
  71. {
  72. return sizeof(zend_shared_segment);
  73. }
  74. zend_shared_memory_handlers zend_alloc_mmap_handlers = {
  75. create_segments,
  76. detach_segment,
  77. segment_type_size
  78. };
  79. #endif /* USE_MMAP */