xref: /freebsd-src/contrib/llvm-project/compiler-rt/lib/memprof/memprof_mapping.h (revision e8d8bef961a50d4dc22501cde4fb9fb0be1b2532)
1*e8d8bef9SDimitry Andric //===-- memprof_mapping.h --------------------------------------*- C++ -*-===//
2*e8d8bef9SDimitry Andric //
3*e8d8bef9SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*e8d8bef9SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5*e8d8bef9SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*e8d8bef9SDimitry Andric //
7*e8d8bef9SDimitry Andric //===----------------------------------------------------------------------===//
8*e8d8bef9SDimitry Andric //
9*e8d8bef9SDimitry Andric // This file is a part of MemProfiler, a memory profiler.
10*e8d8bef9SDimitry Andric //
11*e8d8bef9SDimitry Andric // Defines MemProf memory mapping.
12*e8d8bef9SDimitry Andric //===----------------------------------------------------------------------===//
13*e8d8bef9SDimitry Andric #ifndef MEMPROF_MAPPING_H
14*e8d8bef9SDimitry Andric #define MEMPROF_MAPPING_H
15*e8d8bef9SDimitry Andric 
16*e8d8bef9SDimitry Andric #include "memprof_internal.h"
17*e8d8bef9SDimitry Andric 
18*e8d8bef9SDimitry Andric static const u64 kDefaultShadowScale = 3;
19*e8d8bef9SDimitry Andric #define SHADOW_SCALE kDefaultShadowScale
20*e8d8bef9SDimitry Andric 
21*e8d8bef9SDimitry Andric #define SHADOW_OFFSET __memprof_shadow_memory_dynamic_address
22*e8d8bef9SDimitry Andric 
23*e8d8bef9SDimitry Andric #define SHADOW_GRANULARITY (1ULL << SHADOW_SCALE)
24*e8d8bef9SDimitry Andric #define MEMPROF_ALIGNMENT 32
25*e8d8bef9SDimitry Andric 
26*e8d8bef9SDimitry Andric namespace __memprof {
27*e8d8bef9SDimitry Andric 
28*e8d8bef9SDimitry Andric extern uptr kHighMemEnd; // Initialized in __memprof_init.
29*e8d8bef9SDimitry Andric 
30*e8d8bef9SDimitry Andric } // namespace __memprof
31*e8d8bef9SDimitry Andric 
32*e8d8bef9SDimitry Andric #define SHADOW_ENTRY_SIZE 8
33*e8d8bef9SDimitry Andric 
34*e8d8bef9SDimitry Andric // Size of memory block mapped to a single shadow location
35*e8d8bef9SDimitry Andric #define MEM_GRANULARITY 64ULL
36*e8d8bef9SDimitry Andric 
37*e8d8bef9SDimitry Andric #define SHADOW_MASK ~(MEM_GRANULARITY - 1)
38*e8d8bef9SDimitry Andric 
39*e8d8bef9SDimitry Andric #define MEM_TO_SHADOW(mem)                                                     \
40*e8d8bef9SDimitry Andric   ((((mem) & SHADOW_MASK) >> SHADOW_SCALE) + (SHADOW_OFFSET))
41*e8d8bef9SDimitry Andric 
42*e8d8bef9SDimitry Andric #define kLowMemBeg 0
43*e8d8bef9SDimitry Andric #define kLowMemEnd (SHADOW_OFFSET ? SHADOW_OFFSET - 1 : 0)
44*e8d8bef9SDimitry Andric 
45*e8d8bef9SDimitry Andric #define kLowShadowBeg SHADOW_OFFSET
46*e8d8bef9SDimitry Andric #define kLowShadowEnd (MEM_TO_SHADOW(kLowMemEnd) + SHADOW_ENTRY_SIZE - 1)
47*e8d8bef9SDimitry Andric 
48*e8d8bef9SDimitry Andric #define kHighMemBeg (MEM_TO_SHADOW(kHighMemEnd) + 1 + SHADOW_ENTRY_SIZE - 1)
49*e8d8bef9SDimitry Andric 
50*e8d8bef9SDimitry Andric #define kHighShadowBeg MEM_TO_SHADOW(kHighMemBeg)
51*e8d8bef9SDimitry Andric #define kHighShadowEnd (MEM_TO_SHADOW(kHighMemEnd) + SHADOW_ENTRY_SIZE - 1)
52*e8d8bef9SDimitry Andric 
53*e8d8bef9SDimitry Andric // With the zero shadow base we can not actually map pages starting from 0.
54*e8d8bef9SDimitry Andric // This constant is somewhat arbitrary.
55*e8d8bef9SDimitry Andric #define kZeroBaseShadowStart 0
56*e8d8bef9SDimitry Andric #define kZeroBaseMaxShadowStart (1 << 18)
57*e8d8bef9SDimitry Andric 
58*e8d8bef9SDimitry Andric #define kShadowGapBeg (kLowShadowEnd ? kLowShadowEnd + 1 : kZeroBaseShadowStart)
59*e8d8bef9SDimitry Andric #define kShadowGapEnd (kHighShadowBeg - 1)
60*e8d8bef9SDimitry Andric 
61*e8d8bef9SDimitry Andric namespace __memprof {
62*e8d8bef9SDimitry Andric 
63*e8d8bef9SDimitry Andric inline uptr MemToShadowSize(uptr size) { return size >> SHADOW_SCALE; }
64*e8d8bef9SDimitry Andric inline bool AddrIsInLowMem(uptr a) { return a <= kLowMemEnd; }
65*e8d8bef9SDimitry Andric 
66*e8d8bef9SDimitry Andric inline bool AddrIsInLowShadow(uptr a) {
67*e8d8bef9SDimitry Andric   return a >= kLowShadowBeg && a <= kLowShadowEnd;
68*e8d8bef9SDimitry Andric }
69*e8d8bef9SDimitry Andric 
70*e8d8bef9SDimitry Andric inline bool AddrIsInHighMem(uptr a) {
71*e8d8bef9SDimitry Andric   return kHighMemBeg && a >= kHighMemBeg && a <= kHighMemEnd;
72*e8d8bef9SDimitry Andric }
73*e8d8bef9SDimitry Andric 
74*e8d8bef9SDimitry Andric inline bool AddrIsInHighShadow(uptr a) {
75*e8d8bef9SDimitry Andric   return kHighMemBeg && a >= kHighShadowBeg && a <= kHighShadowEnd;
76*e8d8bef9SDimitry Andric }
77*e8d8bef9SDimitry Andric 
78*e8d8bef9SDimitry Andric inline bool AddrIsInShadowGap(uptr a) {
79*e8d8bef9SDimitry Andric   // In zero-based shadow mode we treat addresses near zero as addresses
80*e8d8bef9SDimitry Andric   // in shadow gap as well.
81*e8d8bef9SDimitry Andric   if (SHADOW_OFFSET == 0)
82*e8d8bef9SDimitry Andric     return a <= kShadowGapEnd;
83*e8d8bef9SDimitry Andric   return a >= kShadowGapBeg && a <= kShadowGapEnd;
84*e8d8bef9SDimitry Andric }
85*e8d8bef9SDimitry Andric 
86*e8d8bef9SDimitry Andric inline bool AddrIsInMem(uptr a) {
87*e8d8bef9SDimitry Andric   return AddrIsInLowMem(a) || AddrIsInHighMem(a) ||
88*e8d8bef9SDimitry Andric          (flags()->protect_shadow_gap == 0 && AddrIsInShadowGap(a));
89*e8d8bef9SDimitry Andric }
90*e8d8bef9SDimitry Andric 
91*e8d8bef9SDimitry Andric inline uptr MemToShadow(uptr p) {
92*e8d8bef9SDimitry Andric   CHECK(AddrIsInMem(p));
93*e8d8bef9SDimitry Andric   return MEM_TO_SHADOW(p);
94*e8d8bef9SDimitry Andric }
95*e8d8bef9SDimitry Andric 
96*e8d8bef9SDimitry Andric inline bool AddrIsInShadow(uptr a) {
97*e8d8bef9SDimitry Andric   return AddrIsInLowShadow(a) || AddrIsInHighShadow(a);
98*e8d8bef9SDimitry Andric }
99*e8d8bef9SDimitry Andric 
100*e8d8bef9SDimitry Andric inline bool AddrIsAlignedByGranularity(uptr a) {
101*e8d8bef9SDimitry Andric   return (a & (SHADOW_GRANULARITY - 1)) == 0;
102*e8d8bef9SDimitry Andric }
103*e8d8bef9SDimitry Andric 
104*e8d8bef9SDimitry Andric inline void RecordAccess(uptr a) {
105*e8d8bef9SDimitry Andric   // If we use a different shadow size then the type below needs adjustment.
106*e8d8bef9SDimitry Andric   CHECK_EQ(SHADOW_ENTRY_SIZE, 8);
107*e8d8bef9SDimitry Andric   u64 *shadow_address = (u64 *)MEM_TO_SHADOW(a);
108*e8d8bef9SDimitry Andric   (*shadow_address)++;
109*e8d8bef9SDimitry Andric }
110*e8d8bef9SDimitry Andric 
111*e8d8bef9SDimitry Andric } // namespace __memprof
112*e8d8bef9SDimitry Andric 
113*e8d8bef9SDimitry Andric #endif // MEMPROF_MAPPING_H
114