xref: /freebsd/contrib/llvm-project/compiler-rt/lib/scudo/standalone/options.h (revision 19261079b74319502c6ffa1249920079f0f69a72)
1 //===-- options.h -----------------------------------------------*- C++ -*-===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #ifndef SCUDO_OPTIONS_H_
10 #define SCUDO_OPTIONS_H_
11 
12 #include "atomic_helpers.h"
13 #include "common.h"
14 #include "memtag.h"
15 
16 namespace scudo {
17 
18 enum class OptionBit {
19   MayReturnNull,
20   FillContents0of2,
21   FillContents1of2,
22   DeallocTypeMismatch,
23   DeleteSizeMismatch,
24   TrackAllocationStacks,
25   UseOddEvenTags,
26   UseMemoryTagging,
27 };
28 
29 struct Options {
30   u32 Val;
31 
32   bool get(OptionBit Opt) const { return Val & (1U << static_cast<u32>(Opt)); }
33 
34   FillContentsMode getFillContentsMode() const {
35     return static_cast<FillContentsMode>(
36         (Val >> static_cast<u32>(OptionBit::FillContents0of2)) & 3);
37   }
38 };
39 
40 template <typename Config> bool useMemoryTagging(Options Options) {
41   return allocatorSupportsMemoryTagging<Config>() &&
42          Options.get(OptionBit::UseMemoryTagging);
43 }
44 
45 struct AtomicOptions {
46   atomic_u32 Val;
47 
48 public:
49   Options load() const { return Options{atomic_load_relaxed(&Val)}; }
50 
51   void clear(OptionBit Opt) {
52     atomic_fetch_and(&Val, ~(1U << static_cast<u32>(Opt)),
53                      memory_order_relaxed);
54   }
55 
56   void set(OptionBit Opt) {
57     atomic_fetch_or(&Val, 1U << static_cast<u32>(Opt), memory_order_relaxed);
58   }
59 
60   void setFillContentsMode(FillContentsMode FillContents) {
61     u32 Opts = atomic_load_relaxed(&Val), NewOpts;
62     do {
63       NewOpts = Opts;
64       NewOpts &= ~(3U << static_cast<u32>(OptionBit::FillContents0of2));
65       NewOpts |= static_cast<u32>(FillContents)
66                  << static_cast<u32>(OptionBit::FillContents0of2);
67     } while (!atomic_compare_exchange_strong(&Val, &Opts, NewOpts,
68                                              memory_order_relaxed));
69   }
70 };
71 
72 } // namespace scudo
73 
74 #endif // SCUDO_OPTIONS_H_
75