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