File: mozalloc_oom.cpp

package info (click to toggle)
mozjs78 78.15.0-7
  • links: PTS, VCS
  • area: main
  • in suites: bookworm
  • size: 739,892 kB
  • sloc: javascript: 1,344,214; cpp: 1,215,708; python: 526,544; ansic: 433,835; xml: 118,736; sh: 26,176; asm: 16,664; makefile: 11,537; yacc: 4,486; perl: 2,564; ada: 1,681; lex: 1,414; pascal: 1,139; cs: 879; exp: 499; java: 164; ruby: 68; sql: 45; csh: 35; sed: 18; lisp: 2
file content (56 lines) | stat: -rw-r--r-- 2,070 bytes parent folder | download | duplicates (8)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
 * vim: sw=2 ts=4 et :
 */
/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

#include "mozilla/mozalloc_abort.h"
#include "mozilla/mozalloc_oom.h"
#include "mozilla/Assertions.h"

static mozalloc_oom_abort_handler gAbortHandler;

#define OOM_MSG_LEADER "out of memory: 0x"
#define OOM_MSG_DIGITS "0000000000000000"  // large enough for 2^64
#define OOM_MSG_TRAILER " bytes requested"
#define OOM_MSG_FIRST_DIGIT_OFFSET sizeof(OOM_MSG_LEADER) - 1
#define OOM_MSG_LAST_DIGIT_OFFSET \
  sizeof(OOM_MSG_LEADER) + sizeof(OOM_MSG_DIGITS) - 3

static const char* hex = "0123456789ABCDEF";

void mozalloc_handle_oom(size_t size) {
  char oomMsg[] = OOM_MSG_LEADER OOM_MSG_DIGITS OOM_MSG_TRAILER;
  size_t i;

  // NB: this is handle_oom() stage 1, which simply aborts on OOM.
  // we might proceed to a stage 2 in which an attempt is made to
  // reclaim memory
  // Warning: when stage 2 is done by, for example, notifying
  // "memory-pressure" synchronously, please audit all
  // nsExpirationTrackers and ensure that the actions they take
  // on memory-pressure notifications (via NotifyExpired) are safe.
  // Note that Document::SelectorCache::NotifyExpired is _known_
  // to not be safe: it will delete the selector it's caching,
  // which might be in use at the time under querySelector or
  // querySelectorAll.

  if (gAbortHandler) gAbortHandler(size);

  static_assert(OOM_MSG_FIRST_DIGIT_OFFSET > 0,
                "Loop below will never terminate (i can't go below 0)");

  // Insert size into the diagnostic message using only primitive operations
  for (i = OOM_MSG_LAST_DIGIT_OFFSET; size && i >= OOM_MSG_FIRST_DIGIT_OFFSET;
       i--) {
    oomMsg[i] = hex[size % 16];
    size /= 16;
  }

  mozalloc_abort(oomMsg);
}

void mozalloc_set_oom_abort_handler(mozalloc_oom_abort_handler handler) {
  gAbortHandler = handler;
}