almalloc.cpp 1.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667
  1. #include "config.h"
  2. #include "almalloc.h"
  3. #include <cassert>
  4. #include <cstddef>
  5. #include <cstdlib>
  6. #include <cstring>
  7. #ifdef HAVE_MALLOC_H
  8. #include <malloc.h>
  9. #endif
  10. #define ALIGNED_ALLOC_AVAILABLE (__STDC_VERSION__ >= 201112L || __cplusplus >= 201703L)
  11. void *al_malloc(size_t alignment, size_t size)
  12. {
  13. assert((alignment & (alignment-1)) == 0);
  14. alignment = std::max(alignment, alignof(std::max_align_t));
  15. #if ALIGNED_ALLOC_AVAILABLE
  16. size = (size+(alignment-1))&~(alignment-1);
  17. return aligned_alloc(alignment, size);
  18. #elif defined(HAVE_POSIX_MEMALIGN)
  19. void *ret;
  20. if(posix_memalign(&ret, alignment, size) == 0)
  21. return ret;
  22. return nullptr;
  23. #elif defined(HAVE__ALIGNED_MALLOC)
  24. return _aligned_malloc(size, alignment);
  25. #else
  26. auto *ret = static_cast<char*>(malloc(size+alignment));
  27. if(ret != nullptr)
  28. {
  29. *(ret++) = 0x00;
  30. while((reinterpret_cast<uintptr_t>(ret)&(alignment-1)) != 0)
  31. *(ret++) = 0x55;
  32. }
  33. return ret;
  34. #endif
  35. }
  36. void *al_calloc(size_t alignment, size_t size)
  37. {
  38. void *ret = al_malloc(alignment, size);
  39. if(ret) memset(ret, 0, size);
  40. return ret;
  41. }
  42. void al_free(void *ptr) noexcept
  43. {
  44. #if ALIGNED_ALLOC_AVAILABLE || defined(HAVE_POSIX_MEMALIGN)
  45. free(ptr);
  46. #elif defined(HAVE__ALIGNED_MALLOC)
  47. _aligned_free(ptr);
  48. #else
  49. if(ptr != nullptr)
  50. {
  51. auto *finder = static_cast<char*>(ptr);
  52. do {
  53. --finder;
  54. } while(*finder == 0x55);
  55. free(finder);
  56. }
  57. #endif
  58. }