annotate libsanitizer/sanitizer_common/sanitizer_allocator_bytemap.h @ 122:fb3d53c41846

do not expand code segment
author mir3636
date Thu, 22 Mar 2018 17:37:58 +0900
parents 04ced10e8804
children 1830386684a0
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
111
kono
parents:
diff changeset
1 //===-- sanitizer_allocator_bytemap.h ---------------------------*- C++ -*-===//
kono
parents:
diff changeset
2 //
kono
parents:
diff changeset
3 // This file is distributed under the University of Illinois Open Source
kono
parents:
diff changeset
4 // License. See LICENSE.TXT for details.
kono
parents:
diff changeset
5 //
kono
parents:
diff changeset
6 //===----------------------------------------------------------------------===//
kono
parents:
diff changeset
7 //
kono
parents:
diff changeset
8 // Part of the Sanitizer Allocator.
kono
parents:
diff changeset
9 //
kono
parents:
diff changeset
10 //===----------------------------------------------------------------------===//
kono
parents:
diff changeset
11 #ifndef SANITIZER_ALLOCATOR_H
kono
parents:
diff changeset
12 #error This file must be included inside sanitizer_allocator.h
kono
parents:
diff changeset
13 #endif
kono
parents:
diff changeset
14
kono
parents:
diff changeset
15 // Maps integers in rage [0, kSize) to u8 values.
kono
parents:
diff changeset
16 template<u64 kSize>
kono
parents:
diff changeset
17 class FlatByteMap {
kono
parents:
diff changeset
18 public:
kono
parents:
diff changeset
19 void TestOnlyInit() {
kono
parents:
diff changeset
20 internal_memset(map_, 0, sizeof(map_));
kono
parents:
diff changeset
21 }
kono
parents:
diff changeset
22
kono
parents:
diff changeset
23 void set(uptr idx, u8 val) {
kono
parents:
diff changeset
24 CHECK_LT(idx, kSize);
kono
parents:
diff changeset
25 CHECK_EQ(0U, map_[idx]);
kono
parents:
diff changeset
26 map_[idx] = val;
kono
parents:
diff changeset
27 }
kono
parents:
diff changeset
28 u8 operator[] (uptr idx) {
kono
parents:
diff changeset
29 CHECK_LT(idx, kSize);
kono
parents:
diff changeset
30 // FIXME: CHECK may be too expensive here.
kono
parents:
diff changeset
31 return map_[idx];
kono
parents:
diff changeset
32 }
kono
parents:
diff changeset
33 private:
kono
parents:
diff changeset
34 u8 map_[kSize];
kono
parents:
diff changeset
35 };
kono
parents:
diff changeset
36
kono
parents:
diff changeset
37 // TwoLevelByteMap maps integers in range [0, kSize1*kSize2) to u8 values.
kono
parents:
diff changeset
38 // It is implemented as a two-dimensional array: array of kSize1 pointers
kono
parents:
diff changeset
39 // to kSize2-byte arrays. The secondary arrays are mmaped on demand.
kono
parents:
diff changeset
40 // Each value is initially zero and can be set to something else only once.
kono
parents:
diff changeset
41 // Setting and getting values from multiple threads is safe w/o extra locking.
kono
parents:
diff changeset
42 template <u64 kSize1, u64 kSize2, class MapUnmapCallback = NoOpMapUnmapCallback>
kono
parents:
diff changeset
43 class TwoLevelByteMap {
kono
parents:
diff changeset
44 public:
kono
parents:
diff changeset
45 void TestOnlyInit() {
kono
parents:
diff changeset
46 internal_memset(map1_, 0, sizeof(map1_));
kono
parents:
diff changeset
47 mu_.Init();
kono
parents:
diff changeset
48 }
kono
parents:
diff changeset
49
kono
parents:
diff changeset
50 void TestOnlyUnmap() {
kono
parents:
diff changeset
51 for (uptr i = 0; i < kSize1; i++) {
kono
parents:
diff changeset
52 u8 *p = Get(i);
kono
parents:
diff changeset
53 if (!p) continue;
kono
parents:
diff changeset
54 MapUnmapCallback().OnUnmap(reinterpret_cast<uptr>(p), kSize2);
kono
parents:
diff changeset
55 UnmapOrDie(p, kSize2);
kono
parents:
diff changeset
56 }
kono
parents:
diff changeset
57 }
kono
parents:
diff changeset
58
kono
parents:
diff changeset
59 uptr size() const { return kSize1 * kSize2; }
kono
parents:
diff changeset
60 uptr size1() const { return kSize1; }
kono
parents:
diff changeset
61 uptr size2() const { return kSize2; }
kono
parents:
diff changeset
62
kono
parents:
diff changeset
63 void set(uptr idx, u8 val) {
kono
parents:
diff changeset
64 CHECK_LT(idx, kSize1 * kSize2);
kono
parents:
diff changeset
65 u8 *map2 = GetOrCreate(idx / kSize2);
kono
parents:
diff changeset
66 CHECK_EQ(0U, map2[idx % kSize2]);
kono
parents:
diff changeset
67 map2[idx % kSize2] = val;
kono
parents:
diff changeset
68 }
kono
parents:
diff changeset
69
kono
parents:
diff changeset
70 u8 operator[] (uptr idx) const {
kono
parents:
diff changeset
71 CHECK_LT(idx, kSize1 * kSize2);
kono
parents:
diff changeset
72 u8 *map2 = Get(idx / kSize2);
kono
parents:
diff changeset
73 if (!map2) return 0;
kono
parents:
diff changeset
74 return map2[idx % kSize2];
kono
parents:
diff changeset
75 }
kono
parents:
diff changeset
76
kono
parents:
diff changeset
77 private:
kono
parents:
diff changeset
78 u8 *Get(uptr idx) const {
kono
parents:
diff changeset
79 CHECK_LT(idx, kSize1);
kono
parents:
diff changeset
80 return reinterpret_cast<u8 *>(
kono
parents:
diff changeset
81 atomic_load(&map1_[idx], memory_order_acquire));
kono
parents:
diff changeset
82 }
kono
parents:
diff changeset
83
kono
parents:
diff changeset
84 u8 *GetOrCreate(uptr idx) {
kono
parents:
diff changeset
85 u8 *res = Get(idx);
kono
parents:
diff changeset
86 if (!res) {
kono
parents:
diff changeset
87 SpinMutexLock l(&mu_);
kono
parents:
diff changeset
88 if (!(res = Get(idx))) {
kono
parents:
diff changeset
89 res = (u8*)MmapOrDie(kSize2, "TwoLevelByteMap");
kono
parents:
diff changeset
90 MapUnmapCallback().OnMap(reinterpret_cast<uptr>(res), kSize2);
kono
parents:
diff changeset
91 atomic_store(&map1_[idx], reinterpret_cast<uptr>(res),
kono
parents:
diff changeset
92 memory_order_release);
kono
parents:
diff changeset
93 }
kono
parents:
diff changeset
94 }
kono
parents:
diff changeset
95 return res;
kono
parents:
diff changeset
96 }
kono
parents:
diff changeset
97
kono
parents:
diff changeset
98 atomic_uintptr_t map1_[kSize1];
kono
parents:
diff changeset
99 StaticSpinMutex mu_;
kono
parents:
diff changeset
100 };