mozalloc_oom.cpp 1.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354
  1. /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
  2. * vim: sw=4 ts=4 et :
  3. */
  4. /* This Source Code Form is subject to the terms of the Mozilla Public
  5. * License, v. 2.0. If a copy of the MPL was not distributed with this
  6. * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
  7. #include "mozilla/mozalloc_abort.h"
  8. #include "mozilla/mozalloc_oom.h"
  9. #include "mozilla/Assertions.h"
  10. static mozalloc_oom_abort_handler gAbortHandler;
  11. #define OOM_MSG_LEADER "out of memory: 0x"
  12. #define OOM_MSG_DIGITS "0000000000000000" // large enough for 2^64
  13. #define OOM_MSG_TRAILER " bytes requested"
  14. #define OOM_MSG_FIRST_DIGIT_OFFSET sizeof(OOM_MSG_LEADER) - 1
  15. #define OOM_MSG_LAST_DIGIT_OFFSET sizeof(OOM_MSG_LEADER) + \
  16. sizeof(OOM_MSG_DIGITS) - 3
  17. static const char *hex = "0123456789ABCDEF";
  18. void
  19. mozalloc_handle_oom(size_t size)
  20. {
  21. char oomMsg[] = OOM_MSG_LEADER OOM_MSG_DIGITS OOM_MSG_TRAILER;
  22. size_t i;
  23. // NB: this is handle_oom() stage 1, which simply aborts on OOM.
  24. // we might proceed to a stage 2 in which an attempt is made to
  25. // reclaim memory
  26. if (gAbortHandler)
  27. gAbortHandler(size);
  28. static_assert(OOM_MSG_FIRST_DIGIT_OFFSET > 0,
  29. "Loop below will never terminate (i can't go below 0)");
  30. // Insert size into the diagnostic message using only primitive operations
  31. for (i = OOM_MSG_LAST_DIGIT_OFFSET;
  32. size && i >= OOM_MSG_FIRST_DIGIT_OFFSET; i--) {
  33. oomMsg[i] = hex[size % 16];
  34. size /= 16;
  35. }
  36. mozalloc_abort(oomMsg);
  37. }
  38. void
  39. mozalloc_set_oom_abort_handler(mozalloc_oom_abort_handler handler)
  40. {
  41. gAbortHandler = handler;
  42. }