memzero.c 1.9 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374
  1. #ifndef __STDC_WANT_LIB_EXT1__
  2. #define __STDC_WANT_LIB_EXT1__ 1 // C11's bounds-checking interface.
  3. #endif
  4. #include <string.h>
  5. #ifdef _WIN32
  6. #include <windows.h>
  7. #endif
  8. #ifdef __unix__
  9. #include <strings.h>
  10. #include <sys/param.h>
  11. #endif
  12. // C11's bounds-checking interface.
  13. #if defined(__STDC_LIB_EXT1__)
  14. #define HAVE_MEMSET_S 1
  15. #endif
  16. // GNU C Library version 2.25 or later.
  17. #if defined(__GLIBC__) && (__GLIBC__ > 2 || (__GLIBC__ == 2 && __GLIBC_MINOR__ >= 25))
  18. #define HAVE_EXPLICIT_BZERO 1
  19. #endif
  20. // Newlib
  21. #if defined(__NEWLIB__)
  22. #define HAVE_EXPLICIT_BZERO 1
  23. #endif
  24. // FreeBSD version 11.0 or later.
  25. #if defined(__FreeBSD__) && __FreeBSD_version >= 1100037
  26. #define HAVE_EXPLICIT_BZERO 1
  27. #endif
  28. // OpenBSD version 5.5 or later.
  29. #if defined(__OpenBSD__) && OpenBSD >= 201405
  30. #define HAVE_EXPLICIT_BZERO 1
  31. #endif
  32. // NetBSD version 7.2 or later.
  33. #if defined(__NetBSD__) && __NetBSD_Version__ >= 702000000
  34. #define HAVE_EXPLICIT_MEMSET 1
  35. #endif
  36. // Adapted from
  37. // https://github.com/jedisct1/libsodium/blob/1647f0d53ae0e370378a9195477e3df0a792408f/src/libsodium/sodium/utils.c#L102-L130
  38. void memzero(void* const pnt, const size_t len) {
  39. #ifdef _WIN32
  40. SecureZeroMemory(pnt, len);
  41. #elif defined(HAVE_MEMSET_S)
  42. memset_s(pnt, (rsize_t)len, 0, (rsize_t)len);
  43. // #elif defined(HAVE_EXPLICIT_BZERO)
  44. // explicit_bzero(pnt, len);
  45. #elif defined(HAVE_EXPLICIT_MEMSET)
  46. explicit_memset(pnt, 0, len);
  47. #else
  48. volatile unsigned char* volatile pnt_ = (volatile unsigned char* volatile)pnt;
  49. size_t i = (size_t)0U;
  50. while(i < len) {
  51. pnt_[i++] = 0U;
  52. }
  53. #endif
  54. // explicitly mark the memory as overwritten for the Clang MemorySanitizer
  55. // this is only included at compile time if MemorySanitizer is enabled and
  56. // should not come with any downsides during regular builds
  57. #if defined(__has_feature)
  58. #if __has_feature(memory_sanitizer)
  59. memset(pnt, 0, len);
  60. #endif
  61. #endif
  62. }