forked from OSchip/llvm-project
204 lines
6.9 KiB
C++
204 lines
6.9 KiB
C++
//===-- esan_shadow.h -------------------------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file is a part of EfficiencySanitizer, a family of performance tuners.
|
|
//
|
|
// Shadow memory mappings for the esan run-time.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef ESAN_SHADOW_H
|
|
#define ESAN_SHADOW_H
|
|
|
|
#include <sanitizer_common/sanitizer_platform.h>
|
|
|
|
#if SANITIZER_WORDSIZE != 64
|
|
#error Only 64-bit is supported
|
|
#endif
|
|
|
|
namespace __esan {
|
|
|
|
#if SANITIZER_LINUX && defined(__x86_64__)
|
|
// Linux x86_64
|
|
//
|
|
// Application memory falls into these 5 regions (ignoring the corner case
|
|
// of PIE with a non-zero PT_LOAD base):
|
|
//
|
|
// [0x00000000'00000000, 0x00000100'00000000) non-PIE + heap
|
|
// [0x00005500'00000000, 0x00005700'00000000) PIE
|
|
// [0x00007e00'00000000, 0x00007fff'ff600000) libraries + stack, part 1
|
|
// [0x00007fff'ff601000, 0x00008000'00000000) libraries + stack, part 2
|
|
// [0xffffffff'ff600000, 0xffffffff'ff601000) vsyscall
|
|
//
|
|
// Although we can ignore the vsyscall for the most part as there are few data
|
|
// references there (other sanitizers ignore it), we enforce a gap inside the
|
|
// library region to distinguish the vsyscall's shadow, considering this gap to
|
|
// be an invalid app region.
|
|
// We disallow application memory outside of those 5 regions.
|
|
// Our regions assume that the stack rlimit is less than a terabyte (otherwise
|
|
// the Linux kernel's default mmap region drops below 0x7e00'), which we enforce
|
|
// at init time (we can support larger and unlimited sizes for shadow
|
|
// scaledowns, but it is difficult for 1:1 mappings).
|
|
//
|
|
// Our shadow memory is scaled from a 1:1 mapping and supports a scale
|
|
// specified at library initialization time that can be any power-of-2
|
|
// scaledown (1x, 2x, 4x, 8x, 16x, etc.).
|
|
//
|
|
// We model our shadow memory after Umbra, a library used by the Dr. Memory
|
|
// tool: https://github.com/DynamoRIO/drmemory/blob/master/umbra/umbra_x64.c.
|
|
// We use Umbra's scheme as it was designed to support different
|
|
// offsets, it supports two different shadow mappings (which we may want to
|
|
// use for future tools), and it ensures that the shadow of a shadow will
|
|
// not overlap either shadow memory or application memory.
|
|
//
|
|
// This formula translates from application memory to shadow memory:
|
|
//
|
|
// shadow(app) = ((app & 0x00000fff'ffffffff) + offset) >> scale
|
|
//
|
|
// Where the offset for 1:1 is 0x00001300'00000000. For other scales, the
|
|
// offset is shifted left by the scale, except for scales of 1 and 2 where
|
|
// it must be tweaked in order to pass the double-shadow test
|
|
// (see the "shadow(shadow)" comments below):
|
|
// scale == 0: 0x00001300'000000000
|
|
// scale == 1: 0x00002200'000000000
|
|
// scale == 2: 0x00004400'000000000
|
|
// scale >= 3: (0x00001300'000000000 << scale)
|
|
//
|
|
// Do not pass in the open-ended end value to the formula as it will fail.
|
|
//
|
|
// The resulting shadow memory regions for a 0 scaling are:
|
|
//
|
|
// [0x00001300'00000000, 0x00001400'00000000)
|
|
// [0x00001800'00000000, 0x00001a00'00000000)
|
|
// [0x00002100'00000000, 0x000022ff'ff600000)
|
|
// [0x000022ff'ff601000, 0x00002300'00000000)
|
|
// [0x000022ff'ff600000, 0x000022ff'ff601000]
|
|
//
|
|
// We also want to ensure that a wild access by the application into the shadow
|
|
// regions will not corrupt our own shadow memory. shadow(shadow) ends up
|
|
// disjoint from shadow(app):
|
|
//
|
|
// [0x00001600'00000000, 0x00001700'00000000)
|
|
// [0x00001b00'00000000, 0x00001d00'00000000)
|
|
// [0x00001400'00000000, 0x000015ff'ff600000]
|
|
// [0x000015ff'ff601000, 0x00001600'00000000]
|
|
// [0x000015ff'ff600000, 0x000015ff'ff601000]
|
|
|
|
struct ApplicationRegion {
|
|
uptr Start;
|
|
uptr End;
|
|
bool ShadowMergedWithPrev;
|
|
};
|
|
|
|
static const struct ApplicationRegion AppRegions[] = {
|
|
{0x0000000000000000ull, 0x0000010000000000u, false},
|
|
{0x0000550000000000u, 0x0000570000000000u, false},
|
|
// We make one shadow mapping to hold the shadow regions for all 3 of these
|
|
// app regions, as the mappings interleave, and the gap between the 3rd and
|
|
// 4th scales down below a page.
|
|
{0x00007e0000000000u, 0x00007fffff600000u, false},
|
|
{0x00007fffff601000u, 0x0000800000000000u, true},
|
|
{0xffffffffff600000u, 0xffffffffff601000u, true},
|
|
};
|
|
static const u32 NumAppRegions = sizeof(AppRegions)/sizeof(AppRegions[0]);
|
|
|
|
// See the comment above: we do not currently support a stack size rlimit
|
|
// equal to or larger than 1TB.
|
|
static const uptr MaxStackSize = (1ULL << 40) - 4096;
|
|
|
|
class ShadowMapping {
|
|
public:
|
|
static const uptr Mask = 0x00000fffffffffffu;
|
|
// The scale and offset vary by tool.
|
|
uptr Scale;
|
|
uptr Offset;
|
|
void initialize(uptr ShadowScale) {
|
|
static const uptr OffsetArray[3] = {
|
|
0x0000130000000000u,
|
|
0x0000220000000000u,
|
|
0x0000440000000000u,
|
|
};
|
|
Scale = ShadowScale;
|
|
if (Scale <= 2)
|
|
Offset = OffsetArray[Scale];
|
|
else
|
|
Offset = OffsetArray[0] << Scale;
|
|
}
|
|
};
|
|
extern ShadowMapping Mapping;
|
|
#else
|
|
// We'll want to use templatized functions over the ShadowMapping once
|
|
// we support more platforms.
|
|
#error Platform not supported
|
|
#endif
|
|
|
|
static inline bool getAppRegion(u32 i, uptr *Start, uptr *End) {
|
|
if (i >= NumAppRegions)
|
|
return false;
|
|
*Start = AppRegions[i].Start;
|
|
*End = AppRegions[i].End;
|
|
return true;
|
|
}
|
|
|
|
ALWAYS_INLINE
|
|
bool isAppMem(uptr Mem) {
|
|
for (u32 i = 0; i < NumAppRegions; ++i) {
|
|
if (Mem >= AppRegions[i].Start && Mem < AppRegions[i].End)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
ALWAYS_INLINE
|
|
uptr appToShadow(uptr App) {
|
|
return (((App & ShadowMapping::Mask) + Mapping.Offset) >> Mapping.Scale);
|
|
}
|
|
|
|
static inline bool getShadowRegion(u32 i, uptr *Start, uptr *End) {
|
|
if (i >= NumAppRegions)
|
|
return false;
|
|
u32 UnmergedShadowCount = 0;
|
|
u32 AppIdx;
|
|
for (AppIdx = 0; AppIdx < NumAppRegions; ++AppIdx) {
|
|
if (!AppRegions[AppIdx].ShadowMergedWithPrev) {
|
|
if (UnmergedShadowCount == i)
|
|
break;
|
|
UnmergedShadowCount++;
|
|
}
|
|
}
|
|
if (AppIdx >= NumAppRegions || UnmergedShadowCount != i)
|
|
return false;
|
|
*Start = appToShadow(AppRegions[AppIdx].Start);
|
|
// The formula fails for the end itself.
|
|
*End = appToShadow(AppRegions[AppIdx].End - 1) + 1;
|
|
// Merge with adjacent shadow regions:
|
|
for (++AppIdx; AppIdx < NumAppRegions; ++AppIdx) {
|
|
if (!AppRegions[AppIdx].ShadowMergedWithPrev)
|
|
break;
|
|
*Start = Min(*Start, appToShadow(AppRegions[AppIdx].Start));
|
|
*End = Max(*End, appToShadow(AppRegions[AppIdx].End - 1) + 1);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
ALWAYS_INLINE
|
|
bool isShadowMem(uptr Mem) {
|
|
// We assume this is not used on any critical performance path and so there's
|
|
// no need to hardcode the mapping results.
|
|
for (uptr i = 0; i < NumAppRegions; ++i) {
|
|
if (Mem >= appToShadow(AppRegions[i].Start) &&
|
|
Mem < appToShadow(AppRegions[i].End - 1) + 1)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
} // namespace __esan
|
|
|
|
#endif /* ESAN_SHADOW_H */
|