File: rtsan_context.cpp

package info (click to toggle)
llvm-toolchain-21 1%3A21.1.6-3
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 2,245,028 kB
  • sloc: cpp: 7,619,726; ansic: 1,434,018; asm: 1,058,748; python: 252,740; f90: 94,671; objc: 70,685; lisp: 42,813; pascal: 18,401; sh: 8,601; ml: 5,111; perl: 4,720; makefile: 3,675; awk: 3,523; javascript: 2,409; xml: 892; fortran: 770
file content (63 lines) | stat: -rw-r--r-- 2,084 bytes parent folder | download | duplicates (12)
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
57
58
59
60
61
62
63
//===--- rtsan_context.cpp - Realtime Sanitizer -----------------*- C++ -*-===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
//
//===----------------------------------------------------------------------===//

#include "rtsan/rtsan_context.h"
#include "rtsan/rtsan.h"

#include "sanitizer_common/sanitizer_allocator_internal.h"

#include <new>
#include <pthread.h>

using namespace __sanitizer;
using namespace __rtsan;

static pthread_key_t context_key;
static pthread_once_t key_once = PTHREAD_ONCE_INIT;

// InternalFree cannot be passed directly to pthread_key_create
// because it expects a signature with only one arg
static void InternalFreeWrapper(void *ptr) { __sanitizer::InternalFree(ptr); }

static __rtsan::Context &GetContextForThisThreadImpl() {
  auto MakeThreadLocalContextKey = []() {
    CHECK_EQ(pthread_key_create(&context_key, InternalFreeWrapper), 0);
  };

  pthread_once(&key_once, MakeThreadLocalContextKey);
  Context *current_thread_context =
      static_cast<Context *>(pthread_getspecific(context_key));
  if (current_thread_context == nullptr) {
    current_thread_context =
        static_cast<Context *>(InternalAlloc(sizeof(Context)));
    new (current_thread_context) Context();
    pthread_setspecific(context_key, current_thread_context);
  }

  return *current_thread_context;
}

__rtsan::Context::Context() = default;

void __rtsan::Context::RealtimePush() { realtime_depth_++; }

void __rtsan::Context::RealtimePop() { realtime_depth_--; }

void __rtsan::Context::BypassPush() { bypass_depth_++; }

void __rtsan::Context::BypassPop() { bypass_depth_--; }

bool __rtsan::Context::InRealtimeContext() const { return realtime_depth_ > 0; }

bool __rtsan::Context::IsBypassed() const { return bypass_depth_ > 0; }

Context &__rtsan::GetContextForThisThread() {
  return GetContextForThisThreadImpl();
}