LLVM 23.0.0git
AddressSanitizer.cpp
Go to the documentation of this file.
1//===- AddressSanitizer.cpp - memory error detector -----------------------===//
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// This file is a part of AddressSanitizer, an address basic correctness
10// checker.
11// Details of the algorithm:
12// https://github.com/google/sanitizers/wiki/AddressSanitizerAlgorithm
13//
14// FIXME: This sanitizer does not yet handle scalable vectors
15//
16//===----------------------------------------------------------------------===//
17
19#include "llvm/ADT/ArrayRef.h"
20#include "llvm/ADT/DenseMap.h"
23#include "llvm/ADT/SmallSet.h"
25#include "llvm/ADT/Statistic.h"
27#include "llvm/ADT/StringRef.h"
28#include "llvm/ADT/Twine.h"
37#include "llvm/IR/Argument.h"
38#include "llvm/IR/Attributes.h"
39#include "llvm/IR/BasicBlock.h"
40#include "llvm/IR/Comdat.h"
41#include "llvm/IR/Constant.h"
42#include "llvm/IR/Constants.h"
43#include "llvm/IR/DIBuilder.h"
44#include "llvm/IR/DataLayout.h"
46#include "llvm/IR/DebugLoc.h"
49#include "llvm/IR/Function.h"
50#include "llvm/IR/GlobalAlias.h"
51#include "llvm/IR/GlobalValue.h"
53#include "llvm/IR/IRBuilder.h"
54#include "llvm/IR/InlineAsm.h"
55#include "llvm/IR/InstVisitor.h"
56#include "llvm/IR/InstrTypes.h"
57#include "llvm/IR/Instruction.h"
60#include "llvm/IR/Intrinsics.h"
61#include "llvm/IR/LLVMContext.h"
62#include "llvm/IR/MDBuilder.h"
63#include "llvm/IR/Metadata.h"
64#include "llvm/IR/Module.h"
65#include "llvm/IR/Type.h"
66#include "llvm/IR/Use.h"
67#include "llvm/IR/Value.h"
71#include "llvm/Support/Debug.h"
74#include "llvm/Support/ModRef.h"
85#include <algorithm>
86#include <cassert>
87#include <cstddef>
88#include <cstdint>
89#include <iomanip>
90#include <limits>
91#include <sstream>
92#include <string>
93#include <tuple>
94#include <utility>
95
96using namespace llvm;
97
98#define DEBUG_TYPE "asan"
99
101static const uint64_t kDefaultShadowOffset32 = 1ULL << 29;
102static const uint64_t kDefaultShadowOffset64 = 1ULL << 44;
104 std::numeric_limits<uint64_t>::max();
105static const uint64_t kSmallX86_64ShadowOffsetBase = 0x7FFFFFFF; // < 2G.
107static const uint64_t kLinuxKasan_ShadowOffset64 = 0xdffffc0000000000;
108static const uint64_t kPPC64_ShadowOffset64 = 1ULL << 44;
109static const uint64_t kSystemZ_ShadowOffset64 = 1ULL << 52;
110static const uint64_t kMIPS_ShadowOffsetN32 = 1ULL << 29;
111static const uint64_t kMIPS32_ShadowOffset32 = 0x0aaa0000;
112static const uint64_t kMIPS64_ShadowOffset64 = 1ULL << 37;
113static const uint64_t kAArch64_ShadowOffset64 = 1ULL << 36;
114static const uint64_t kLoongArch64_ShadowOffset64 = 1ULL << 46;
116static const uint64_t kFreeBSD_ShadowOffset32 = 1ULL << 30;
117static const uint64_t kFreeBSD_ShadowOffset64 = 1ULL << 46;
118static const uint64_t kFreeBSDAArch64_ShadowOffset64 = 1ULL << 47;
119static const uint64_t kFreeBSDKasan_ShadowOffset64 = 0xdffff7c000000000;
120static const uint64_t kNetBSD_ShadowOffset32 = 1ULL << 30;
121static const uint64_t kNetBSD_ShadowOffset64 = 1ULL << 46;
122static const uint64_t kNetBSDKasan_ShadowOffset64 = 0xdfff900000000000;
123static const uint64_t kPS_ShadowOffset64 = 1ULL << 40;
124static const uint64_t kWindowsShadowOffset32 = 3ULL << 28;
126
127// The shadow memory space is dynamically allocated.
129
130static const size_t kMinStackMallocSize = 1 << 6; // 64B
131static const size_t kMaxStackMallocSize = 1 << 16; // 64K
132static const uintptr_t kCurrentStackFrameMagic = 0x41B58AB3;
133static const uintptr_t kRetiredStackFrameMagic = 0x45E0360E;
134
135const char kAsanModuleCtorName[] = "asan.module_ctor";
136const char kAsanModuleDtorName[] = "asan.module_dtor";
138// On Emscripten, the system needs more than one priorities for constructors.
140const char kAsanReportErrorTemplate[] = "__asan_report_";
141const char kAsanRegisterGlobalsName[] = "__asan_register_globals";
142const char kAsanUnregisterGlobalsName[] = "__asan_unregister_globals";
143const char kAsanRegisterImageGlobalsName[] = "__asan_register_image_globals";
145 "__asan_unregister_image_globals";
146const char kAsanRegisterElfGlobalsName[] = "__asan_register_elf_globals";
147const char kAsanUnregisterElfGlobalsName[] = "__asan_unregister_elf_globals";
148const char kAsanPoisonGlobalsName[] = "__asan_before_dynamic_init";
149const char kAsanUnpoisonGlobalsName[] = "__asan_after_dynamic_init";
150const char kAsanInitName[] = "__asan_init";
151const char kAsanVersionCheckNamePrefix[] = "__asan_version_mismatch_check_v";
152const char kAsanPtrCmp[] = "__sanitizer_ptr_cmp";
153const char kAsanPtrSub[] = "__sanitizer_ptr_sub";
154const char kAsanHandleNoReturnName[] = "__asan_handle_no_return";
155static const int kMaxAsanStackMallocSizeClass = 10;
156const char kAsanStackMallocNameTemplate[] = "__asan_stack_malloc_";
158 "__asan_stack_malloc_always_";
159const char kAsanStackFreeNameTemplate[] = "__asan_stack_free_";
160const char kAsanGenPrefix[] = "___asan_gen_";
161const char kODRGenPrefix[] = "__odr_asan_gen_";
162const char kSanCovGenPrefix[] = "__sancov_gen_";
163const char kAsanSetShadowPrefix[] = "__asan_set_shadow_";
164const char kAsanPoisonStackMemoryName[] = "__asan_poison_stack_memory";
165const char kAsanUnpoisonStackMemoryName[] = "__asan_unpoison_stack_memory";
166
167// ASan version script has __asan_* wildcard. Triple underscore prevents a
168// linker (gold) warning about attempting to export a local symbol.
169const char kAsanGlobalsRegisteredFlagName[] = "___asan_globals_registered";
170
172 "__asan_option_detect_stack_use_after_return";
173
175 "__asan_shadow_memory_dynamic_address";
176
177const char kAsanAllocaPoison[] = "__asan_alloca_poison";
178const char kAsanAllocasUnpoison[] = "__asan_allocas_unpoison";
179
180const char kAMDGPUAddressSharedName[] = "llvm.amdgcn.is.shared";
181const char kAMDGPUAddressPrivateName[] = "llvm.amdgcn.is.private";
182const char kAMDGPUBallotName[] = "llvm.amdgcn.ballot.i64";
183const char kAMDGPUUnreachableName[] = "llvm.amdgcn.unreachable";
184
185// Accesses sizes are powers of two: 1, 2, 4, 8, 16.
186static const size_t kNumberOfAccessSizes = 5;
187
188static const uint64_t kAllocaRzSize = 32;
189
190// ASanAccessInfo implementation constants.
191constexpr size_t kCompileKernelShift = 0;
192constexpr size_t kCompileKernelMask = 0x1;
193constexpr size_t kAccessSizeIndexShift = 1;
194constexpr size_t kAccessSizeIndexMask = 0xf;
195constexpr size_t kIsWriteShift = 5;
196constexpr size_t kIsWriteMask = 0x1;
197
198// Command-line flags.
199
201 "asan-kernel", cl::desc("Enable KernelAddressSanitizer instrumentation"),
202 cl::Hidden, cl::init(false));
203
205 "asan-recover",
206 cl::desc("Enable recovery mode (continue-after-error)."),
207 cl::Hidden, cl::init(false));
208
210 "asan-guard-against-version-mismatch",
211 cl::desc("Guard against compiler/runtime version mismatch."), cl::Hidden,
212 cl::init(true));
213
214// This flag may need to be replaced with -f[no-]asan-reads.
215static cl::opt<bool> ClInstrumentReads("asan-instrument-reads",
216 cl::desc("instrument read instructions"),
217 cl::Hidden, cl::init(true));
218
220 "asan-instrument-writes", cl::desc("instrument write instructions"),
221 cl::Hidden, cl::init(true));
222
223static cl::opt<bool>
224 ClUseStackSafety("asan-use-stack-safety", cl::Hidden, cl::init(true),
225 cl::Hidden, cl::desc("Use Stack Safety analysis results"),
227
229 "asan-instrument-atomics",
230 cl::desc("instrument atomic instructions (rmw, cmpxchg)"), cl::Hidden,
231 cl::init(true));
232
233static cl::opt<bool>
234 ClInstrumentByval("asan-instrument-byval",
235 cl::desc("instrument byval call arguments"), cl::Hidden,
236 cl::init(true));
237
239 "asan-always-slow-path",
240 cl::desc("use instrumentation with slow path for all accesses"), cl::Hidden,
241 cl::init(false));
242
244 "asan-force-dynamic-shadow",
245 cl::desc("Load shadow address into a local variable for each function"),
246 cl::Hidden, cl::init(false));
247
248static cl::opt<bool>
249 ClWithIfunc("asan-with-ifunc",
250 cl::desc("Access dynamic shadow through an ifunc global on "
251 "platforms that support this"),
252 cl::Hidden, cl::init(true));
253
254static cl::opt<int>
255 ClShadowAddrSpace("asan-shadow-addr-space",
256 cl::desc("Address space for pointers to the shadow map"),
257 cl::Hidden, cl::init(0));
258
260 "asan-with-ifunc-suppress-remat",
261 cl::desc("Suppress rematerialization of dynamic shadow address by passing "
262 "it through inline asm in prologue."),
263 cl::Hidden, cl::init(true));
264
265// This flag limits the number of instructions to be instrumented
266// in any given BB. Normally, this should be set to unlimited (INT_MAX),
267// but due to http://llvm.org/bugs/show_bug.cgi?id=12652 we temporary
268// set it to 10000.
270 "asan-max-ins-per-bb", cl::init(10000),
271 cl::desc("maximal number of instructions to instrument in any given BB"),
272 cl::Hidden);
273
274// This flag may need to be replaced with -f[no]asan-stack.
275static cl::opt<bool> ClStack("asan-stack", cl::desc("Handle stack memory"),
276 cl::Hidden, cl::init(true));
278 "asan-max-inline-poisoning-size",
279 cl::desc(
280 "Inline shadow poisoning for blocks up to the given size in bytes."),
281 cl::Hidden, cl::init(64));
282
284 "asan-use-after-return",
285 cl::desc("Sets the mode of detection for stack-use-after-return."),
288 "Never detect stack use after return."),
291 "Detect stack use after return if "
292 "binary flag 'ASAN_OPTIONS=detect_stack_use_after_return' is set."),
294 "Always detect stack use after return.")),
296
297static cl::opt<bool> ClRedzoneByvalArgs("asan-redzone-byval-args",
298 cl::desc("Create redzones for byval "
299 "arguments (extra copy "
300 "required)"), cl::Hidden,
301 cl::init(true));
302
303static cl::opt<bool> ClUseAfterScope("asan-use-after-scope",
304 cl::desc("Check stack-use-after-scope"),
305 cl::Hidden, cl::init(false));
306
307// This flag may need to be replaced with -f[no]asan-globals.
308static cl::opt<bool> ClGlobals("asan-globals",
309 cl::desc("Handle global objects"), cl::Hidden,
310 cl::init(true));
311
312static cl::opt<bool> ClInitializers("asan-initialization-order",
313 cl::desc("Handle C++ initializer order"),
314 cl::Hidden, cl::init(true));
315
317 "asan-detect-invalid-pointer-pair",
318 cl::desc("Instrument <, <=, >, >=, - with pointer operands"), cl::Hidden,
319 cl::init(false));
320
322 "asan-detect-invalid-pointer-cmp",
323 cl::desc("Instrument <, <=, >, >= with pointer operands"), cl::Hidden,
324 cl::init(false));
325
327 "asan-detect-invalid-pointer-sub",
328 cl::desc("Instrument - operations with pointer operands"), cl::Hidden,
329 cl::init(false));
330
332 "asan-realign-stack",
333 cl::desc("Realign stack to the value of this flag (power of two)"),
334 cl::Hidden, cl::init(32));
335
337 "asan-instrumentation-with-call-threshold",
338 cl::desc("If the function being instrumented contains more than "
339 "this number of memory accesses, use callbacks instead of "
340 "inline checks (-1 means never use callbacks)."),
341 cl::Hidden, cl::init(7000));
342
344 "asan-memory-access-callback-prefix",
345 cl::desc("Prefix for memory access callbacks"), cl::Hidden,
346 cl::init("__asan_"));
347
349 "asan-kernel-mem-intrinsic-prefix",
350 cl::desc("Use prefix for memory intrinsics in KASAN mode"), cl::Hidden,
351 cl::init(false));
352
353static cl::opt<bool>
354 ClInstrumentDynamicAllocas("asan-instrument-dynamic-allocas",
355 cl::desc("instrument dynamic allocas"),
356 cl::Hidden, cl::init(true));
357
359 "asan-skip-promotable-allocas",
360 cl::desc("Do not instrument promotable allocas"), cl::Hidden,
361 cl::init(true));
362
364 "asan-constructor-kind",
365 cl::desc("Sets the ASan constructor kind"),
366 cl::values(clEnumValN(AsanCtorKind::None, "none", "No constructors"),
368 "Use global constructors")),
370// These flags allow to change the shadow mapping.
371// The shadow mapping looks like
372// Shadow = (Mem >> scale) + offset
373
374static cl::opt<int> ClMappingScale("asan-mapping-scale",
375 cl::desc("scale of asan shadow mapping"),
376 cl::Hidden, cl::init(0));
377
379 ClMappingOffset("asan-mapping-offset",
380 cl::desc("offset of asan shadow mapping [EXPERIMENTAL]"),
381 cl::Hidden, cl::init(0));
382
383// Optimization flags. Not user visible, used mostly for testing
384// and benchmarking the tool.
385
386static cl::opt<bool> ClOpt("asan-opt", cl::desc("Optimize instrumentation"),
387 cl::Hidden, cl::init(true));
388
389static cl::opt<bool> ClOptimizeCallbacks("asan-optimize-callbacks",
390 cl::desc("Optimize callbacks"),
391 cl::Hidden, cl::init(false));
392
394 "asan-opt-same-temp", cl::desc("Instrument the same temp just once"),
395 cl::Hidden, cl::init(true));
396
397static cl::opt<bool> ClOptGlobals("asan-opt-globals",
398 cl::desc("Don't instrument scalar globals"),
399 cl::Hidden, cl::init(true));
400
402 "asan-opt-stack", cl::desc("Don't instrument scalar stack variables"),
403 cl::Hidden, cl::init(false));
404
406 "asan-stack-dynamic-alloca",
407 cl::desc("Use dynamic alloca to represent stack variables"), cl::Hidden,
408 cl::init(true));
409
411 "asan-force-experiment",
412 cl::desc("Force optimization experiment (for testing)"), cl::Hidden,
413 cl::init(0));
414
415static cl::opt<bool>
416 ClUsePrivateAlias("asan-use-private-alias",
417 cl::desc("Use private aliases for global variables"),
418 cl::Hidden, cl::init(true));
419
420static cl::opt<bool>
421 ClUseOdrIndicator("asan-use-odr-indicator",
422 cl::desc("Use odr indicators to improve ODR reporting"),
423 cl::Hidden, cl::init(true));
424
425static cl::opt<bool>
426 ClUseGlobalsGC("asan-globals-live-support",
427 cl::desc("Use linker features to support dead "
428 "code stripping of globals"),
429 cl::Hidden, cl::init(true));
430
431// This is on by default even though there is a bug in gold:
432// https://sourceware.org/bugzilla/show_bug.cgi?id=19002
433static cl::opt<bool>
434 ClWithComdat("asan-with-comdat",
435 cl::desc("Place ASan constructors in comdat sections"),
436 cl::Hidden, cl::init(true));
437
439 "asan-destructor-kind",
440 cl::desc("Sets the ASan destructor kind. The default is to use the value "
441 "provided to the pass constructor"),
442 cl::values(clEnumValN(AsanDtorKind::None, "none", "No destructors"),
444 "Use global destructors")),
446
449 "asan-instrument-address-spaces",
450 cl::desc("Only instrument variables in the specified address spaces."),
451 cl::Hidden, cl::CommaSeparated, cl::callback([](const unsigned &AddrSpace) {
452 SrcAddrSpaces.insert(AddrSpace);
453 }));
454
455// Debug flags.
456
457static cl::opt<int> ClDebug("asan-debug", cl::desc("debug"), cl::Hidden,
458 cl::init(0));
459
460static cl::opt<int> ClDebugStack("asan-debug-stack", cl::desc("debug stack"),
461 cl::Hidden, cl::init(0));
462
464 cl::desc("Debug func"));
465
466static cl::opt<int> ClDebugMin("asan-debug-min", cl::desc("Debug min inst"),
467 cl::Hidden, cl::init(-1));
468
469static cl::opt<int> ClDebugMax("asan-debug-max", cl::desc("Debug max inst"),
470 cl::Hidden, cl::init(-1));
471
472STATISTIC(NumInstrumentedReads, "Number of instrumented reads");
473STATISTIC(NumInstrumentedWrites, "Number of instrumented writes");
474STATISTIC(NumOptimizedAccessesToGlobalVar,
475 "Number of optimized accesses to global vars");
476STATISTIC(NumOptimizedAccessesToStackVar,
477 "Number of optimized accesses to stack vars");
478
479namespace {
480
481/// This struct defines the shadow mapping using the rule:
482/// shadow = (mem >> Scale) ADD-or-OR Offset.
483/// If InGlobal is true, then
484/// extern char __asan_shadow[];
485/// shadow = (mem >> Scale) + &__asan_shadow
486struct ShadowMapping {
487 int Scale;
489 bool OrShadowOffset;
490 bool InGlobal;
491};
492
493} // end anonymous namespace
494
495static ShadowMapping getShadowMapping(const Triple &TargetTriple, int LongSize,
496 bool IsKasan) {
497 bool IsAndroid = TargetTriple.isAndroid();
498 bool IsIOS = TargetTriple.isiOS() || TargetTriple.isWatchOS() ||
499 TargetTriple.isDriverKit();
500 bool IsMacOS = TargetTriple.isMacOSX();
501 bool IsFreeBSD = TargetTriple.isOSFreeBSD();
502 bool IsNetBSD = TargetTriple.isOSNetBSD();
503 bool IsPS = TargetTriple.isPS();
504 bool IsLinux = TargetTriple.isOSLinux();
505 bool IsPPC64 = TargetTriple.getArch() == Triple::ppc64 ||
506 TargetTriple.getArch() == Triple::ppc64le;
507 bool IsSystemZ = TargetTriple.getArch() == Triple::systemz;
508 bool IsX86_64 = TargetTriple.getArch() == Triple::x86_64;
509 bool IsMIPSN32ABI = TargetTriple.isABIN32();
510 bool IsMIPS32 = TargetTriple.isMIPS32();
511 bool IsMIPS64 = TargetTriple.isMIPS64();
512 bool IsArmOrThumb = TargetTriple.isARM() || TargetTriple.isThumb();
513 bool IsAArch64 = TargetTriple.getArch() == Triple::aarch64 ||
514 TargetTriple.getArch() == Triple::aarch64_be;
515 bool IsLoongArch64 = TargetTriple.isLoongArch64();
516 bool IsRISCV64 = TargetTriple.getArch() == Triple::riscv64;
517 bool IsWindows = TargetTriple.isOSWindows();
518 bool IsFuchsia = TargetTriple.isOSFuchsia();
519 bool IsAMDGPU = TargetTriple.isAMDGPU();
520 bool IsHaiku = TargetTriple.isOSHaiku();
521 bool IsWasm = TargetTriple.isWasm();
522 bool IsBPF = TargetTriple.isBPF();
523
524 ShadowMapping Mapping;
525
526 Mapping.Scale = kDefaultShadowScale;
527 if (ClMappingScale.getNumOccurrences() > 0) {
528 Mapping.Scale = ClMappingScale;
529 }
530
531 if (LongSize == 32) {
532 if (IsAndroid)
533 Mapping.Offset = kDynamicShadowSentinel;
534 else if (IsMIPSN32ABI)
535 Mapping.Offset = kMIPS_ShadowOffsetN32;
536 else if (IsMIPS32)
537 Mapping.Offset = kMIPS32_ShadowOffset32;
538 else if (IsFreeBSD)
539 Mapping.Offset = kFreeBSD_ShadowOffset32;
540 else if (IsNetBSD)
541 Mapping.Offset = kNetBSD_ShadowOffset32;
542 else if (IsIOS)
543 Mapping.Offset = kDynamicShadowSentinel;
544 else if (IsWindows)
545 Mapping.Offset = kWindowsShadowOffset32;
546 else if (IsWasm)
547 Mapping.Offset = kWebAssemblyShadowOffset;
548 else
549 Mapping.Offset = kDefaultShadowOffset32;
550 } else { // LongSize == 64
551 // Fuchsia is always PIE, which means that the beginning of the address
552 // space is always available.
553 if (IsFuchsia) {
554 // kDynamicShadowSentinel tells instrumentation to use the dynamic shadow.
555 Mapping.Offset = kDynamicShadowSentinel;
556 } else if (IsPPC64)
557 Mapping.Offset = kPPC64_ShadowOffset64;
558 else if (IsSystemZ)
559 Mapping.Offset = kSystemZ_ShadowOffset64;
560 else if (IsFreeBSD && IsAArch64)
561 Mapping.Offset = kFreeBSDAArch64_ShadowOffset64;
562 else if (IsFreeBSD && !IsMIPS64) {
563 if (IsKasan)
564 Mapping.Offset = kFreeBSDKasan_ShadowOffset64;
565 else
566 Mapping.Offset = kFreeBSD_ShadowOffset64;
567 } else if (IsNetBSD) {
568 if (IsKasan)
569 Mapping.Offset = kNetBSDKasan_ShadowOffset64;
570 else
571 Mapping.Offset = kNetBSD_ShadowOffset64;
572 } else if (IsPS)
573 Mapping.Offset = kPS_ShadowOffset64;
574 else if (IsLinux && IsX86_64) {
575 if (IsKasan)
576 Mapping.Offset = kLinuxKasan_ShadowOffset64;
577 else
578 Mapping.Offset = (kSmallX86_64ShadowOffsetBase &
579 (kSmallX86_64ShadowOffsetAlignMask << Mapping.Scale));
580 } else if (IsWindows && (IsX86_64 || IsAArch64)) {
581 Mapping.Offset = kWindowsShadowOffset64;
582 } else if (IsMIPS64)
583 Mapping.Offset = kMIPS64_ShadowOffset64;
584 else if (IsIOS)
585 Mapping.Offset = kDynamicShadowSentinel;
586 else if (IsMacOS && IsAArch64)
587 Mapping.Offset = kDynamicShadowSentinel;
588 else if (IsAArch64)
589 Mapping.Offset = kAArch64_ShadowOffset64;
590 else if (IsLoongArch64)
591 Mapping.Offset = kLoongArch64_ShadowOffset64;
592 else if (IsRISCV64)
593 Mapping.Offset = kRISCV64_ShadowOffset64;
594 else if (IsAMDGPU)
595 Mapping.Offset = (kSmallX86_64ShadowOffsetBase &
596 (kSmallX86_64ShadowOffsetAlignMask << Mapping.Scale));
597 else if (IsHaiku && IsX86_64)
598 Mapping.Offset = (kSmallX86_64ShadowOffsetBase &
599 (kSmallX86_64ShadowOffsetAlignMask << Mapping.Scale));
600 else if (IsBPF)
601 Mapping.Offset = kDynamicShadowSentinel;
602 else if (IsWasm)
603 Mapping.Offset = kWebAssemblyShadowOffset;
604 else
605 Mapping.Offset = kDefaultShadowOffset64;
606 }
607
609 Mapping.Offset = kDynamicShadowSentinel;
610 }
611
612 if (ClMappingOffset.getNumOccurrences() > 0) {
613 Mapping.Offset = ClMappingOffset;
614 }
615
616 // OR-ing shadow offset if more efficient (at least on x86) if the offset
617 // is a power of two, but on ppc64 and loongarch64 we have to use add since
618 // the shadow offset is not necessarily 1/8-th of the address space. On
619 // SystemZ, we could OR the constant in a single instruction, but it's more
620 // efficient to load it once and use indexed addressing.
621 Mapping.OrShadowOffset = !IsAArch64 && !IsPPC64 && !IsSystemZ && !IsPS &&
622 !IsRISCV64 && !IsLoongArch64 &&
623 !(Mapping.Offset & (Mapping.Offset - 1)) &&
624 Mapping.Offset != kDynamicShadowSentinel;
625 Mapping.InGlobal = ClWithIfunc && IsAndroid && IsArmOrThumb;
626
627 return Mapping;
628}
629
630void llvm::getAddressSanitizerParams(const Triple &TargetTriple, int LongSize,
631 bool IsKasan, uint64_t *ShadowBase,
632 int *MappingScale, bool *OrShadowOffset) {
633 auto Mapping = getShadowMapping(TargetTriple, LongSize, IsKasan);
634 *ShadowBase = Mapping.Offset;
635 *MappingScale = Mapping.Scale;
636 *OrShadowOffset = Mapping.OrShadowOffset;
637}
638
640 // Adding sanitizer checks invalidates previously inferred memory attributes.
641 //
642 // This is not only true for sanitized functions, because AttrInfer can
643 // infer those attributes on libc functions, which is not true if those
644 // are instrumented (Android) or intercepted.
645 //
646 // We might want to model ASan shadow memory more opaquely to get rid of
647 // this problem altogether, by hiding the shadow memory write in an
648 // intrinsic, essentially like in the AArch64StackTagging pass. But that's
649 // for another day.
650
651 bool Changed = false;
652 // We add memory(readwrite) to functions that don't already have that set and
653 // can access any non-inaccessible memory. Sanitizer instrumentation can
654 // read/write shadow memory, which is IRMemLocation::Other. Sanitizer
655 // instrumentation can instrument any memory accesses to non-inaccessible
656 // memory.
657 if (!F.getMemoryEffects()
658 .getWithoutLoc(IRMemLocation::InaccessibleMem)
659 .doesNotAccessMemory() &&
660 !isModAndRefSet(F.getMemoryEffects().getModRef(IRMemLocation::Other))) {
661 F.setMemoryEffects(F.getMemoryEffects() |
663 Changed = true;
664 }
665 // HWASan reads from argument memory even for previously write-only accesses.
666 if (ReadsArgMem) {
667 if (F.getMemoryEffects().getModRef(IRMemLocation::ArgMem) ==
669 F.setMemoryEffects(F.getMemoryEffects() |
671 Changed = true;
672 }
673 for (Argument &A : F.args()) {
674 if (A.hasAttribute(Attribute::WriteOnly)) {
675 A.removeAttr(Attribute::WriteOnly);
676 Changed = true;
677 }
678 }
679 }
680 if (Changed) {
681 // nobuiltin makes sure later passes don't restore assumptions about
682 // the function.
683 F.addFnAttr(Attribute::NoBuiltin);
684 }
685}
686
692
700
701static uint64_t getRedzoneSizeForScale(int MappingScale) {
702 // Redzone used for stack and globals is at least 32 bytes.
703 // For scales 6 and 7, the redzone has to be 64 and 128 bytes respectively.
704 return std::max(32U, 1U << MappingScale);
705}
706
708 if (TargetTriple.isOSEmscripten())
710 else
712}
713
714static Twine genName(StringRef suffix) {
715 return Twine(kAsanGenPrefix) + suffix;
716}
717
718namespace {
719
720class AsanFunctionInserter {
721public:
722 AsanFunctionInserter(Module &M) : M(M) {}
723
724 template <typename... ArgTypes>
725 FunctionCallee insertFunction(StringRef Name, ArgTypes &&...Args) {
726 return M.getOrInsertFunction(Name, std::forward<ArgTypes>(Args)...);
727 }
728
729private:
730 Module &M;
731};
732
733} // end anonymous namespace
734
735namespace {
736/// Helper RAII class to post-process inserted asan runtime calls during a
737/// pass on a single Function. Upon end of scope, detects and applies the
738/// required funclet OpBundle.
739class RuntimeCallInserter {
740 Function *OwnerFn = nullptr;
741 bool TrackInsertedCalls = false;
742 SmallVector<CallInst *> InsertedCalls;
743
744public:
745 RuntimeCallInserter(Function &Fn) : OwnerFn(&Fn) {
746 if (Fn.hasPersonalityFn()) {
747 auto Personality = classifyEHPersonality(Fn.getPersonalityFn());
748 if (isScopedEHPersonality(Personality))
749 TrackInsertedCalls = true;
750 }
751 }
752
753 ~RuntimeCallInserter() {
754 if (InsertedCalls.empty())
755 return;
756 assert(TrackInsertedCalls && "Calls were wrongly tracked");
757
758 DenseMap<BasicBlock *, ColorVector> BlockColors = colorEHFunclets(*OwnerFn);
759 for (CallInst *CI : InsertedCalls) {
760 BasicBlock *BB = CI->getParent();
761 assert(BB && "Instruction doesn't belong to a BasicBlock");
762 assert(BB->getParent() == OwnerFn &&
763 "Instruction doesn't belong to the expected Function!");
764
765 ColorVector &Colors = BlockColors[BB];
766 // funclet opbundles are only valid in monochromatic BBs.
767 // Note that unreachable BBs are seen as colorless by colorEHFunclets()
768 // and will be DCE'ed later.
769 if (Colors.empty())
770 continue;
771 if (Colors.size() != 1) {
772 OwnerFn->getContext().emitError(
773 "Instruction's BasicBlock is not monochromatic");
774 continue;
775 }
776
777 BasicBlock *Color = Colors.front();
778 BasicBlock::iterator EHPadIt = Color->getFirstNonPHIIt();
779
780 if (EHPadIt != Color->end() && EHPadIt->isEHPad()) {
781 // Replace CI with a clone with an added funclet OperandBundle
782 OperandBundleDef OB("funclet", &*EHPadIt);
784 OB, CI->getIterator());
785 NewCall->copyMetadata(*CI);
786 CI->replaceAllUsesWith(NewCall);
787 CI->eraseFromParent();
788 }
789 }
790 }
791
792 CallInst *createRuntimeCall(IRBuilder<> &IRB, FunctionCallee Callee,
793 ArrayRef<Value *> Args = {},
794 const Twine &Name = "") {
795 assert(IRB.GetInsertBlock()->getParent() == OwnerFn);
796
797 CallInst *Inst = IRB.CreateCall(Callee, Args, Name, nullptr);
798 if (TrackInsertedCalls)
799 InsertedCalls.push_back(Inst);
800 return Inst;
801 }
802};
803
804/// AddressSanitizer: instrument the code in module to find memory bugs.
805struct AddressSanitizer {
806 AddressSanitizer(Module &M, const StackSafetyGlobalInfo *SSGI,
807 int InstrumentationWithCallsThreshold,
808 uint32_t MaxInlinePoisoningSize, bool CompileKernel = false,
809 bool Recover = false, bool UseAfterScope = false,
810 AsanDetectStackUseAfterReturnMode UseAfterReturn =
811 AsanDetectStackUseAfterReturnMode::Runtime)
812 : M(M), Inserter(M),
813 CompileKernel(ClEnableKasan.getNumOccurrences() > 0 ? ClEnableKasan
814 : CompileKernel),
815 Recover(ClRecover.getNumOccurrences() > 0 ? ClRecover : Recover),
816 UseAfterScope(UseAfterScope || ClUseAfterScope),
817 UseAfterReturn(ClUseAfterReturn.getNumOccurrences() ? ClUseAfterReturn
818 : UseAfterReturn),
819 SSGI(SSGI),
820 InstrumentationWithCallsThreshold(
821 ClInstrumentationWithCallsThreshold.getNumOccurrences() > 0
823 : InstrumentationWithCallsThreshold),
824 MaxInlinePoisoningSize(ClMaxInlinePoisoningSize.getNumOccurrences() > 0
826 : MaxInlinePoisoningSize) {
827 C = &(M.getContext());
828 DL = &M.getDataLayout();
829 LongSize = M.getDataLayout().getPointerSizeInBits();
830 IntptrTy = Type::getIntNTy(*C, LongSize);
831 PtrTy = PointerType::getUnqual(*C);
832 Int32Ty = Type::getInt32Ty(*C);
833 TargetTriple = M.getTargetTriple();
834
835 Mapping = getShadowMapping(TargetTriple, LongSize, this->CompileKernel);
836
837 assert(this->UseAfterReturn != AsanDetectStackUseAfterReturnMode::Invalid);
838 }
839
840 TypeSize getAllocaSizeInBytes(const AllocaInst &AI) const {
841 return *AI.getAllocationSize(AI.getDataLayout());
842 }
843
844 /// Check if we want (and can) handle this alloca.
845 bool isInterestingAlloca(const AllocaInst &AI);
846
847 bool ignoreAccess(Instruction *Inst, Value *Ptr);
849 Instruction *I, SmallVectorImpl<InterestingMemoryOperand> &Interesting,
850 const TargetTransformInfo *TTI);
851
852 void instrumentMop(ObjectSizeOffsetVisitor &ObjSizeVis,
853 InterestingMemoryOperand &O, bool UseCalls,
854 const DataLayout &DL, RuntimeCallInserter &RTCI);
855 void instrumentPointerComparisonOrSubtraction(Instruction *I,
856 RuntimeCallInserter &RTCI);
857 void instrumentAddress(Instruction *OrigIns, Instruction *InsertBefore,
858 Value *Addr, MaybeAlign Alignment,
859 uint32_t TypeStoreSize, bool IsWrite,
860 Value *SizeArgument, bool UseCalls, uint32_t Exp,
861 RuntimeCallInserter &RTCI);
862 Instruction *instrumentAMDGPUAddress(Instruction *OrigIns,
863 Instruction *InsertBefore, Value *Addr,
864 uint32_t TypeStoreSize, bool IsWrite,
865 Value *SizeArgument);
866 Instruction *genAMDGPUReportBlock(IRBuilder<> &IRB, Value *Cond,
867 bool Recover);
868 void instrumentUnusualSizeOrAlignment(Instruction *I,
869 Instruction *InsertBefore, Value *Addr,
870 TypeSize TypeStoreSize, bool IsWrite,
871 Value *SizeArgument, bool UseCalls,
872 uint32_t Exp,
873 RuntimeCallInserter &RTCI);
874 void instrumentMaskedLoadOrStore(AddressSanitizer *Pass, const DataLayout &DL,
875 Type *IntptrTy, Value *Mask, Value *EVL,
876 Value *Stride, Instruction *I, Value *Addr,
877 MaybeAlign Alignment, unsigned Granularity,
878 Type *OpType, bool IsWrite,
879 Value *SizeArgument, bool UseCalls,
880 uint32_t Exp, RuntimeCallInserter &RTCI);
881 Value *createSlowPathCmp(IRBuilder<> &IRB, Value *AddrLong,
882 Value *ShadowValue, uint32_t TypeStoreSize);
883 Instruction *generateCrashCode(Instruction *InsertBefore, Value *Addr,
884 bool IsWrite, size_t AccessSizeIndex,
885 Value *SizeArgument, uint32_t Exp,
886 RuntimeCallInserter &RTCI);
887 void instrumentMemIntrinsic(MemIntrinsic *MI, RuntimeCallInserter &RTCI);
888 Value *memToShadow(Value *Shadow, IRBuilder<> &IRB);
889 bool suppressInstrumentationSiteForDebug(int &Instrumented);
890 bool instrumentFunction(Function &F, const TargetLibraryInfo *TLI,
891 const TargetTransformInfo *TTI);
892 bool maybeInsertAsanInitAtFunctionEntry(Function &F);
893 bool maybeInsertDynamicShadowAtFunctionEntry(Function &F);
894 void markEscapedLocalAllocas(Function &F);
895 void markCatchParametersAsUninteresting(Function &F);
896
897private:
898 friend struct FunctionStackPoisoner;
899
900 void initializeCallbacks(const TargetLibraryInfo *TLI);
901
902 bool LooksLikeCodeInBug11395(Instruction *I);
903 bool GlobalIsLinkerInitialized(GlobalVariable *G);
904 bool isSafeAccess(ObjectSizeOffsetVisitor &ObjSizeVis, Value *Addr,
905 TypeSize TypeStoreSize) const;
906
907 /// Helper to cleanup per-function state.
908 struct FunctionStateRAII {
909 AddressSanitizer *Pass;
910
911 FunctionStateRAII(AddressSanitizer *Pass) : Pass(Pass) {
912 assert(Pass->ProcessedAllocas.empty() &&
913 "last pass forgot to clear cache");
914 assert(!Pass->LocalDynamicShadow);
915 }
916
917 ~FunctionStateRAII() {
918 Pass->LocalDynamicShadow = nullptr;
919 Pass->ProcessedAllocas.clear();
920 }
921 };
922
923 Module &M;
924 AsanFunctionInserter Inserter;
925 LLVMContext *C;
926 const DataLayout *DL;
927 Triple TargetTriple;
928 int LongSize;
929 bool CompileKernel;
930 bool Recover;
931 bool UseAfterScope;
933 Type *IntptrTy;
934 Type *Int32Ty;
935 PointerType *PtrTy;
936 ShadowMapping Mapping;
937 FunctionCallee AsanHandleNoReturnFunc;
938 FunctionCallee AsanPtrCmpFunction, AsanPtrSubFunction;
939 Constant *AsanShadowGlobal;
940
941 // These arrays is indexed by AccessIsWrite, Experiment and log2(AccessSize).
942 FunctionCallee AsanErrorCallback[2][2][kNumberOfAccessSizes];
943 FunctionCallee AsanMemoryAccessCallback[2][2][kNumberOfAccessSizes];
944
945 // These arrays is indexed by AccessIsWrite and Experiment.
946 FunctionCallee AsanErrorCallbackSized[2][2];
947 FunctionCallee AsanMemoryAccessCallbackSized[2][2];
948
949 FunctionCallee AsanMemmove, AsanMemcpy, AsanMemset;
950 Value *LocalDynamicShadow = nullptr;
951 const StackSafetyGlobalInfo *SSGI;
952 DenseMap<const AllocaInst *, bool> ProcessedAllocas;
953
954 FunctionCallee AMDGPUAddressShared;
955 FunctionCallee AMDGPUAddressPrivate;
956 int InstrumentationWithCallsThreshold;
957 uint32_t MaxInlinePoisoningSize;
958};
959
960class ModuleAddressSanitizer {
961public:
962 ModuleAddressSanitizer(Module &M, bool InsertVersionCheck,
963 bool CompileKernel = false, bool Recover = false,
964 bool UseGlobalsGC = true, bool UseOdrIndicator = true,
965 AsanDtorKind DestructorKind = AsanDtorKind::Global,
966 AsanCtorKind ConstructorKind = AsanCtorKind::Global)
967 : M(M), Inserter(M),
968 CompileKernel(ClEnableKasan.getNumOccurrences() > 0 ? ClEnableKasan
969 : CompileKernel),
970 InsertVersionCheck(ClInsertVersionCheck.getNumOccurrences() > 0
972 : InsertVersionCheck),
973 Recover(ClRecover.getNumOccurrences() > 0 ? ClRecover : Recover),
974 UseGlobalsGC(UseGlobalsGC && ClUseGlobalsGC && !this->CompileKernel),
975 // Enable aliases as they should have no downside with ODR indicators.
976 UsePrivateAlias(ClUsePrivateAlias.getNumOccurrences() > 0
978 : UseOdrIndicator),
979 UseOdrIndicator(ClUseOdrIndicator.getNumOccurrences() > 0
981 : UseOdrIndicator),
982 // Not a typo: ClWithComdat is almost completely pointless without
983 // ClUseGlobalsGC (because then it only works on modules without
984 // globals, which are rare); it is a prerequisite for ClUseGlobalsGC;
985 // and both suffer from gold PR19002 for which UseGlobalsGC constructor
986 // argument is designed as workaround. Therefore, disable both
987 // ClWithComdat and ClUseGlobalsGC unless the frontend says it's ok to
988 // do globals-gc.
989 UseCtorComdat(UseGlobalsGC && ClWithComdat && !this->CompileKernel),
990 DestructorKind(DestructorKind),
991 ConstructorKind(ClConstructorKind.getNumOccurrences() > 0
993 : ConstructorKind) {
994 C = &(M.getContext());
995 int LongSize = M.getDataLayout().getPointerSizeInBits();
996 IntptrTy = Type::getIntNTy(*C, LongSize);
997 PtrTy = PointerType::getUnqual(*C);
998 TargetTriple = M.getTargetTriple();
999 Mapping = getShadowMapping(TargetTriple, LongSize, this->CompileKernel);
1000
1001 if (ClOverrideDestructorKind != AsanDtorKind::Invalid)
1002 this->DestructorKind = ClOverrideDestructorKind;
1003 assert(this->DestructorKind != AsanDtorKind::Invalid);
1004 }
1005
1006 bool instrumentModule();
1007
1008private:
1009 void initializeCallbacks();
1010
1011 void instrumentGlobals(IRBuilder<> &IRB, bool *CtorComdat);
1012 void InstrumentGlobalsCOFF(IRBuilder<> &IRB,
1013 ArrayRef<GlobalVariable *> ExtendedGlobals,
1014 ArrayRef<Constant *> MetadataInitializers);
1015 void instrumentGlobalsELF(IRBuilder<> &IRB,
1016 ArrayRef<GlobalVariable *> ExtendedGlobals,
1017 ArrayRef<Constant *> MetadataInitializers,
1018 const std::string &UniqueModuleId);
1019 void InstrumentGlobalsMachO(IRBuilder<> &IRB,
1020 ArrayRef<GlobalVariable *> ExtendedGlobals,
1021 ArrayRef<Constant *> MetadataInitializers);
1022 void
1023 InstrumentGlobalsWithMetadataArray(IRBuilder<> &IRB,
1024 ArrayRef<GlobalVariable *> ExtendedGlobals,
1025 ArrayRef<Constant *> MetadataInitializers);
1026
1027 GlobalVariable *CreateMetadataGlobal(Constant *Initializer,
1028 StringRef OriginalName);
1029 void SetComdatForGlobalMetadata(GlobalVariable *G, GlobalVariable *Metadata,
1030 StringRef InternalSuffix);
1031 Instruction *CreateAsanModuleDtor();
1032
1033 const GlobalVariable *getExcludedAliasedGlobal(const GlobalAlias &GA) const;
1034 bool shouldInstrumentGlobal(GlobalVariable *G) const;
1035 bool ShouldUseMachOGlobalsSection() const;
1036 StringRef getGlobalMetadataSection() const;
1037 void poisonOneInitializer(Function &GlobalInit);
1038 void createInitializerPoisonCalls();
1039 uint64_t getMinRedzoneSizeForGlobal() const {
1040 return getRedzoneSizeForScale(Mapping.Scale);
1041 }
1042 uint64_t getRedzoneSizeForGlobal(uint64_t SizeInBytes) const;
1043 int GetAsanVersion() const;
1044 GlobalVariable *getOrCreateModuleName();
1045
1046 Module &M;
1047 AsanFunctionInserter Inserter;
1048 bool CompileKernel;
1049 bool InsertVersionCheck;
1050 bool Recover;
1051 bool UseGlobalsGC;
1052 bool UsePrivateAlias;
1053 bool UseOdrIndicator;
1054 bool UseCtorComdat;
1055 AsanDtorKind DestructorKind;
1056 AsanCtorKind ConstructorKind;
1057 Type *IntptrTy;
1058 PointerType *PtrTy;
1059 LLVMContext *C;
1060 Triple TargetTriple;
1061 ShadowMapping Mapping;
1062 FunctionCallee AsanPoisonGlobals;
1063 FunctionCallee AsanUnpoisonGlobals;
1064 FunctionCallee AsanRegisterGlobals;
1065 FunctionCallee AsanUnregisterGlobals;
1066 FunctionCallee AsanRegisterImageGlobals;
1067 FunctionCallee AsanUnregisterImageGlobals;
1068 FunctionCallee AsanRegisterElfGlobals;
1069 FunctionCallee AsanUnregisterElfGlobals;
1070
1071 Function *AsanCtorFunction = nullptr;
1072 Function *AsanDtorFunction = nullptr;
1073 GlobalVariable *ModuleName = nullptr;
1074};
1075
1076// Stack poisoning does not play well with exception handling.
1077// When an exception is thrown, we essentially bypass the code
1078// that unpoisones the stack. This is why the run-time library has
1079// to intercept __cxa_throw (as well as longjmp, etc) and unpoison the entire
1080// stack in the interceptor. This however does not work inside the
1081// actual function which catches the exception. Most likely because the
1082// compiler hoists the load of the shadow value somewhere too high.
1083// This causes asan to report a non-existing bug on 453.povray.
1084// It sounds like an LLVM bug.
1085struct FunctionStackPoisoner : public InstVisitor<FunctionStackPoisoner> {
1086 Function &F;
1087 AddressSanitizer &ASan;
1088 RuntimeCallInserter &RTCI;
1089 DIBuilder DIB;
1090 LLVMContext *C;
1091 Type *IntptrTy;
1092 Type *IntptrPtrTy;
1093 ShadowMapping Mapping;
1094
1096 SmallVector<AllocaInst *, 16> StaticAllocasToMoveUp;
1097 SmallVector<Instruction *, 8> RetVec;
1098
1099 FunctionCallee AsanStackMallocFunc[kMaxAsanStackMallocSizeClass + 1],
1100 AsanStackFreeFunc[kMaxAsanStackMallocSizeClass + 1];
1101 FunctionCallee AsanSetShadowFunc[0x100] = {};
1102 FunctionCallee AsanPoisonStackMemoryFunc, AsanUnpoisonStackMemoryFunc;
1103 FunctionCallee AsanAllocaPoisonFunc, AsanAllocasUnpoisonFunc;
1104
1105 // Stores a place and arguments of poisoning/unpoisoning call for alloca.
1106 struct AllocaPoisonCall {
1107 IntrinsicInst *InsBefore;
1108 AllocaInst *AI;
1109 uint64_t Size;
1110 bool DoPoison;
1111 };
1112 SmallVector<AllocaPoisonCall, 8> DynamicAllocaPoisonCallVec;
1113 SmallVector<AllocaPoisonCall, 8> StaticAllocaPoisonCallVec;
1114
1115 SmallVector<AllocaInst *, 1> DynamicAllocaVec;
1116 SmallVector<IntrinsicInst *, 1> StackRestoreVec;
1117 AllocaInst *DynamicAllocaLayout = nullptr;
1118 IntrinsicInst *LocalEscapeCall = nullptr;
1119
1120 bool HasInlineAsm = false;
1121 bool HasReturnsTwiceCall = false;
1122 bool PoisonStack;
1123
1124 FunctionStackPoisoner(Function &F, AddressSanitizer &ASan,
1125 RuntimeCallInserter &RTCI)
1126 : F(F), ASan(ASan), RTCI(RTCI),
1127 DIB(*F.getParent(), /*AllowUnresolved*/ false), C(ASan.C),
1128 IntptrTy(ASan.IntptrTy),
1129 IntptrPtrTy(PointerType::get(IntptrTy->getContext(), 0)),
1130 Mapping(ASan.Mapping),
1131 PoisonStack(ClStack && !F.getParent()->getTargetTriple().isAMDGPU()) {}
1132
1133 bool runOnFunction() {
1134 if (!PoisonStack)
1135 return false;
1136
1138 copyArgsPassedByValToAllocas();
1139
1140 // Collect alloca, ret, lifetime instructions etc.
1141 for (BasicBlock *BB : depth_first(&F.getEntryBlock())) visit(*BB);
1142
1143 if (AllocaVec.empty() && DynamicAllocaVec.empty()) return false;
1144
1145 initializeCallbacks(*F.getParent());
1146
1147 processDynamicAllocas();
1148 processStaticAllocas();
1149
1150 if (ClDebugStack) {
1151 LLVM_DEBUG(dbgs() << F);
1152 }
1153 return true;
1154 }
1155
1156 // Arguments marked with the "byval" attribute are implicitly copied without
1157 // using an alloca instruction. To produce redzones for those arguments, we
1158 // copy them a second time into memory allocated with an alloca instruction.
1159 void copyArgsPassedByValToAllocas();
1160
1161 // Finds all Alloca instructions and puts
1162 // poisoned red zones around all of them.
1163 // Then unpoison everything back before the function returns.
1164 void processStaticAllocas();
1165 void processDynamicAllocas();
1166
1167 void createDynamicAllocasInitStorage();
1168
1169 // ----------------------- Visitors.
1170 /// Collect all Ret instructions, or the musttail call instruction if it
1171 /// precedes the return instruction.
1172 void visitReturnInst(ReturnInst &RI) {
1173 if (CallInst *CI = RI.getParent()->getTerminatingMustTailCall())
1174 RetVec.push_back(CI);
1175 else
1176 RetVec.push_back(&RI);
1177 }
1178
1179 /// Collect all Resume instructions.
1180 void visitResumeInst(ResumeInst &RI) { RetVec.push_back(&RI); }
1181
1182 /// Collect all CatchReturnInst instructions.
1183 void visitCleanupReturnInst(CleanupReturnInst &CRI) { RetVec.push_back(&CRI); }
1184
1185 void unpoisonDynamicAllocasBeforeInst(Instruction *InstBefore,
1186 Value *SavedStack) {
1187 IRBuilder<> IRB(InstBefore);
1188 Value *DynamicAreaPtr = IRB.CreatePtrToInt(SavedStack, IntptrTy);
1189 // When we insert _asan_allocas_unpoison before @llvm.stackrestore, we
1190 // need to adjust extracted SP to compute the address of the most recent
1191 // alloca. We have a special @llvm.get.dynamic.area.offset intrinsic for
1192 // this purpose.
1193 if (!isa<ReturnInst>(InstBefore)) {
1194 Value *DynamicAreaOffset = IRB.CreateIntrinsic(
1195 Intrinsic::get_dynamic_area_offset, {IntptrTy}, {});
1196
1197 DynamicAreaPtr = IRB.CreateAdd(IRB.CreatePtrToInt(SavedStack, IntptrTy),
1198 DynamicAreaOffset);
1199 }
1200
1201 RTCI.createRuntimeCall(
1202 IRB, AsanAllocasUnpoisonFunc,
1203 {IRB.CreateLoad(IntptrTy, DynamicAllocaLayout), DynamicAreaPtr});
1204 }
1205
1206 // Unpoison dynamic allocas redzones.
1207 void unpoisonDynamicAllocas() {
1208 for (Instruction *Ret : RetVec)
1209 unpoisonDynamicAllocasBeforeInst(Ret, DynamicAllocaLayout);
1210
1211 for (Instruction *StackRestoreInst : StackRestoreVec)
1212 unpoisonDynamicAllocasBeforeInst(StackRestoreInst,
1213 StackRestoreInst->getOperand(0));
1214 }
1215
1216 // Deploy and poison redzones around dynamic alloca call. To do this, we
1217 // should replace this call with another one with changed parameters and
1218 // replace all its uses with new address, so
1219 // addr = alloca type, old_size, align
1220 // is replaced by
1221 // new_size = (old_size + additional_size) * sizeof(type)
1222 // tmp = alloca i8, new_size, max(align, 32)
1223 // addr = tmp + 32 (first 32 bytes are for the left redzone).
1224 // Additional_size is added to make new memory allocation contain not only
1225 // requested memory, but also left, partial and right redzones.
1226 void handleDynamicAllocaCall(AllocaInst *AI);
1227
1228 /// Collect Alloca instructions we want (and can) handle.
1229 void visitAllocaInst(AllocaInst &AI) {
1230 // FIXME: Handle scalable vectors instead of ignoring them.
1231 const Type *AllocaType = AI.getAllocatedType();
1232 const auto *STy = dyn_cast<StructType>(AllocaType);
1233 if (!ASan.isInterestingAlloca(AI) || isa<ScalableVectorType>(AllocaType) ||
1234 (STy && STy->containsHomogeneousScalableVectorTypes())) {
1235 if (AI.isStaticAlloca()) {
1236 // Skip over allocas that are present *before* the first instrumented
1237 // alloca, we don't want to move those around.
1238 if (AllocaVec.empty())
1239 return;
1240
1241 StaticAllocasToMoveUp.push_back(&AI);
1242 }
1243 return;
1244 }
1245
1246 if (!AI.isStaticAlloca())
1247 DynamicAllocaVec.push_back(&AI);
1248 else
1249 AllocaVec.push_back(&AI);
1250 }
1251
1252 /// Collect lifetime intrinsic calls to check for use-after-scope
1253 /// errors.
1254 void visitIntrinsicInst(IntrinsicInst &II) {
1255 Intrinsic::ID ID = II.getIntrinsicID();
1256 if (ID == Intrinsic::stackrestore) StackRestoreVec.push_back(&II);
1257 if (ID == Intrinsic::localescape) LocalEscapeCall = &II;
1258 if (!ASan.UseAfterScope)
1259 return;
1260 if (!II.isLifetimeStartOrEnd())
1261 return;
1262 // Find alloca instruction that corresponds to llvm.lifetime argument.
1263 AllocaInst *AI = dyn_cast<AllocaInst>(II.getArgOperand(0));
1264 // We're interested only in allocas we can handle.
1265 if (!AI || !ASan.isInterestingAlloca(*AI))
1266 return;
1267
1268 std::optional<TypeSize> Size = AI->getAllocationSize(AI->getDataLayout());
1269 // Check that size is known and can be stored in IntptrTy.
1270 // TODO: Add support for scalable vectors if possible.
1271 if (!Size || Size->isScalable() ||
1273 return;
1274
1275 bool DoPoison = (ID == Intrinsic::lifetime_end);
1276 AllocaPoisonCall APC = {&II, AI, *Size, DoPoison};
1277 if (AI->isStaticAlloca())
1278 StaticAllocaPoisonCallVec.push_back(APC);
1280 DynamicAllocaPoisonCallVec.push_back(APC);
1281 }
1282
1283 void visitCallBase(CallBase &CB) {
1284 if (CallInst *CI = dyn_cast<CallInst>(&CB)) {
1285 HasInlineAsm |= CI->isInlineAsm() && &CB != ASan.LocalDynamicShadow;
1286 HasReturnsTwiceCall |= CI->canReturnTwice();
1287 }
1288 }
1289
1290 // ---------------------- Helpers.
1291 void initializeCallbacks(Module &M);
1292
1293 // Copies bytes from ShadowBytes into shadow memory for indexes where
1294 // ShadowMask is not zero. If ShadowMask[i] is zero, we assume that
1295 // ShadowBytes[i] is constantly zero and doesn't need to be overwritten.
1296 void copyToShadow(ArrayRef<uint8_t> ShadowMask, ArrayRef<uint8_t> ShadowBytes,
1297 IRBuilder<> &IRB, Value *ShadowBase);
1298 void copyToShadow(ArrayRef<uint8_t> ShadowMask, ArrayRef<uint8_t> ShadowBytes,
1299 size_t Begin, size_t End, IRBuilder<> &IRB,
1300 Value *ShadowBase);
1301 void copyToShadowInline(ArrayRef<uint8_t> ShadowMask,
1302 ArrayRef<uint8_t> ShadowBytes, size_t Begin,
1303 size_t End, IRBuilder<> &IRB, Value *ShadowBase);
1304
1305 void poisonAlloca(Value *V, uint64_t Size, IRBuilder<> &IRB, bool DoPoison);
1306
1307 Value *createAllocaForLayout(IRBuilder<> &IRB, const ASanStackFrameLayout &L,
1308 bool Dynamic);
1309 PHINode *createPHI(IRBuilder<> &IRB, Value *Cond, Value *ValueIfTrue,
1310 Instruction *ThenTerm, Value *ValueIfFalse);
1311};
1312
1313} // end anonymous namespace
1314
1316 raw_ostream &OS, function_ref<StringRef(StringRef)> MapClassName2PassName) {
1318 OS, MapClassName2PassName);
1319 OS << '<';
1320 if (Options.CompileKernel)
1321 OS << "kernel;";
1322 if (Options.UseAfterScope)
1323 OS << "use-after-scope";
1324 OS << '>';
1325}
1326
1328 const AddressSanitizerOptions &Options, bool UseGlobalGC,
1329 bool UseOdrIndicator, AsanDtorKind DestructorKind,
1330 AsanCtorKind ConstructorKind)
1331 : Options(Options), UseGlobalGC(UseGlobalGC),
1332 UseOdrIndicator(UseOdrIndicator), DestructorKind(DestructorKind),
1333 ConstructorKind(ConstructorKind) {}
1334
1337 // Return early if nosanitize_address module flag is present for the module.
1338 // This implies that asan pass has already run before.
1339 if (checkIfAlreadyInstrumented(M, "nosanitize_address"))
1340 return PreservedAnalyses::all();
1341
1342 ModuleAddressSanitizer ModuleSanitizer(
1343 M, Options.InsertVersionCheck, Options.CompileKernel, Options.Recover,
1344 UseGlobalGC, UseOdrIndicator, DestructorKind, ConstructorKind);
1345 bool Modified = false;
1346 auto &FAM = MAM.getResult<FunctionAnalysisManagerModuleProxy>(M).getManager();
1347 const StackSafetyGlobalInfo *const SSGI =
1348 ClUseStackSafety ? &MAM.getResult<StackSafetyGlobalAnalysis>(M) : nullptr;
1349 for (Function &F : M) {
1350 if (F.empty())
1351 continue;
1352 if (F.getLinkage() == GlobalValue::AvailableExternallyLinkage)
1353 continue;
1354 if (!ClDebugFunc.empty() && ClDebugFunc == F.getName())
1355 continue;
1356 if (F.getName().starts_with("__asan_"))
1357 continue;
1358 if (F.isPresplitCoroutine())
1359 continue;
1360 AddressSanitizer FunctionSanitizer(
1361 M, SSGI, Options.InstrumentationWithCallsThreshold,
1362 Options.MaxInlinePoisoningSize, Options.CompileKernel, Options.Recover,
1363 Options.UseAfterScope, Options.UseAfterReturn);
1364 const TargetLibraryInfo &TLI = FAM.getResult<TargetLibraryAnalysis>(F);
1365 const TargetTransformInfo &TTI = FAM.getResult<TargetIRAnalysis>(F);
1366 Modified |= FunctionSanitizer.instrumentFunction(F, &TLI, &TTI);
1367 }
1368 Modified |= ModuleSanitizer.instrumentModule();
1369 if (!Modified)
1370 return PreservedAnalyses::all();
1371
1373 // GlobalsAA is considered stateless and does not get invalidated unless
1374 // explicitly invalidated; PreservedAnalyses::none() is not enough. Sanitizers
1375 // make changes that require GlobalsAA to be invalidated.
1376 PA.abandon<GlobalsAA>();
1377 return PA;
1378}
1379
1381 size_t Res = llvm::countr_zero(TypeSize / 8);
1383 return Res;
1384}
1385
1386/// Check if \p G has been created by a trusted compiler pass.
1388 // Do not instrument @llvm.global_ctors, @llvm.used, etc.
1389 if (G->getName().starts_with("llvm.") ||
1390 // Do not instrument gcov counter arrays.
1391 G->getName().starts_with("__llvm_gcov_ctr") ||
1392 // Do not instrument rtti proxy symbols for function sanitizer.
1393 G->getName().starts_with("__llvm_rtti_proxy"))
1394 return true;
1395
1396 // Do not instrument asan globals.
1397 if (G->getName().starts_with(kAsanGenPrefix) ||
1398 G->getName().starts_with(kSanCovGenPrefix) ||
1399 G->getName().starts_with(kODRGenPrefix))
1400 return true;
1401
1402 return false;
1403}
1404
1406 Type *PtrTy = cast<PointerType>(Addr->getType()->getScalarType());
1407 unsigned int AddrSpace = PtrTy->getPointerAddressSpace();
1408 // Globals in address space 1 and 4 are supported for AMDGPU.
1409 if (AddrSpace == 3 || AddrSpace == 5)
1410 return true;
1411 return false;
1412}
1413
1414static bool isSupportedAddrspace(const Triple &TargetTriple, Value *Addr) {
1415 Type *PtrTy = cast<PointerType>(Addr->getType()->getScalarType());
1416 unsigned int AddrSpace = PtrTy->getPointerAddressSpace();
1417
1418 if (!SrcAddrSpaces.empty())
1419 return SrcAddrSpaces.count(AddrSpace);
1420
1421 if (TargetTriple.isAMDGPU())
1422 return !isUnsupportedAMDGPUAddrspace(Addr);
1423
1424 return AddrSpace == 0;
1425}
1426
1427Value *AddressSanitizer::memToShadow(Value *Shadow, IRBuilder<> &IRB) {
1428 // Shadow >> scale
1429 Shadow = IRB.CreateLShr(Shadow, Mapping.Scale);
1430 if (Mapping.Offset == 0) return Shadow;
1431 // (Shadow >> scale) | offset
1432 Value *ShadowBase;
1433 if (LocalDynamicShadow)
1434 ShadowBase = LocalDynamicShadow;
1435 else
1436 ShadowBase = ConstantInt::get(IntptrTy, Mapping.Offset);
1437 if (Mapping.OrShadowOffset)
1438 return IRB.CreateOr(Shadow, ShadowBase);
1439 else
1440 return IRB.CreateAdd(Shadow, ShadowBase);
1441}
1442
1443// Instrument memset/memmove/memcpy
1444void AddressSanitizer::instrumentMemIntrinsic(MemIntrinsic *MI,
1445 RuntimeCallInserter &RTCI) {
1447 if (isa<MemTransferInst>(MI)) {
1448 RTCI.createRuntimeCall(
1449 IRB, isa<MemMoveInst>(MI) ? AsanMemmove : AsanMemcpy,
1450 {IRB.CreateAddrSpaceCast(MI->getOperand(0), PtrTy),
1451 IRB.CreateAddrSpaceCast(MI->getOperand(1), PtrTy),
1452 IRB.CreateIntCast(MI->getOperand(2), IntptrTy, false)});
1453 } else if (isa<MemSetInst>(MI)) {
1454 RTCI.createRuntimeCall(
1455 IRB, AsanMemset,
1456 {IRB.CreateAddrSpaceCast(MI->getOperand(0), PtrTy),
1457 IRB.CreateIntCast(MI->getOperand(1), IRB.getInt32Ty(), false),
1458 IRB.CreateIntCast(MI->getOperand(2), IntptrTy, false)});
1459 }
1460 MI->eraseFromParent();
1461}
1462
1463/// Check if we want (and can) handle this alloca.
1464bool AddressSanitizer::isInterestingAlloca(const AllocaInst &AI) {
1465 auto [It, Inserted] = ProcessedAllocas.try_emplace(&AI);
1466
1467 if (!Inserted)
1468 return It->getSecond();
1469
1470 bool IsInteresting =
1471 (AI.getAllocatedType()->isSized() &&
1472 // alloca() may be called with 0 size, ignore it.
1473 ((!AI.isStaticAlloca()) || !getAllocaSizeInBytes(AI).isZero()) &&
1474 // We are only interested in allocas not promotable to registers.
1475 // Promotable allocas are common under -O0.
1477 // inalloca allocas are not treated as static, and we don't want
1478 // dynamic alloca instrumentation for them as well.
1479 !AI.isUsedWithInAlloca() &&
1480 // swifterror allocas are register promoted by ISel
1481 !AI.isSwiftError() &&
1482 // safe allocas are not interesting
1483 !(SSGI && SSGI->isSafe(AI)));
1484
1485 It->second = IsInteresting;
1486 return IsInteresting;
1487}
1488
1489bool AddressSanitizer::ignoreAccess(Instruction *Inst, Value *Ptr) {
1490 // Check whether the target supports sanitizing the address space
1491 // of the pointer.
1492 if (!isSupportedAddrspace(TargetTriple, Ptr))
1493 return true;
1494
1495 // Ignore swifterror addresses.
1496 // swifterror memory addresses are mem2reg promoted by instruction
1497 // selection. As such they cannot have regular uses like an instrumentation
1498 // function and it makes no sense to track them as memory.
1499 if (Ptr->isSwiftError())
1500 return true;
1501
1502 // Treat memory accesses to promotable allocas as non-interesting since they
1503 // will not cause memory violations. This greatly speeds up the instrumented
1504 // executable at -O0.
1505 if (auto AI = dyn_cast_or_null<AllocaInst>(Ptr))
1506 if (ClSkipPromotableAllocas && !isInterestingAlloca(*AI))
1507 return true;
1508
1509 if (SSGI != nullptr && SSGI->stackAccessIsSafe(*Inst) &&
1510 findAllocaForValue(Ptr))
1511 return true;
1512
1513 return false;
1514}
1515
1516void AddressSanitizer::getInterestingMemoryOperands(
1518 const TargetTransformInfo *TTI) {
1519 // Do not instrument the load fetching the dynamic shadow address.
1520 if (LocalDynamicShadow == I)
1521 return;
1522
1523 if (LoadInst *LI = dyn_cast<LoadInst>(I)) {
1524 if (!ClInstrumentReads || ignoreAccess(I, LI->getPointerOperand()))
1525 return;
1526 Interesting.emplace_back(I, LI->getPointerOperandIndex(), false,
1527 LI->getType(), LI->getAlign());
1528 } else if (StoreInst *SI = dyn_cast<StoreInst>(I)) {
1529 if (!ClInstrumentWrites || ignoreAccess(I, SI->getPointerOperand()))
1530 return;
1531 Interesting.emplace_back(I, SI->getPointerOperandIndex(), true,
1532 SI->getValueOperand()->getType(), SI->getAlign());
1533 } else if (AtomicRMWInst *RMW = dyn_cast<AtomicRMWInst>(I)) {
1534 if (!ClInstrumentAtomics || ignoreAccess(I, RMW->getPointerOperand()))
1535 return;
1536 Interesting.emplace_back(I, RMW->getPointerOperandIndex(), true,
1537 RMW->getValOperand()->getType(), std::nullopt);
1538 } else if (AtomicCmpXchgInst *XCHG = dyn_cast<AtomicCmpXchgInst>(I)) {
1539 if (!ClInstrumentAtomics || ignoreAccess(I, XCHG->getPointerOperand()))
1540 return;
1541 Interesting.emplace_back(I, XCHG->getPointerOperandIndex(), true,
1542 XCHG->getCompareOperand()->getType(),
1543 std::nullopt);
1544 } else if (auto CI = dyn_cast<CallInst>(I)) {
1545 switch (CI->getIntrinsicID()) {
1546 case Intrinsic::masked_load:
1547 case Intrinsic::masked_store:
1548 case Intrinsic::masked_gather:
1549 case Intrinsic::masked_scatter: {
1550 bool IsWrite = CI->getType()->isVoidTy();
1551 // Masked store has an initial operand for the value.
1552 unsigned OpOffset = IsWrite ? 1 : 0;
1553 if (IsWrite ? !ClInstrumentWrites : !ClInstrumentReads)
1554 return;
1555
1556 auto BasePtr = CI->getOperand(OpOffset);
1557 if (ignoreAccess(I, BasePtr))
1558 return;
1559 Type *Ty = IsWrite ? CI->getArgOperand(0)->getType() : CI->getType();
1560 MaybeAlign Alignment = CI->getParamAlign(0);
1561 Value *Mask = CI->getOperand(1 + OpOffset);
1562 Interesting.emplace_back(I, OpOffset, IsWrite, Ty, Alignment, Mask);
1563 break;
1564 }
1565 case Intrinsic::masked_expandload:
1566 case Intrinsic::masked_compressstore: {
1567 bool IsWrite = CI->getIntrinsicID() == Intrinsic::masked_compressstore;
1568 unsigned OpOffset = IsWrite ? 1 : 0;
1569 if (IsWrite ? !ClInstrumentWrites : !ClInstrumentReads)
1570 return;
1571 auto BasePtr = CI->getOperand(OpOffset);
1572 if (ignoreAccess(I, BasePtr))
1573 return;
1574 MaybeAlign Alignment = BasePtr->getPointerAlignment(*DL);
1575 Type *Ty = IsWrite ? CI->getArgOperand(0)->getType() : CI->getType();
1576
1577 IRBuilder IB(I);
1578 Value *Mask = CI->getOperand(1 + OpOffset);
1579 // Use the popcount of Mask as the effective vector length.
1580 Type *ExtTy = VectorType::get(IntptrTy, cast<VectorType>(Ty));
1581 Value *ExtMask = IB.CreateZExt(Mask, ExtTy);
1582 Value *EVL = IB.CreateAddReduce(ExtMask);
1583 Value *TrueMask = ConstantInt::get(Mask->getType(), 1);
1584 Interesting.emplace_back(I, OpOffset, IsWrite, Ty, Alignment, TrueMask,
1585 EVL);
1586 break;
1587 }
1588 case Intrinsic::vp_load:
1589 case Intrinsic::vp_store:
1590 case Intrinsic::experimental_vp_strided_load:
1591 case Intrinsic::experimental_vp_strided_store: {
1592 auto *VPI = cast<VPIntrinsic>(CI);
1593 unsigned IID = CI->getIntrinsicID();
1594 bool IsWrite = CI->getType()->isVoidTy();
1595 if (IsWrite ? !ClInstrumentWrites : !ClInstrumentReads)
1596 return;
1597 unsigned PtrOpNo = *VPI->getMemoryPointerParamPos(IID);
1598 Type *Ty = IsWrite ? CI->getArgOperand(0)->getType() : CI->getType();
1599 MaybeAlign Alignment = VPI->getOperand(PtrOpNo)->getPointerAlignment(*DL);
1600 Value *Stride = nullptr;
1601 if (IID == Intrinsic::experimental_vp_strided_store ||
1602 IID == Intrinsic::experimental_vp_strided_load) {
1603 Stride = VPI->getOperand(PtrOpNo + 1);
1604 // Use the pointer alignment as the element alignment if the stride is a
1605 // multiple of the pointer alignment. Otherwise, the element alignment
1606 // should be Align(1).
1607 unsigned PointerAlign = Alignment.valueOrOne().value();
1608 if (!isa<ConstantInt>(Stride) ||
1609 cast<ConstantInt>(Stride)->getZExtValue() % PointerAlign != 0)
1610 Alignment = Align(1);
1611 }
1612 Interesting.emplace_back(I, PtrOpNo, IsWrite, Ty, Alignment,
1613 VPI->getMaskParam(), VPI->getVectorLengthParam(),
1614 Stride);
1615 break;
1616 }
1617 case Intrinsic::vp_gather:
1618 case Intrinsic::vp_scatter: {
1619 auto *VPI = cast<VPIntrinsic>(CI);
1620 unsigned IID = CI->getIntrinsicID();
1621 bool IsWrite = IID == Intrinsic::vp_scatter;
1622 if (IsWrite ? !ClInstrumentWrites : !ClInstrumentReads)
1623 return;
1624 unsigned PtrOpNo = *VPI->getMemoryPointerParamPos(IID);
1625 Type *Ty = IsWrite ? CI->getArgOperand(0)->getType() : CI->getType();
1626 MaybeAlign Alignment = VPI->getPointerAlignment();
1627 Interesting.emplace_back(I, PtrOpNo, IsWrite, Ty, Alignment,
1628 VPI->getMaskParam(),
1629 VPI->getVectorLengthParam());
1630 break;
1631 }
1632 default:
1633 if (auto *II = dyn_cast<IntrinsicInst>(I)) {
1634 MemIntrinsicInfo IntrInfo;
1635 if (TTI->getTgtMemIntrinsic(II, IntrInfo))
1636 Interesting = IntrInfo.InterestingOperands;
1637 return;
1638 }
1639 for (unsigned ArgNo = 0; ArgNo < CI->arg_size(); ArgNo++) {
1640 if (!ClInstrumentByval || !CI->isByValArgument(ArgNo) ||
1641 ignoreAccess(I, CI->getArgOperand(ArgNo)))
1642 continue;
1643 Type *Ty = CI->getParamByValType(ArgNo);
1644 Interesting.emplace_back(I, ArgNo, false, Ty, Align(1));
1645 }
1646 }
1647 }
1648}
1649
1650static bool isPointerOperand(Value *V) {
1651 return V->getType()->isPointerTy() || isa<PtrToIntInst>(V);
1652}
1653
1654// This is a rough heuristic; it may cause both false positives and
1655// false negatives. The proper implementation requires cooperation with
1656// the frontend.
1658 if (ICmpInst *Cmp = dyn_cast<ICmpInst>(I)) {
1659 if (!Cmp->isRelational())
1660 return false;
1661 } else {
1662 return false;
1663 }
1664 return isPointerOperand(I->getOperand(0)) &&
1665 isPointerOperand(I->getOperand(1));
1666}
1667
1668// This is a rough heuristic; it may cause both false positives and
1669// false negatives. The proper implementation requires cooperation with
1670// the frontend.
1673 if (BO->getOpcode() != Instruction::Sub)
1674 return false;
1675 } else {
1676 return false;
1677 }
1678 return isPointerOperand(I->getOperand(0)) &&
1679 isPointerOperand(I->getOperand(1));
1680}
1681
1682bool AddressSanitizer::GlobalIsLinkerInitialized(GlobalVariable *G) {
1683 // If a global variable does not have dynamic initialization we don't
1684 // have to instrument it. However, if a global does not have initializer
1685 // at all, we assume it has dynamic initializer (in other TU).
1686 if (!G->hasInitializer())
1687 return false;
1688
1689 if (G->hasSanitizerMetadata() && G->getSanitizerMetadata().IsDynInit)
1690 return false;
1691
1692 return true;
1693}
1694
1695void AddressSanitizer::instrumentPointerComparisonOrSubtraction(
1696 Instruction *I, RuntimeCallInserter &RTCI) {
1697 IRBuilder<> IRB(I);
1698 FunctionCallee F = isa<ICmpInst>(I) ? AsanPtrCmpFunction : AsanPtrSubFunction;
1699 Value *Param[2] = {I->getOperand(0), I->getOperand(1)};
1700 for (Value *&i : Param) {
1701 if (i->getType()->isPointerTy())
1702 i = IRB.CreatePointerCast(i, IntptrTy);
1703 }
1704 RTCI.createRuntimeCall(IRB, F, Param);
1705}
1706
1707static void doInstrumentAddress(AddressSanitizer *Pass, Instruction *I,
1708 Instruction *InsertBefore, Value *Addr,
1709 MaybeAlign Alignment, unsigned Granularity,
1710 TypeSize TypeStoreSize, bool IsWrite,
1711 Value *SizeArgument, bool UseCalls,
1712 uint32_t Exp, RuntimeCallInserter &RTCI) {
1713 // Instrument a 1-, 2-, 4-, 8-, or 16- byte access with one check
1714 // if the data is properly aligned.
1715 if (!TypeStoreSize.isScalable()) {
1716 const auto FixedSize = TypeStoreSize.getFixedValue();
1717 switch (FixedSize) {
1718 case 8:
1719 case 16:
1720 case 32:
1721 case 64:
1722 case 128:
1723 if (!Alignment || *Alignment >= Granularity ||
1724 *Alignment >= FixedSize / 8)
1725 return Pass->instrumentAddress(I, InsertBefore, Addr, Alignment,
1726 FixedSize, IsWrite, nullptr, UseCalls,
1727 Exp, RTCI);
1728 }
1729 }
1730 Pass->instrumentUnusualSizeOrAlignment(I, InsertBefore, Addr, TypeStoreSize,
1731 IsWrite, nullptr, UseCalls, Exp, RTCI);
1732}
1733
1734void AddressSanitizer::instrumentMaskedLoadOrStore(
1735 AddressSanitizer *Pass, const DataLayout &DL, Type *IntptrTy, Value *Mask,
1736 Value *EVL, Value *Stride, Instruction *I, Value *Addr,
1737 MaybeAlign Alignment, unsigned Granularity, Type *OpType, bool IsWrite,
1738 Value *SizeArgument, bool UseCalls, uint32_t Exp,
1739 RuntimeCallInserter &RTCI) {
1740 auto *VTy = cast<VectorType>(OpType);
1741 TypeSize ElemTypeSize = DL.getTypeStoreSizeInBits(VTy->getScalarType());
1742 auto Zero = ConstantInt::get(IntptrTy, 0);
1743
1744 IRBuilder IB(I);
1745 Instruction *LoopInsertBefore = I;
1746 if (EVL) {
1747 // The end argument of SplitBlockAndInsertForLane is assumed bigger
1748 // than zero, so we should check whether EVL is zero here.
1749 Type *EVLType = EVL->getType();
1750 Value *IsEVLZero = IB.CreateICmpNE(EVL, ConstantInt::get(EVLType, 0));
1751 LoopInsertBefore = SplitBlockAndInsertIfThen(IsEVLZero, I, false);
1752 IB.SetInsertPoint(LoopInsertBefore);
1753 // Cast EVL to IntptrTy.
1754 EVL = IB.CreateZExtOrTrunc(EVL, IntptrTy);
1755 // To avoid undefined behavior for extracting with out of range index, use
1756 // the minimum of evl and element count as trip count.
1757 Value *EC = IB.CreateElementCount(IntptrTy, VTy->getElementCount());
1758 EVL = IB.CreateBinaryIntrinsic(Intrinsic::umin, EVL, EC);
1759 } else {
1760 EVL = IB.CreateElementCount(IntptrTy, VTy->getElementCount());
1761 }
1762
1763 // Cast Stride to IntptrTy.
1764 if (Stride)
1765 Stride = IB.CreateZExtOrTrunc(Stride, IntptrTy);
1766
1767 SplitBlockAndInsertForEachLane(EVL, LoopInsertBefore->getIterator(),
1768 [&](IRBuilderBase &IRB, Value *Index) {
1769 Value *MaskElem = IRB.CreateExtractElement(Mask, Index);
1770 if (auto *MaskElemC = dyn_cast<ConstantInt>(MaskElem)) {
1771 if (MaskElemC->isZero())
1772 // No check
1773 return;
1774 // Unconditional check
1775 } else {
1776 // Conditional check
1777 Instruction *ThenTerm = SplitBlockAndInsertIfThen(
1778 MaskElem, &*IRB.GetInsertPoint(), false);
1779 IRB.SetInsertPoint(ThenTerm);
1780 }
1781
1782 Value *InstrumentedAddress;
1783 if (isa<VectorType>(Addr->getType())) {
1784 assert(
1785 cast<VectorType>(Addr->getType())->getElementType()->isPointerTy() &&
1786 "Expected vector of pointer.");
1787 InstrumentedAddress = IRB.CreateExtractElement(Addr, Index);
1788 } else if (Stride) {
1789 Index = IRB.CreateMul(Index, Stride);
1790 InstrumentedAddress = IRB.CreatePtrAdd(Addr, Index);
1791 } else {
1792 InstrumentedAddress = IRB.CreateGEP(VTy, Addr, {Zero, Index});
1793 }
1794 doInstrumentAddress(Pass, I, &*IRB.GetInsertPoint(), InstrumentedAddress,
1795 Alignment, Granularity, ElemTypeSize, IsWrite,
1796 SizeArgument, UseCalls, Exp, RTCI);
1797 });
1798}
1799
1800void AddressSanitizer::instrumentMop(ObjectSizeOffsetVisitor &ObjSizeVis,
1801 InterestingMemoryOperand &O, bool UseCalls,
1802 const DataLayout &DL,
1803 RuntimeCallInserter &RTCI) {
1804 Value *Addr = O.getPtr();
1805
1806 // Optimization experiments.
1807 // The experiments can be used to evaluate potential optimizations that remove
1808 // instrumentation (assess false negatives). Instead of completely removing
1809 // some instrumentation, you set Exp to a non-zero value (mask of optimization
1810 // experiments that want to remove instrumentation of this instruction).
1811 // If Exp is non-zero, this pass will emit special calls into runtime
1812 // (e.g. __asan_report_exp_load1 instead of __asan_report_load1). These calls
1813 // make runtime terminate the program in a special way (with a different
1814 // exit status). Then you run the new compiler on a buggy corpus, collect
1815 // the special terminations (ideally, you don't see them at all -- no false
1816 // negatives) and make the decision on the optimization.
1817 uint32_t Exp = ClForceExperiment;
1818
1819 if (ClOpt && ClOptGlobals) {
1820 // If initialization order checking is disabled, a simple access to a
1821 // dynamically initialized global is always valid.
1823 if (G && (!ClInitializers || GlobalIsLinkerInitialized(G)) &&
1824 isSafeAccess(ObjSizeVis, Addr, O.TypeStoreSize)) {
1825 NumOptimizedAccessesToGlobalVar++;
1826 return;
1827 }
1828 }
1829
1830 if (ClOpt && ClOptStack) {
1831 // A direct inbounds access to a stack variable is always valid.
1833 isSafeAccess(ObjSizeVis, Addr, O.TypeStoreSize)) {
1834 NumOptimizedAccessesToStackVar++;
1835 return;
1836 }
1837 }
1838
1839 if (O.IsWrite)
1840 NumInstrumentedWrites++;
1841 else
1842 NumInstrumentedReads++;
1843
1844 if (O.MaybeByteOffset) {
1845 Type *Ty = Type::getInt8Ty(*C);
1846 IRBuilder IB(O.getInsn());
1847
1848 Value *OffsetOp = O.MaybeByteOffset;
1849 if (TargetTriple.isRISCV()) {
1850 Type *OffsetTy = OffsetOp->getType();
1851 // RVV indexed loads/stores zero-extend offset operands which are narrower
1852 // than XLEN to XLEN.
1853 if (OffsetTy->getScalarType()->getIntegerBitWidth() <
1854 static_cast<unsigned>(LongSize)) {
1855 VectorType *OrigType = cast<VectorType>(OffsetTy);
1856 Type *ExtendTy = VectorType::get(IntptrTy, OrigType);
1857 OffsetOp = IB.CreateZExt(OffsetOp, ExtendTy);
1858 }
1859 }
1860 Addr = IB.CreateGEP(Ty, Addr, {OffsetOp});
1861 }
1862
1863 unsigned Granularity = 1 << Mapping.Scale;
1864 if (O.MaybeMask) {
1865 instrumentMaskedLoadOrStore(this, DL, IntptrTy, O.MaybeMask, O.MaybeEVL,
1866 O.MaybeStride, O.getInsn(), Addr, O.Alignment,
1867 Granularity, O.OpType, O.IsWrite, nullptr,
1868 UseCalls, Exp, RTCI);
1869 } else {
1870 doInstrumentAddress(this, O.getInsn(), O.getInsn(), Addr, O.Alignment,
1871 Granularity, O.TypeStoreSize, O.IsWrite, nullptr,
1872 UseCalls, Exp, RTCI);
1873 }
1874}
1875
1876Instruction *AddressSanitizer::generateCrashCode(Instruction *InsertBefore,
1877 Value *Addr, bool IsWrite,
1878 size_t AccessSizeIndex,
1879 Value *SizeArgument,
1880 uint32_t Exp,
1881 RuntimeCallInserter &RTCI) {
1882 InstrumentationIRBuilder IRB(InsertBefore);
1883 Value *ExpVal = Exp == 0 ? nullptr : ConstantInt::get(IRB.getInt32Ty(), Exp);
1884 CallInst *Call = nullptr;
1885 if (SizeArgument) {
1886 if (Exp == 0)
1887 Call = RTCI.createRuntimeCall(IRB, AsanErrorCallbackSized[IsWrite][0],
1888 {Addr, SizeArgument});
1889 else
1890 Call = RTCI.createRuntimeCall(IRB, AsanErrorCallbackSized[IsWrite][1],
1891 {Addr, SizeArgument, ExpVal});
1892 } else {
1893 if (Exp == 0)
1894 Call = RTCI.createRuntimeCall(
1895 IRB, AsanErrorCallback[IsWrite][0][AccessSizeIndex], Addr);
1896 else
1897 Call = RTCI.createRuntimeCall(
1898 IRB, AsanErrorCallback[IsWrite][1][AccessSizeIndex], {Addr, ExpVal});
1899 }
1900
1902 return Call;
1903}
1904
1905Value *AddressSanitizer::createSlowPathCmp(IRBuilder<> &IRB, Value *AddrLong,
1906 Value *ShadowValue,
1907 uint32_t TypeStoreSize) {
1908 size_t Granularity = static_cast<size_t>(1) << Mapping.Scale;
1909 // Addr & (Granularity - 1)
1910 Value *LastAccessedByte =
1911 IRB.CreateAnd(AddrLong, ConstantInt::get(IntptrTy, Granularity - 1));
1912 // (Addr & (Granularity - 1)) + size - 1
1913 if (TypeStoreSize / 8 > 1)
1914 LastAccessedByte = IRB.CreateAdd(
1915 LastAccessedByte, ConstantInt::get(IntptrTy, TypeStoreSize / 8 - 1));
1916 // (uint8_t) ((Addr & (Granularity-1)) + size - 1)
1917 LastAccessedByte =
1918 IRB.CreateIntCast(LastAccessedByte, ShadowValue->getType(), false);
1919 // ((uint8_t) ((Addr & (Granularity-1)) + size - 1)) >= ShadowValue
1920 return IRB.CreateICmpSGE(LastAccessedByte, ShadowValue);
1921}
1922
1923Instruction *AddressSanitizer::instrumentAMDGPUAddress(
1924 Instruction *OrigIns, Instruction *InsertBefore, Value *Addr,
1925 uint32_t TypeStoreSize, bool IsWrite, Value *SizeArgument) {
1926 // Do not instrument unsupported addrspaces.
1928 return nullptr;
1929 Type *PtrTy = cast<PointerType>(Addr->getType()->getScalarType());
1930 // Follow host instrumentation for global and constant addresses.
1931 if (PtrTy->getPointerAddressSpace() != 0)
1932 return InsertBefore;
1933 // Instrument generic addresses in supported addressspaces.
1934 IRBuilder<> IRB(InsertBefore);
1935 Value *IsShared = IRB.CreateCall(AMDGPUAddressShared, {Addr});
1936 Value *IsPrivate = IRB.CreateCall(AMDGPUAddressPrivate, {Addr});
1937 Value *IsSharedOrPrivate = IRB.CreateOr(IsShared, IsPrivate);
1938 Value *Cmp = IRB.CreateNot(IsSharedOrPrivate);
1939 Value *AddrSpaceZeroLanding =
1940 SplitBlockAndInsertIfThen(Cmp, InsertBefore, false);
1941 InsertBefore = cast<Instruction>(AddrSpaceZeroLanding);
1942 return InsertBefore;
1943}
1944
1945Instruction *AddressSanitizer::genAMDGPUReportBlock(IRBuilder<> &IRB,
1946 Value *Cond, bool Recover) {
1947 Value *ReportCond = Cond;
1948 if (!Recover) {
1949 auto Ballot = Inserter.insertFunction(kAMDGPUBallotName, IRB.getInt64Ty(),
1950 IRB.getInt1Ty());
1951 ReportCond = IRB.CreateIsNotNull(IRB.CreateCall(Ballot, {Cond}));
1952 }
1953
1954 auto *Trm =
1955 SplitBlockAndInsertIfThen(ReportCond, &*IRB.GetInsertPoint(), false,
1957 Trm->getParent()->setName("asan.report");
1958
1959 if (Recover)
1960 return Trm;
1961
1962 Trm = SplitBlockAndInsertIfThen(Cond, Trm, false);
1963 IRB.SetInsertPoint(Trm);
1964 return IRB.CreateCall(
1965 Inserter.insertFunction(kAMDGPUUnreachableName, IRB.getVoidTy()), {});
1966}
1967
1968void AddressSanitizer::instrumentAddress(Instruction *OrigIns,
1969 Instruction *InsertBefore, Value *Addr,
1970 MaybeAlign Alignment,
1971 uint32_t TypeStoreSize, bool IsWrite,
1972 Value *SizeArgument, bool UseCalls,
1973 uint32_t Exp,
1974 RuntimeCallInserter &RTCI) {
1975 if (TargetTriple.isAMDGPU()) {
1976 InsertBefore = instrumentAMDGPUAddress(OrigIns, InsertBefore, Addr,
1977 TypeStoreSize, IsWrite, SizeArgument);
1978 if (!InsertBefore)
1979 return;
1980 }
1981
1982 InstrumentationIRBuilder IRB(InsertBefore);
1983 size_t AccessSizeIndex = TypeStoreSizeToSizeIndex(TypeStoreSize);
1984
1985 if (UseCalls && ClOptimizeCallbacks) {
1986 const ASanAccessInfo AccessInfo(IsWrite, CompileKernel, AccessSizeIndex);
1987 IRB.CreateIntrinsic(Intrinsic::asan_check_memaccess, {},
1988 {IRB.CreatePointerCast(Addr, PtrTy),
1989 ConstantInt::get(Int32Ty, AccessInfo.Packed)});
1990 return;
1991 }
1992
1993 Value *AddrLong = IRB.CreatePointerCast(Addr, IntptrTy);
1994 if (UseCalls) {
1995 if (Exp == 0)
1996 RTCI.createRuntimeCall(
1997 IRB, AsanMemoryAccessCallback[IsWrite][0][AccessSizeIndex], AddrLong);
1998 else
1999 RTCI.createRuntimeCall(
2000 IRB, AsanMemoryAccessCallback[IsWrite][1][AccessSizeIndex],
2001 {AddrLong, ConstantInt::get(IRB.getInt32Ty(), Exp)});
2002 return;
2003 }
2004
2005 Type *ShadowTy =
2006 IntegerType::get(*C, std::max(8U, TypeStoreSize >> Mapping.Scale));
2007 Type *ShadowPtrTy = PointerType::get(*C, ClShadowAddrSpace);
2008 Value *ShadowPtr = memToShadow(AddrLong, IRB);
2009 const uint64_t ShadowAlign =
2010 std::max<uint64_t>(Alignment.valueOrOne().value() >> Mapping.Scale, 1);
2011 Value *ShadowValue = IRB.CreateAlignedLoad(
2012 ShadowTy, IRB.CreateIntToPtr(ShadowPtr, ShadowPtrTy), Align(ShadowAlign));
2013
2014 Value *Cmp = IRB.CreateIsNotNull(ShadowValue);
2015 size_t Granularity = 1ULL << Mapping.Scale;
2016 Instruction *CrashTerm = nullptr;
2017
2018 bool GenSlowPath = (ClAlwaysSlowPath || (TypeStoreSize < 8 * Granularity));
2019
2020 if (TargetTriple.isAMDGCN()) {
2021 if (GenSlowPath) {
2022 auto *Cmp2 = createSlowPathCmp(IRB, AddrLong, ShadowValue, TypeStoreSize);
2023 Cmp = IRB.CreateAnd(Cmp, Cmp2);
2024 }
2025 CrashTerm = genAMDGPUReportBlock(IRB, Cmp, Recover);
2026 } else if (GenSlowPath) {
2027 // We use branch weights for the slow path check, to indicate that the slow
2028 // path is rarely taken. This seems to be the case for SPEC benchmarks.
2030 Cmp, InsertBefore, false, MDBuilder(*C).createUnlikelyBranchWeights());
2031 BasicBlock *NextBB = cast<UncondBrInst>(CheckTerm)->getSuccessor();
2032 IRB.SetInsertPoint(CheckTerm);
2033 Value *Cmp2 = createSlowPathCmp(IRB, AddrLong, ShadowValue, TypeStoreSize);
2034 if (Recover) {
2035 CrashTerm = SplitBlockAndInsertIfThen(Cmp2, CheckTerm, false);
2036 } else {
2037 BasicBlock *CrashBlock =
2038 BasicBlock::Create(*C, "", NextBB->getParent(), NextBB);
2039 CrashTerm = new UnreachableInst(*C, CrashBlock);
2040 CondBrInst *NewTerm = CondBrInst::Create(Cmp2, CrashBlock, NextBB);
2041 ReplaceInstWithInst(CheckTerm, NewTerm);
2042 }
2043 } else {
2044 CrashTerm = SplitBlockAndInsertIfThen(Cmp, InsertBefore, !Recover);
2045 }
2046
2047 Instruction *Crash = generateCrashCode(
2048 CrashTerm, AddrLong, IsWrite, AccessSizeIndex, SizeArgument, Exp, RTCI);
2049 if (OrigIns->getDebugLoc())
2050 Crash->setDebugLoc(OrigIns->getDebugLoc());
2051}
2052
2053// Instrument unusual size or unusual alignment.
2054// We can not do it with a single check, so we do 1-byte check for the first
2055// and the last bytes. We call __asan_report_*_n(addr, real_size) to be able
2056// to report the actual access size.
2057void AddressSanitizer::instrumentUnusualSizeOrAlignment(
2058 Instruction *I, Instruction *InsertBefore, Value *Addr,
2059 TypeSize TypeStoreSize, bool IsWrite, Value *SizeArgument, bool UseCalls,
2060 uint32_t Exp, RuntimeCallInserter &RTCI) {
2061 InstrumentationIRBuilder IRB(InsertBefore);
2062 Value *NumBits = IRB.CreateTypeSize(IntptrTy, TypeStoreSize);
2063 Value *Size = IRB.CreateLShr(NumBits, ConstantInt::get(IntptrTy, 3));
2064
2065 Value *AddrLong = IRB.CreatePointerCast(Addr, IntptrTy);
2066 if (UseCalls) {
2067 if (Exp == 0)
2068 RTCI.createRuntimeCall(IRB, AsanMemoryAccessCallbackSized[IsWrite][0],
2069 {AddrLong, Size});
2070 else
2071 RTCI.createRuntimeCall(
2072 IRB, AsanMemoryAccessCallbackSized[IsWrite][1],
2073 {AddrLong, Size, ConstantInt::get(IRB.getInt32Ty(), Exp)});
2074 } else {
2075 Value *SizeMinusOne = IRB.CreateSub(Size, ConstantInt::get(IntptrTy, 1));
2076 Value *LastByte = IRB.CreateIntToPtr(
2077 IRB.CreateAdd(AddrLong, SizeMinusOne),
2078 Addr->getType());
2079 instrumentAddress(I, InsertBefore, Addr, {}, 8, IsWrite, Size, false, Exp,
2080 RTCI);
2081 instrumentAddress(I, InsertBefore, LastByte, {}, 8, IsWrite, Size, false,
2082 Exp, RTCI);
2083 }
2084}
2085
2086void ModuleAddressSanitizer::poisonOneInitializer(Function &GlobalInit) {
2087 // Set up the arguments to our poison/unpoison functions.
2088 IRBuilder<> IRB(&GlobalInit.front(),
2089 GlobalInit.front().getFirstInsertionPt());
2090
2091 // Add a call to poison all external globals before the given function starts.
2092 Value *ModuleNameAddr =
2093 ConstantExpr::getPointerCast(getOrCreateModuleName(), IntptrTy);
2094 IRB.CreateCall(AsanPoisonGlobals, ModuleNameAddr);
2095
2096 // Add calls to unpoison all globals before each return instruction.
2097 for (auto &BB : GlobalInit)
2099 CallInst::Create(AsanUnpoisonGlobals, "", RI->getIterator());
2100}
2101
2102void ModuleAddressSanitizer::createInitializerPoisonCalls() {
2103 GlobalVariable *GV = M.getGlobalVariable("llvm.global_ctors");
2104 if (!GV)
2105 return;
2106
2108 if (!CA)
2109 return;
2110
2111 for (Use &OP : CA->operands()) {
2112 if (isa<ConstantAggregateZero>(OP)) continue;
2114
2115 // Must have a function or null ptr.
2116 if (Function *F = dyn_cast<Function>(CS->getOperand(1))) {
2117 if (F->getName() == kAsanModuleCtorName) continue;
2118 auto *Priority = cast<ConstantInt>(CS->getOperand(0));
2119 // Don't instrument CTORs that will run before asan.module_ctor.
2120 if (Priority->getLimitedValue() <= GetCtorAndDtorPriority(TargetTriple))
2121 continue;
2122 poisonOneInitializer(*F);
2123 }
2124 }
2125}
2126
2127const GlobalVariable *
2128ModuleAddressSanitizer::getExcludedAliasedGlobal(const GlobalAlias &GA) const {
2129 // In case this function should be expanded to include rules that do not just
2130 // apply when CompileKernel is true, either guard all existing rules with an
2131 // 'if (CompileKernel) { ... }' or be absolutely sure that all these rules
2132 // should also apply to user space.
2133 assert(CompileKernel && "Only expecting to be called when compiling kernel");
2134
2135 const Constant *C = GA.getAliasee();
2136
2137 // When compiling the kernel, globals that are aliased by symbols prefixed
2138 // by "__" are special and cannot be padded with a redzone.
2139 if (GA.getName().starts_with("__"))
2140 return dyn_cast<GlobalVariable>(C->stripPointerCastsAndAliases());
2141
2142 return nullptr;
2143}
2144
2145bool ModuleAddressSanitizer::shouldInstrumentGlobal(GlobalVariable *G) const {
2146 Type *Ty = G->getValueType();
2147 LLVM_DEBUG(dbgs() << "GLOBAL: " << *G << "\n");
2148
2149 if (G->hasSanitizerMetadata() && G->getSanitizerMetadata().NoAddress)
2150 return false;
2151 if (!Ty->isSized()) return false;
2152 if (!G->hasInitializer()) return false;
2153 if (!isSupportedAddrspace(TargetTriple, G))
2154 return false;
2155 if (GlobalWasGeneratedByCompiler(G)) return false; // Our own globals.
2156 // Two problems with thread-locals:
2157 // - The address of the main thread's copy can't be computed at link-time.
2158 // - Need to poison all copies, not just the main thread's one.
2159 if (G->isThreadLocal()) return false;
2160 // For now, just ignore this Global if the alignment is large.
2161 if (G->getAlign() && *G->getAlign() > getMinRedzoneSizeForGlobal()) return false;
2162
2163 // For non-COFF targets, only instrument globals known to be defined by this
2164 // TU.
2165 // FIXME: We can instrument comdat globals on ELF if we are using the
2166 // GC-friendly metadata scheme.
2167 if (!TargetTriple.isOSBinFormatCOFF()) {
2168 if (!G->hasExactDefinition() || G->hasComdat())
2169 return false;
2170 } else {
2171 // On COFF, don't instrument non-ODR linkages.
2172 if (G->isInterposable())
2173 return false;
2174 // If the global has AvailableExternally linkage, then it is not in this
2175 // module, which means it does not need to be instrumented.
2176 if (G->hasAvailableExternallyLinkage())
2177 return false;
2178 }
2179
2180 // If a comdat is present, it must have a selection kind that implies ODR
2181 // semantics: no duplicates, any, or exact match.
2182 if (Comdat *C = G->getComdat()) {
2183 switch (C->getSelectionKind()) {
2184 case Comdat::Any:
2185 case Comdat::ExactMatch:
2187 break;
2188 case Comdat::Largest:
2189 case Comdat::SameSize:
2190 return false;
2191 }
2192 }
2193
2194 if (G->hasSection()) {
2195 // The kernel uses explicit sections for mostly special global variables
2196 // that we should not instrument. E.g. the kernel may rely on their layout
2197 // without redzones, or remove them at link time ("discard.*"), etc.
2198 if (CompileKernel)
2199 return false;
2200
2201 StringRef Section = G->getSection();
2202
2203 // Globals from llvm.metadata aren't emitted, do not instrument them.
2204 if (Section == "llvm.metadata") return false;
2205 // Do not instrument globals from special LLVM sections.
2206 if (Section.contains("__llvm") || Section.contains("__LLVM"))
2207 return false;
2208
2209 // Do not instrument function pointers to initialization and termination
2210 // routines: dynamic linker will not properly handle redzones.
2211 if (Section.starts_with(".preinit_array") ||
2212 Section.starts_with(".init_array") ||
2213 Section.starts_with(".fini_array")) {
2214 return false;
2215 }
2216
2217 // Do not instrument user-defined sections (with names resembling
2218 // valid C identifiers)
2219 if (TargetTriple.isOSBinFormatELF()) {
2220 if (llvm::all_of(Section,
2221 [](char c) { return llvm::isAlnum(c) || c == '_'; }))
2222 return false;
2223 }
2224
2225 // On COFF, if the section name contains '$', it is highly likely that the
2226 // user is using section sorting to create an array of globals similar to
2227 // the way initialization callbacks are registered in .init_array and
2228 // .CRT$XCU. The ATL also registers things in .ATL$__[azm]. Adding redzones
2229 // to such globals is counterproductive, because the intent is that they
2230 // will form an array, and out-of-bounds accesses are expected.
2231 // See https://github.com/google/sanitizers/issues/305
2232 // and http://msdn.microsoft.com/en-US/en-en/library/bb918180(v=vs.120).aspx
2233 if (TargetTriple.isOSBinFormatCOFF() && Section.contains('$')) {
2234 LLVM_DEBUG(dbgs() << "Ignoring global in sorted section (contains '$'): "
2235 << *G << "\n");
2236 return false;
2237 }
2238
2239 if (TargetTriple.isOSBinFormatMachO()) {
2240 StringRef ParsedSegment, ParsedSection;
2241 unsigned TAA = 0, StubSize = 0;
2242 bool TAAParsed;
2244 Section, ParsedSegment, ParsedSection, TAA, TAAParsed, StubSize));
2245
2246 // Ignore the globals from the __OBJC section. The ObjC runtime assumes
2247 // those conform to /usr/lib/objc/runtime.h, so we can't add redzones to
2248 // them.
2249 if (ParsedSegment == "__OBJC" ||
2250 (ParsedSegment == "__DATA" && ParsedSection.starts_with("__objc_"))) {
2251 LLVM_DEBUG(dbgs() << "Ignoring ObjC runtime global: " << *G << "\n");
2252 return false;
2253 }
2254 // See https://github.com/google/sanitizers/issues/32
2255 // Constant CFString instances are compiled in the following way:
2256 // -- the string buffer is emitted into
2257 // __TEXT,__cstring,cstring_literals
2258 // -- the constant NSConstantString structure referencing that buffer
2259 // is placed into __DATA,__cfstring
2260 // Therefore there's no point in placing redzones into __DATA,__cfstring.
2261 // Moreover, it causes the linker to crash on OS X 10.7
2262 if (ParsedSegment == "__DATA" && ParsedSection == "__cfstring") {
2263 LLVM_DEBUG(dbgs() << "Ignoring CFString: " << *G << "\n");
2264 return false;
2265 }
2266 // The linker merges the contents of cstring_literals and removes the
2267 // trailing zeroes.
2268 if (ParsedSegment == "__TEXT" && (TAA & MachO::S_CSTRING_LITERALS)) {
2269 LLVM_DEBUG(dbgs() << "Ignoring a cstring literal: " << *G << "\n");
2270 return false;
2271 }
2272 }
2273 }
2274
2275 if (CompileKernel) {
2276 // Globals that prefixed by "__" are special and cannot be padded with a
2277 // redzone.
2278 if (G->getName().starts_with("__"))
2279 return false;
2280 }
2281
2282 return true;
2283}
2284
2285// On Mach-O platforms, we emit global metadata in a separate section of the
2286// binary in order to allow the linker to properly dead strip. This is only
2287// supported on recent versions of ld64.
2288bool ModuleAddressSanitizer::ShouldUseMachOGlobalsSection() const {
2289 if (!TargetTriple.isOSBinFormatMachO())
2290 return false;
2291
2292 if (TargetTriple.isMacOSX() && !TargetTriple.isMacOSXVersionLT(10, 11))
2293 return true;
2294 if (TargetTriple.isiOS() /* or tvOS */ && !TargetTriple.isOSVersionLT(9))
2295 return true;
2296 if (TargetTriple.isWatchOS() && !TargetTriple.isOSVersionLT(2))
2297 return true;
2298 if (TargetTriple.isDriverKit())
2299 return true;
2300 if (TargetTriple.isXROS())
2301 return true;
2302
2303 return false;
2304}
2305
2306StringRef ModuleAddressSanitizer::getGlobalMetadataSection() const {
2307 switch (TargetTriple.getObjectFormat()) {
2308 case Triple::COFF: return ".ASAN$GL";
2309 case Triple::ELF: return "asan_globals";
2310 case Triple::MachO: return "__DATA,__asan_globals,regular";
2311 case Triple::Wasm:
2312 case Triple::GOFF:
2313 case Triple::SPIRV:
2314 case Triple::XCOFF:
2317 "ModuleAddressSanitizer not implemented for object file format");
2319 break;
2320 }
2321 llvm_unreachable("unsupported object format");
2322}
2323
2324void ModuleAddressSanitizer::initializeCallbacks() {
2325 IRBuilder<> IRB(*C);
2326
2327 // Declare our poisoning and unpoisoning functions.
2328 AsanPoisonGlobals = Inserter.insertFunction(kAsanPoisonGlobalsName,
2329 IRB.getVoidTy(), IntptrTy);
2330 AsanUnpoisonGlobals =
2331 Inserter.insertFunction(kAsanUnpoisonGlobalsName, IRB.getVoidTy());
2332
2333 // Declare functions that register/unregister globals.
2334 AsanRegisterGlobals = Inserter.insertFunction(
2335 kAsanRegisterGlobalsName, IRB.getVoidTy(), IntptrTy, IntptrTy);
2336 AsanUnregisterGlobals = Inserter.insertFunction(
2337 kAsanUnregisterGlobalsName, IRB.getVoidTy(), IntptrTy, IntptrTy);
2338
2339 // Declare the functions that find globals in a shared object and then invoke
2340 // the (un)register function on them.
2341 AsanRegisterImageGlobals = Inserter.insertFunction(
2342 kAsanRegisterImageGlobalsName, IRB.getVoidTy(), IntptrTy);
2343 AsanUnregisterImageGlobals = Inserter.insertFunction(
2345
2346 AsanRegisterElfGlobals =
2347 Inserter.insertFunction(kAsanRegisterElfGlobalsName, IRB.getVoidTy(),
2348 IntptrTy, IntptrTy, IntptrTy);
2349 AsanUnregisterElfGlobals =
2350 Inserter.insertFunction(kAsanUnregisterElfGlobalsName, IRB.getVoidTy(),
2351 IntptrTy, IntptrTy, IntptrTy);
2352}
2353
2354// Put the metadata and the instrumented global in the same group. This ensures
2355// that the metadata is discarded if the instrumented global is discarded.
2356void ModuleAddressSanitizer::SetComdatForGlobalMetadata(
2357 GlobalVariable *G, GlobalVariable *Metadata, StringRef InternalSuffix) {
2358 Module &M = *G->getParent();
2359 Comdat *C = G->getComdat();
2360 if (!C) {
2361 if (!G->hasName()) {
2362 // If G is unnamed, it must be internal. Give it an artificial name
2363 // so we can put it in a comdat.
2364 assert(G->hasLocalLinkage());
2365 G->setName(genName("anon_global"));
2366 }
2367
2368 if (!InternalSuffix.empty() && G->hasLocalLinkage()) {
2369 std::string Name = std::string(G->getName());
2370 Name += InternalSuffix;
2371 C = M.getOrInsertComdat(Name);
2372 } else {
2373 C = M.getOrInsertComdat(G->getName());
2374 }
2375
2376 // Make this IMAGE_COMDAT_SELECT_NODUPLICATES on COFF. Also upgrade private
2377 // linkage to internal linkage so that a symbol table entry is emitted. This
2378 // is necessary in order to create the comdat group.
2379 if (TargetTriple.isOSBinFormatCOFF()) {
2380 C->setSelectionKind(Comdat::NoDeduplicate);
2381 if (G->hasPrivateLinkage())
2382 G->setLinkage(GlobalValue::InternalLinkage);
2383 }
2384 G->setComdat(C);
2385 }
2386
2387 assert(G->hasComdat());
2388 Metadata->setComdat(G->getComdat());
2389}
2390
2391// Create a separate metadata global and put it in the appropriate ASan
2392// global registration section.
2394ModuleAddressSanitizer::CreateMetadataGlobal(Constant *Initializer,
2395 StringRef OriginalName) {
2396 auto Linkage = TargetTriple.isOSBinFormatMachO()
2400 M, Initializer->getType(), false, Linkage, Initializer,
2401 Twine("__asan_global_") + GlobalValue::dropLLVMManglingEscape(OriginalName));
2402 Metadata->setSection(getGlobalMetadataSection());
2403 // Place metadata in a large section for x86-64 ELF binaries to mitigate
2404 // relocation pressure.
2406 return Metadata;
2407}
2408
2409Instruction *ModuleAddressSanitizer::CreateAsanModuleDtor() {
2410 AsanDtorFunction = Function::createWithDefaultAttr(
2413 AsanDtorFunction->addFnAttr(Attribute::NoUnwind);
2414 // Ensure Dtor cannot be discarded, even if in a comdat.
2415 appendToUsed(M, {AsanDtorFunction});
2416 BasicBlock *AsanDtorBB = BasicBlock::Create(*C, "", AsanDtorFunction);
2417
2418 return ReturnInst::Create(*C, AsanDtorBB);
2419}
2420
2421void ModuleAddressSanitizer::InstrumentGlobalsCOFF(
2422 IRBuilder<> &IRB, ArrayRef<GlobalVariable *> ExtendedGlobals,
2423 ArrayRef<Constant *> MetadataInitializers) {
2424 assert(ExtendedGlobals.size() == MetadataInitializers.size());
2425 auto &DL = M.getDataLayout();
2426
2427 SmallVector<GlobalValue *, 16> MetadataGlobals(ExtendedGlobals.size());
2428 for (size_t i = 0; i < ExtendedGlobals.size(); i++) {
2429 Constant *Initializer = MetadataInitializers[i];
2430 GlobalVariable *G = ExtendedGlobals[i];
2431 GlobalVariable *Metadata = CreateMetadataGlobal(Initializer, G->getName());
2432 MDNode *MD = MDNode::get(M.getContext(), ValueAsMetadata::get(G));
2433 Metadata->setMetadata(LLVMContext::MD_associated, MD);
2434 MetadataGlobals[i] = Metadata;
2435
2436 // The MSVC linker always inserts padding when linking incrementally. We
2437 // cope with that by aligning each struct to its size, which must be a power
2438 // of two.
2439 unsigned SizeOfGlobalStruct = DL.getTypeAllocSize(Initializer->getType());
2440 assert(isPowerOf2_32(SizeOfGlobalStruct) &&
2441 "global metadata will not be padded appropriately");
2442 Metadata->setAlignment(assumeAligned(SizeOfGlobalStruct));
2443
2444 SetComdatForGlobalMetadata(G, Metadata, "");
2445 }
2446
2447 // Update llvm.compiler.used, adding the new metadata globals. This is
2448 // needed so that during LTO these variables stay alive.
2449 if (!MetadataGlobals.empty())
2450 appendToCompilerUsed(M, MetadataGlobals);
2451}
2452
2453void ModuleAddressSanitizer::instrumentGlobalsELF(
2454 IRBuilder<> &IRB, ArrayRef<GlobalVariable *> ExtendedGlobals,
2455 ArrayRef<Constant *> MetadataInitializers,
2456 const std::string &UniqueModuleId) {
2457 assert(ExtendedGlobals.size() == MetadataInitializers.size());
2458
2459 // Putting globals in a comdat changes the semantic and potentially cause
2460 // false negative odr violations at link time. If odr indicators are used, we
2461 // keep the comdat sections, as link time odr violations will be detected on
2462 // the odr indicator symbols.
2463 bool UseComdatForGlobalsGC = UseOdrIndicator && !UniqueModuleId.empty();
2464
2465 SmallVector<GlobalValue *, 16> MetadataGlobals(ExtendedGlobals.size());
2466 for (size_t i = 0; i < ExtendedGlobals.size(); i++) {
2467 GlobalVariable *G = ExtendedGlobals[i];
2469 CreateMetadataGlobal(MetadataInitializers[i], G->getName());
2470 MDNode *MD = MDNode::get(M.getContext(), ValueAsMetadata::get(G));
2471 Metadata->setMetadata(LLVMContext::MD_associated, MD);
2472 MetadataGlobals[i] = Metadata;
2473
2474 if (UseComdatForGlobalsGC)
2475 SetComdatForGlobalMetadata(G, Metadata, UniqueModuleId);
2476 }
2477
2478 // Update llvm.compiler.used, adding the new metadata globals. This is
2479 // needed so that during LTO these variables stay alive.
2480 if (!MetadataGlobals.empty())
2481 appendToCompilerUsed(M, MetadataGlobals);
2482
2483 // RegisteredFlag serves two purposes. First, we can pass it to dladdr()
2484 // to look up the loaded image that contains it. Second, we can store in it
2485 // whether registration has already occurred, to prevent duplicate
2486 // registration.
2487 //
2488 // Common linkage ensures that there is only one global per shared library.
2489 GlobalVariable *RegisteredFlag = new GlobalVariable(
2490 M, IntptrTy, false, GlobalVariable::CommonLinkage,
2491 ConstantInt::get(IntptrTy, 0), kAsanGlobalsRegisteredFlagName);
2493
2494 // Create start and stop symbols.
2495 GlobalVariable *StartELFMetadata = new GlobalVariable(
2496 M, IntptrTy, false, GlobalVariable::ExternalWeakLinkage, nullptr,
2497 "__start_" + getGlobalMetadataSection());
2499 GlobalVariable *StopELFMetadata = new GlobalVariable(
2500 M, IntptrTy, false, GlobalVariable::ExternalWeakLinkage, nullptr,
2501 "__stop_" + getGlobalMetadataSection());
2503
2504 // Create a call to register the globals with the runtime.
2505 if (ConstructorKind == AsanCtorKind::Global)
2506 IRB.CreateCall(AsanRegisterElfGlobals,
2507 {IRB.CreatePointerCast(RegisteredFlag, IntptrTy),
2508 IRB.CreatePointerCast(StartELFMetadata, IntptrTy),
2509 IRB.CreatePointerCast(StopELFMetadata, IntptrTy)});
2510
2511 // We also need to unregister globals at the end, e.g., when a shared library
2512 // gets closed.
2513 if (DestructorKind != AsanDtorKind::None && !MetadataGlobals.empty()) {
2514 IRBuilder<> IrbDtor(CreateAsanModuleDtor());
2515 IrbDtor.CreateCall(AsanUnregisterElfGlobals,
2516 {IRB.CreatePointerCast(RegisteredFlag, IntptrTy),
2517 IRB.CreatePointerCast(StartELFMetadata, IntptrTy),
2518 IRB.CreatePointerCast(StopELFMetadata, IntptrTy)});
2519 }
2520}
2521
2522void ModuleAddressSanitizer::InstrumentGlobalsMachO(
2523 IRBuilder<> &IRB, ArrayRef<GlobalVariable *> ExtendedGlobals,
2524 ArrayRef<Constant *> MetadataInitializers) {
2525 assert(ExtendedGlobals.size() == MetadataInitializers.size());
2526
2527 // On recent Mach-O platforms, use a structure which binds the liveness of
2528 // the global variable to the metadata struct. Keep the list of "Liveness" GV
2529 // created to be added to llvm.compiler.used
2530 StructType *LivenessTy = StructType::get(IntptrTy, IntptrTy);
2531 SmallVector<GlobalValue *, 16> LivenessGlobals(ExtendedGlobals.size());
2532
2533 for (size_t i = 0; i < ExtendedGlobals.size(); i++) {
2534 Constant *Initializer = MetadataInitializers[i];
2535 GlobalVariable *G = ExtendedGlobals[i];
2536 GlobalVariable *Metadata = CreateMetadataGlobal(Initializer, G->getName());
2537
2538 // On recent Mach-O platforms, we emit the global metadata in a way that
2539 // allows the linker to properly strip dead globals.
2540 auto LivenessBinder =
2541 ConstantStruct::get(LivenessTy, Initializer->getAggregateElement(0u),
2543 GlobalVariable *Liveness = new GlobalVariable(
2544 M, LivenessTy, false, GlobalVariable::InternalLinkage, LivenessBinder,
2545 Twine("__asan_binder_") + G->getName());
2546 Liveness->setSection("__DATA,__asan_liveness,regular,live_support");
2547 LivenessGlobals[i] = Liveness;
2548 }
2549
2550 // Update llvm.compiler.used, adding the new liveness globals. This is
2551 // needed so that during LTO these variables stay alive. The alternative
2552 // would be to have the linker handling the LTO symbols, but libLTO
2553 // current API does not expose access to the section for each symbol.
2554 if (!LivenessGlobals.empty())
2555 appendToCompilerUsed(M, LivenessGlobals);
2556
2557 // RegisteredFlag serves two purposes. First, we can pass it to dladdr()
2558 // to look up the loaded image that contains it. Second, we can store in it
2559 // whether registration has already occurred, to prevent duplicate
2560 // registration.
2561 //
2562 // common linkage ensures that there is only one global per shared library.
2563 GlobalVariable *RegisteredFlag = new GlobalVariable(
2564 M, IntptrTy, false, GlobalVariable::CommonLinkage,
2565 ConstantInt::get(IntptrTy, 0), kAsanGlobalsRegisteredFlagName);
2567
2568 if (ConstructorKind == AsanCtorKind::Global)
2569 IRB.CreateCall(AsanRegisterImageGlobals,
2570 {IRB.CreatePointerCast(RegisteredFlag, IntptrTy)});
2571
2572 // We also need to unregister globals at the end, e.g., when a shared library
2573 // gets closed.
2574 if (DestructorKind != AsanDtorKind::None) {
2575 IRBuilder<> IrbDtor(CreateAsanModuleDtor());
2576 IrbDtor.CreateCall(AsanUnregisterImageGlobals,
2577 {IRB.CreatePointerCast(RegisteredFlag, IntptrTy)});
2578 }
2579}
2580
2581void ModuleAddressSanitizer::InstrumentGlobalsWithMetadataArray(
2582 IRBuilder<> &IRB, ArrayRef<GlobalVariable *> ExtendedGlobals,
2583 ArrayRef<Constant *> MetadataInitializers) {
2584 assert(ExtendedGlobals.size() == MetadataInitializers.size());
2585 unsigned N = ExtendedGlobals.size();
2586 assert(N > 0);
2587
2588 // On platforms that don't have a custom metadata section, we emit an array
2589 // of global metadata structures.
2590 ArrayType *ArrayOfGlobalStructTy =
2591 ArrayType::get(MetadataInitializers[0]->getType(), N);
2592 auto AllGlobals = new GlobalVariable(
2593 M, ArrayOfGlobalStructTy, false, GlobalVariable::InternalLinkage,
2594 ConstantArray::get(ArrayOfGlobalStructTy, MetadataInitializers), "");
2595 if (Mapping.Scale > 3)
2596 AllGlobals->setAlignment(Align(1ULL << Mapping.Scale));
2597
2598 if (ConstructorKind == AsanCtorKind::Global)
2599 IRB.CreateCall(AsanRegisterGlobals,
2600 {IRB.CreatePointerCast(AllGlobals, IntptrTy),
2601 ConstantInt::get(IntptrTy, N)});
2602
2603 // We also need to unregister globals at the end, e.g., when a shared library
2604 // gets closed.
2605 if (DestructorKind != AsanDtorKind::None) {
2606 IRBuilder<> IrbDtor(CreateAsanModuleDtor());
2607 IrbDtor.CreateCall(AsanUnregisterGlobals,
2608 {IRB.CreatePointerCast(AllGlobals, IntptrTy),
2609 ConstantInt::get(IntptrTy, N)});
2610 }
2611}
2612
2613// This function replaces all global variables with new variables that have
2614// trailing redzones. It also creates a function that poisons
2615// redzones and inserts this function into llvm.global_ctors.
2616// Sets *CtorComdat to true if the global registration code emitted into the
2617// asan constructor is comdat-compatible.
2618void ModuleAddressSanitizer::instrumentGlobals(IRBuilder<> &IRB,
2619 bool *CtorComdat) {
2620 // Build set of globals that are aliased by some GA, where
2621 // getExcludedAliasedGlobal(GA) returns the relevant GlobalVariable.
2622 SmallPtrSet<const GlobalVariable *, 16> AliasedGlobalExclusions;
2623 if (CompileKernel) {
2624 for (auto &GA : M.aliases()) {
2625 if (const GlobalVariable *GV = getExcludedAliasedGlobal(GA))
2626 AliasedGlobalExclusions.insert(GV);
2627 }
2628 }
2629
2630 SmallVector<GlobalVariable *, 16> GlobalsToChange;
2631 for (auto &G : M.globals()) {
2632 if (!AliasedGlobalExclusions.count(&G) && shouldInstrumentGlobal(&G))
2633 GlobalsToChange.push_back(&G);
2634 }
2635
2636 size_t n = GlobalsToChange.size();
2637 auto &DL = M.getDataLayout();
2638
2639 // A global is described by a structure
2640 // size_t beg;
2641 // size_t size;
2642 // size_t size_with_redzone;
2643 // const char *name;
2644 // const char *module_name;
2645 // size_t has_dynamic_init;
2646 // size_t padding_for_windows_msvc_incremental_link;
2647 // size_t odr_indicator;
2648 // We initialize an array of such structures and pass it to a run-time call.
2649 StructType *GlobalStructTy =
2650 StructType::get(IntptrTy, IntptrTy, IntptrTy, IntptrTy, IntptrTy,
2651 IntptrTy, IntptrTy, IntptrTy);
2653 SmallVector<Constant *, 16> Initializers(n);
2654
2655 for (size_t i = 0; i < n; i++) {
2656 GlobalVariable *G = GlobalsToChange[i];
2657
2659 if (G->hasSanitizerMetadata())
2660 MD = G->getSanitizerMetadata();
2661
2662 // The runtime library tries demangling symbol names in the descriptor but
2663 // functionality like __cxa_demangle may be unavailable (e.g.
2664 // -static-libstdc++). So we demangle the symbol names here.
2665 std::string NameForGlobal = G->getName().str();
2668 /*AllowMerging*/ true, genName("global"));
2669
2670 Type *Ty = G->getValueType();
2671 const uint64_t SizeInBytes = DL.getTypeAllocSize(Ty);
2672 const uint64_t RightRedzoneSize = getRedzoneSizeForGlobal(SizeInBytes);
2673 Type *RightRedZoneTy = ArrayType::get(IRB.getInt8Ty(), RightRedzoneSize);
2674
2675 StructType *NewTy = StructType::get(Ty, RightRedZoneTy);
2676 Constant *NewInitializer = ConstantStruct::get(
2677 NewTy, G->getInitializer(), Constant::getNullValue(RightRedZoneTy));
2678
2679 // Create a new global variable with enough space for a redzone.
2680 GlobalValue::LinkageTypes Linkage = G->getLinkage();
2681 if (G->isConstant() && Linkage == GlobalValue::PrivateLinkage)
2683 GlobalVariable *NewGlobal = new GlobalVariable(
2684 M, NewTy, G->isConstant(), Linkage, NewInitializer, "", G,
2685 G->getThreadLocalMode(), G->getAddressSpace());
2686 NewGlobal->copyAttributesFrom(G);
2687 NewGlobal->setComdat(G->getComdat());
2688 NewGlobal->setAlignment(Align(getMinRedzoneSizeForGlobal()));
2689 // Don't fold globals with redzones. ODR violation detector and redzone
2690 // poisoning implicitly creates a dependence on the global's address, so it
2691 // is no longer valid for it to be marked unnamed_addr.
2693
2694 // Move null-terminated C strings to "__asan_cstring" section on Darwin.
2695 if (TargetTriple.isOSBinFormatMachO() && !G->hasSection() &&
2696 G->isConstant()) {
2697 auto Seq = dyn_cast<ConstantDataSequential>(G->getInitializer());
2698 if (Seq && Seq->isCString())
2699 NewGlobal->setSection("__TEXT,__asan_cstring,regular");
2700 }
2701
2702 // Transfer the debug info and type metadata. The payload starts at offset
2703 // zero so we can copy the metadata over as is.
2704 NewGlobal->copyMetadata(G, 0);
2705
2706 G->replaceAllUsesWith(NewGlobal);
2707 NewGlobal->takeName(G);
2708 G->eraseFromParent();
2709 NewGlobals[i] = NewGlobal;
2710
2711 Constant *ODRIndicator = Constant::getNullValue(IntptrTy);
2712 GlobalValue *InstrumentedGlobal = NewGlobal;
2713
2714 bool CanUsePrivateAliases =
2715 TargetTriple.isOSBinFormatELF() || TargetTriple.isOSBinFormatMachO() ||
2716 TargetTriple.isOSBinFormatWasm();
2717 if (CanUsePrivateAliases && UsePrivateAlias) {
2718 // Create local alias for NewGlobal to avoid crash on ODR between
2719 // instrumented and non-instrumented libraries.
2720 InstrumentedGlobal =
2722 }
2723
2724 // ODR should not happen for local linkage.
2725 if (NewGlobal->hasLocalLinkage()) {
2726 ODRIndicator = ConstantInt::getAllOnesValue(IntptrTy);
2727 } else if (UseOdrIndicator) {
2728 // With local aliases, we need to provide another externally visible
2729 // symbol __odr_asan_XXX to detect ODR violation.
2730 auto *ODRIndicatorSym =
2731 new GlobalVariable(M, IRB.getInt8Ty(), false, Linkage,
2733 kODRGenPrefix + NameForGlobal, nullptr,
2734 NewGlobal->getThreadLocalMode());
2735
2736 // Set meaningful attributes for indicator symbol.
2737 ODRIndicatorSym->setVisibility(NewGlobal->getVisibility());
2738 ODRIndicatorSym->setDLLStorageClass(NewGlobal->getDLLStorageClass());
2739 ODRIndicatorSym->setAlignment(Align(1));
2740 ODRIndicator = ConstantExpr::getPtrToInt(ODRIndicatorSym, IntptrTy);
2741 }
2742
2743 Constant *Initializer = ConstantStruct::get(
2744 GlobalStructTy,
2745 ConstantExpr::getPointerCast(InstrumentedGlobal, IntptrTy),
2746 ConstantInt::get(IntptrTy, SizeInBytes),
2747 ConstantInt::get(IntptrTy, SizeInBytes + RightRedzoneSize),
2748 ConstantExpr::getPointerCast(Name, IntptrTy),
2749 ConstantExpr::getPointerCast(getOrCreateModuleName(), IntptrTy),
2750 ConstantInt::get(IntptrTy, MD.IsDynInit),
2751 Constant::getNullValue(IntptrTy), ODRIndicator);
2752
2753 LLVM_DEBUG(dbgs() << "NEW GLOBAL: " << *NewGlobal << "\n");
2754
2755 Initializers[i] = Initializer;
2756 }
2757
2758 // Add instrumented globals to llvm.compiler.used list to avoid LTO from
2759 // ConstantMerge'ing them.
2760 SmallVector<GlobalValue *, 16> GlobalsToAddToUsedList;
2761 for (size_t i = 0; i < n; i++) {
2762 GlobalVariable *G = NewGlobals[i];
2763 if (G->getName().empty()) continue;
2764 GlobalsToAddToUsedList.push_back(G);
2765 }
2766 appendToCompilerUsed(M, ArrayRef<GlobalValue *>(GlobalsToAddToUsedList));
2767
2768 if (UseGlobalsGC && TargetTriple.isOSBinFormatELF()) {
2769 // Use COMDAT and register globals even if n == 0 to ensure that (a) the
2770 // linkage unit will only have one module constructor, and (b) the register
2771 // function will be called. The module destructor is not created when n ==
2772 // 0.
2773 *CtorComdat = true;
2774 instrumentGlobalsELF(IRB, NewGlobals, Initializers, getUniqueModuleId(&M));
2775 } else if (n == 0) {
2776 // When UseGlobalsGC is false, COMDAT can still be used if n == 0, because
2777 // all compile units will have identical module constructor/destructor.
2778 *CtorComdat = TargetTriple.isOSBinFormatELF();
2779 } else {
2780 *CtorComdat = false;
2781 if (UseGlobalsGC && TargetTriple.isOSBinFormatCOFF()) {
2782 InstrumentGlobalsCOFF(IRB, NewGlobals, Initializers);
2783 } else if (UseGlobalsGC && ShouldUseMachOGlobalsSection()) {
2784 InstrumentGlobalsMachO(IRB, NewGlobals, Initializers);
2785 } else {
2786 InstrumentGlobalsWithMetadataArray(IRB, NewGlobals, Initializers);
2787 }
2788 }
2789
2790 // Create calls for poisoning before initializers run and unpoisoning after.
2791 if (ClInitializers)
2792 createInitializerPoisonCalls();
2793
2794 LLVM_DEBUG(dbgs() << M);
2795}
2796
2797uint64_t
2798ModuleAddressSanitizer::getRedzoneSizeForGlobal(uint64_t SizeInBytes) const {
2799 constexpr uint64_t kMaxRZ = 1 << 18;
2800 const uint64_t MinRZ = getMinRedzoneSizeForGlobal();
2801
2802 uint64_t RZ = 0;
2803 if (SizeInBytes <= MinRZ / 2) {
2804 // Reduce redzone size for small size objects, e.g. int, char[1]. MinRZ is
2805 // at least 32 bytes, optimize when SizeInBytes is less than or equal to
2806 // half of MinRZ.
2807 RZ = MinRZ - SizeInBytes;
2808 } else {
2809 // Calculate RZ, where MinRZ <= RZ <= MaxRZ, and RZ ~ 1/4 * SizeInBytes.
2810 RZ = std::clamp((SizeInBytes / MinRZ / 4) * MinRZ, MinRZ, kMaxRZ);
2811
2812 // Round up to multiple of MinRZ.
2813 if (SizeInBytes % MinRZ)
2814 RZ += MinRZ - (SizeInBytes % MinRZ);
2815 }
2816
2817 assert((RZ + SizeInBytes) % MinRZ == 0);
2818
2819 return RZ;
2820}
2821
2822int ModuleAddressSanitizer::GetAsanVersion() const {
2823 int LongSize = M.getDataLayout().getPointerSizeInBits();
2824 bool isAndroid = M.getTargetTriple().isAndroid();
2825 int Version = 8;
2826 // 32-bit Android is one version ahead because of the switch to dynamic
2827 // shadow.
2828 Version += (LongSize == 32 && isAndroid);
2829 return Version;
2830}
2831
2832GlobalVariable *ModuleAddressSanitizer::getOrCreateModuleName() {
2833 if (!ModuleName) {
2834 // We shouldn't merge same module names, as this string serves as unique
2835 // module ID in runtime.
2836 ModuleName =
2837 createPrivateGlobalForString(M, M.getModuleIdentifier(),
2838 /*AllowMerging*/ false, genName("module"));
2839 }
2840 return ModuleName;
2841}
2842
2843bool ModuleAddressSanitizer::instrumentModule() {
2844 initializeCallbacks();
2845
2846 for (Function &F : M)
2847 removeASanIncompatibleFnAttributes(F, /*ReadsArgMem=*/false);
2848
2849 // Create a module constructor. A destructor is created lazily because not all
2850 // platforms, and not all modules need it.
2851 if (ConstructorKind == AsanCtorKind::Global) {
2852 if (CompileKernel) {
2853 // The kernel always builds with its own runtime, and therefore does not
2854 // need the init and version check calls.
2855 AsanCtorFunction = createSanitizerCtor(M, kAsanModuleCtorName);
2856 } else {
2857 std::string AsanVersion = std::to_string(GetAsanVersion());
2858 std::string VersionCheckName =
2859 InsertVersionCheck ? (kAsanVersionCheckNamePrefix + AsanVersion) : "";
2860 std::tie(AsanCtorFunction, std::ignore) =
2862 M, kAsanModuleCtorName, kAsanInitName, /*InitArgTypes=*/{},
2863 /*InitArgs=*/{}, VersionCheckName);
2864 }
2865 }
2866
2867 bool CtorComdat = true;
2868 if (ClGlobals) {
2869 assert(AsanCtorFunction || ConstructorKind == AsanCtorKind::None);
2870 if (AsanCtorFunction) {
2871 IRBuilder<> IRB(AsanCtorFunction->getEntryBlock().getTerminator());
2872 instrumentGlobals(IRB, &CtorComdat);
2873 } else {
2874 IRBuilder<> IRB(*C);
2875 instrumentGlobals(IRB, &CtorComdat);
2876 }
2877 }
2878
2879 const uint64_t Priority = GetCtorAndDtorPriority(TargetTriple);
2880
2881 // Put the constructor and destructor in comdat if both
2882 // (1) global instrumentation is not TU-specific
2883 // (2) target is ELF.
2884 if (UseCtorComdat && TargetTriple.isOSBinFormatELF() && CtorComdat) {
2885 if (AsanCtorFunction) {
2886 AsanCtorFunction->setComdat(M.getOrInsertComdat(kAsanModuleCtorName));
2887 appendToGlobalCtors(M, AsanCtorFunction, Priority, AsanCtorFunction);
2888 }
2889 if (AsanDtorFunction) {
2890 AsanDtorFunction->setComdat(M.getOrInsertComdat(kAsanModuleDtorName));
2891 appendToGlobalDtors(M, AsanDtorFunction, Priority, AsanDtorFunction);
2892 }
2893 } else {
2894 if (AsanCtorFunction)
2895 appendToGlobalCtors(M, AsanCtorFunction, Priority);
2896 if (AsanDtorFunction)
2897 appendToGlobalDtors(M, AsanDtorFunction, Priority);
2898 }
2899
2900 return true;
2901}
2902
2903void AddressSanitizer::initializeCallbacks(const TargetLibraryInfo *TLI) {
2904 IRBuilder<> IRB(*C);
2905 // Create __asan_report* callbacks.
2906 // IsWrite, TypeSize and Exp are encoded in the function name.
2907 for (int Exp = 0; Exp < 2; Exp++) {
2908 for (size_t AccessIsWrite = 0; AccessIsWrite <= 1; AccessIsWrite++) {
2909 const std::string TypeStr = AccessIsWrite ? "store" : "load";
2910 const std::string ExpStr = Exp ? "exp_" : "";
2911 const std::string EndingStr = Recover ? "_noabort" : "";
2912
2913 SmallVector<Type *, 3> Args2 = {IntptrTy, IntptrTy};
2914 SmallVector<Type *, 2> Args1{1, IntptrTy};
2915 AttributeList AL2;
2916 AttributeList AL1;
2917 if (Exp) {
2918 Type *ExpType = Type::getInt32Ty(*C);
2919 Args2.push_back(ExpType);
2920 Args1.push_back(ExpType);
2921 if (auto AK = TLI->getExtAttrForI32Param(false)) {
2922 AL2 = AL2.addParamAttribute(*C, 2, AK);
2923 AL1 = AL1.addParamAttribute(*C, 1, AK);
2924 }
2925 }
2926 AsanErrorCallbackSized[AccessIsWrite][Exp] = Inserter.insertFunction(
2927 kAsanReportErrorTemplate + ExpStr + TypeStr + "_n" + EndingStr,
2928 FunctionType::get(IRB.getVoidTy(), Args2, false), AL2);
2929
2930 AsanMemoryAccessCallbackSized[AccessIsWrite][Exp] =
2931 Inserter.insertFunction(
2932 ClMemoryAccessCallbackPrefix + ExpStr + TypeStr + "N" + EndingStr,
2933 FunctionType::get(IRB.getVoidTy(), Args2, false), AL2);
2934
2935 for (size_t AccessSizeIndex = 0; AccessSizeIndex < kNumberOfAccessSizes;
2936 AccessSizeIndex++) {
2937 const std::string Suffix = TypeStr + itostr(1ULL << AccessSizeIndex);
2938 AsanErrorCallback[AccessIsWrite][Exp][AccessSizeIndex] =
2939 Inserter.insertFunction(
2940 kAsanReportErrorTemplate + ExpStr + Suffix + EndingStr,
2941 FunctionType::get(IRB.getVoidTy(), Args1, false), AL1);
2942
2943 AsanMemoryAccessCallback[AccessIsWrite][Exp][AccessSizeIndex] =
2944 Inserter.insertFunction(
2945 ClMemoryAccessCallbackPrefix + ExpStr + Suffix + EndingStr,
2946 FunctionType::get(IRB.getVoidTy(), Args1, false), AL1);
2947 }
2948 }
2949 }
2950
2951 const std::string MemIntrinCallbackPrefix =
2952 (CompileKernel && !ClKasanMemIntrinCallbackPrefix)
2953 ? std::string("")
2955 AsanMemmove = Inserter.insertFunction(MemIntrinCallbackPrefix + "memmove",
2956 PtrTy, PtrTy, PtrTy, IntptrTy);
2957 AsanMemcpy = Inserter.insertFunction(MemIntrinCallbackPrefix + "memcpy",
2958 PtrTy, PtrTy, PtrTy, IntptrTy);
2959 AsanMemset =
2960 Inserter.insertFunction(MemIntrinCallbackPrefix + "memset",
2961 TLI->getAttrList(C, {1},
2962 /*Signed=*/false),
2963 PtrTy, PtrTy, IRB.getInt32Ty(), IntptrTy);
2964
2965 AsanHandleNoReturnFunc =
2966 Inserter.insertFunction(kAsanHandleNoReturnName, IRB.getVoidTy());
2967
2968 AsanPtrCmpFunction =
2969 Inserter.insertFunction(kAsanPtrCmp, IRB.getVoidTy(), IntptrTy, IntptrTy);
2970 AsanPtrSubFunction =
2971 Inserter.insertFunction(kAsanPtrSub, IRB.getVoidTy(), IntptrTy, IntptrTy);
2972 if (Mapping.InGlobal)
2973 AsanShadowGlobal = M.getOrInsertGlobal("__asan_shadow",
2974 ArrayType::get(IRB.getInt8Ty(), 0));
2975
2976 AMDGPUAddressShared =
2977 Inserter.insertFunction(kAMDGPUAddressSharedName, IRB.getInt1Ty(), PtrTy);
2978 AMDGPUAddressPrivate = Inserter.insertFunction(kAMDGPUAddressPrivateName,
2979 IRB.getInt1Ty(), PtrTy);
2980}
2981
2982bool AddressSanitizer::maybeInsertAsanInitAtFunctionEntry(Function &F) {
2983 // For each NSObject descendant having a +load method, this method is invoked
2984 // by the ObjC runtime before any of the static constructors is called.
2985 // Therefore we need to instrument such methods with a call to __asan_init
2986 // at the beginning in order to initialize our runtime before any access to
2987 // the shadow memory.
2988 // We cannot just ignore these methods, because they may call other
2989 // instrumented functions.
2990 if (F.getName().contains(" load]")) {
2991 FunctionCallee AsanInitFunction =
2992 declareSanitizerInitFunction(*F.getParent(), kAsanInitName, {});
2993 IRBuilder<> IRB(&F.front(), F.front().begin());
2994 IRB.CreateCall(AsanInitFunction, {});
2995 return true;
2996 }
2997 return false;
2998}
2999
3000bool AddressSanitizer::maybeInsertDynamicShadowAtFunctionEntry(Function &F) {
3001 // Generate code only when dynamic addressing is needed.
3002 if (Mapping.Offset != kDynamicShadowSentinel)
3003 return false;
3004
3005 IRBuilder<> IRB(&F.front().front());
3006 if (Mapping.InGlobal) {
3008 // An empty inline asm with input reg == output reg.
3009 // An opaque pointer-to-int cast, basically.
3011 FunctionType::get(IntptrTy, {AsanShadowGlobal->getType()}, false),
3012 StringRef(""), StringRef("=r,0"),
3013 /*hasSideEffects=*/false);
3014 LocalDynamicShadow =
3015 IRB.CreateCall(Asm, {AsanShadowGlobal}, ".asan.shadow");
3016 } else {
3017 LocalDynamicShadow =
3018 IRB.CreatePointerCast(AsanShadowGlobal, IntptrTy, ".asan.shadow");
3019 }
3020 } else {
3021 Value *GlobalDynamicAddress = F.getParent()->getOrInsertGlobal(
3023 LocalDynamicShadow = IRB.CreateLoad(IntptrTy, GlobalDynamicAddress);
3024 }
3025 return true;
3026}
3027
3028void AddressSanitizer::markEscapedLocalAllocas(Function &F) {
3029 // Find the one possible call to llvm.localescape and pre-mark allocas passed
3030 // to it as uninteresting. This assumes we haven't started processing allocas
3031 // yet. This check is done up front because iterating the use list in
3032 // isInterestingAlloca would be algorithmically slower.
3033 assert(ProcessedAllocas.empty() && "must process localescape before allocas");
3034
3035 // Try to get the declaration of llvm.localescape. If it's not in the module,
3036 // we can exit early.
3037 if (!F.getParent()->getFunction("llvm.localescape")) return;
3038
3039 // Look for a call to llvm.localescape call in the entry block. It can't be in
3040 // any other block.
3041 for (Instruction &I : F.getEntryBlock()) {
3043 if (II && II->getIntrinsicID() == Intrinsic::localescape) {
3044 // We found a call. Mark all the allocas passed in as uninteresting.
3045 for (Value *Arg : II->args()) {
3046 AllocaInst *AI = dyn_cast<AllocaInst>(Arg->stripPointerCasts());
3047 assert(AI && AI->isStaticAlloca() &&
3048 "non-static alloca arg to localescape");
3049 ProcessedAllocas[AI] = false;
3050 }
3051 break;
3052 }
3053 }
3054}
3055// Mitigation for https://github.com/google/sanitizers/issues/749
3056// We don't instrument Windows catch-block parameters to avoid
3057// interfering with exception handling assumptions.
3058void AddressSanitizer::markCatchParametersAsUninteresting(Function &F) {
3059 for (BasicBlock &BB : F) {
3060 for (Instruction &I : BB) {
3061 if (auto *CatchPad = dyn_cast<CatchPadInst>(&I)) {
3062 // Mark the parameters to a catch-block as uninteresting to avoid
3063 // instrumenting them.
3064 for (Value *Operand : CatchPad->arg_operands())
3065 if (auto *AI = dyn_cast<AllocaInst>(Operand))
3066 ProcessedAllocas[AI] = false;
3067 }
3068 }
3069 }
3070}
3071
3072bool AddressSanitizer::suppressInstrumentationSiteForDebug(int &Instrumented) {
3073 bool ShouldInstrument =
3074 ClDebugMin < 0 || ClDebugMax < 0 ||
3075 (Instrumented >= ClDebugMin && Instrumented <= ClDebugMax);
3076 Instrumented++;
3077 return !ShouldInstrument;
3078}
3079
3080bool AddressSanitizer::instrumentFunction(Function &F,
3081 const TargetLibraryInfo *TLI,
3082 const TargetTransformInfo *TTI) {
3083 bool FunctionModified = false;
3084
3085 // Do not apply any instrumentation for naked functions.
3086 if (F.hasFnAttribute(Attribute::Naked))
3087 return FunctionModified;
3088
3089 // If needed, insert __asan_init before checking for SanitizeAddress attr.
3090 // This function needs to be called even if the function body is not
3091 // instrumented.
3092 if (maybeInsertAsanInitAtFunctionEntry(F))
3093 FunctionModified = true;
3094
3095 // Leave if the function doesn't need instrumentation.
3096 if (!F.hasFnAttribute(Attribute::SanitizeAddress)) return FunctionModified;
3097
3098 if (F.hasFnAttribute(Attribute::DisableSanitizerInstrumentation))
3099 return FunctionModified;
3100
3101 LLVM_DEBUG(dbgs() << "ASAN instrumenting:\n" << F << "\n");
3102
3103 initializeCallbacks(TLI);
3104
3105 FunctionStateRAII CleanupObj(this);
3106
3107 RuntimeCallInserter RTCI(F);
3108
3109 FunctionModified |= maybeInsertDynamicShadowAtFunctionEntry(F);
3110
3111 // We can't instrument allocas used with llvm.localescape. Only static allocas
3112 // can be passed to that intrinsic.
3113 markEscapedLocalAllocas(F);
3114
3115 if (TargetTriple.isOSWindows())
3116 markCatchParametersAsUninteresting(F);
3117
3118 // We want to instrument every address only once per basic block (unless there
3119 // are calls between uses).
3120 SmallPtrSet<Value *, 16> TempsToInstrument;
3121 SmallVector<InterestingMemoryOperand, 16> OperandsToInstrument;
3122 SmallVector<MemIntrinsic *, 16> IntrinToInstrument;
3123 SmallVector<Instruction *, 8> NoReturnCalls;
3125 SmallVector<Instruction *, 16> PointerComparisonsOrSubtracts;
3126
3127 // Fill the set of memory operations to instrument.
3128 for (auto &BB : F) {
3129 AllBlocks.push_back(&BB);
3130 TempsToInstrument.clear();
3131 int NumInsnsPerBB = 0;
3132 for (auto &Inst : BB) {
3133 if (LooksLikeCodeInBug11395(&Inst)) return false;
3134 // Skip instructions inserted by another instrumentation.
3135 if (Inst.hasMetadata(LLVMContext::MD_nosanitize))
3136 continue;
3137 SmallVector<InterestingMemoryOperand, 1> InterestingOperands;
3138 getInterestingMemoryOperands(&Inst, InterestingOperands, TTI);
3139
3140 if (!InterestingOperands.empty()) {
3141 for (auto &Operand : InterestingOperands) {
3142 if (ClOpt && ClOptSameTemp) {
3143 Value *Ptr = Operand.getPtr();
3144 // If we have a mask, skip instrumentation if we've already
3145 // instrumented the full object. But don't add to TempsToInstrument
3146 // because we might get another load/store with a different mask.
3147 if (Operand.MaybeMask) {
3148 if (TempsToInstrument.count(Ptr))
3149 continue; // We've seen this (whole) temp in the current BB.
3150 } else {
3151 if (!TempsToInstrument.insert(Ptr).second)
3152 continue; // We've seen this temp in the current BB.
3153 }
3154 }
3155 OperandsToInstrument.push_back(Operand);
3156 NumInsnsPerBB++;
3157 }
3158 } else if (((ClInvalidPointerPairs || ClInvalidPointerCmp) &&
3162 PointerComparisonsOrSubtracts.push_back(&Inst);
3163 } else if (MemIntrinsic *MI = dyn_cast<MemIntrinsic>(&Inst)) {
3164 // ok, take it.
3165 IntrinToInstrument.push_back(MI);
3166 NumInsnsPerBB++;
3167 } else {
3168 if (auto *CB = dyn_cast<CallBase>(&Inst)) {
3169 // A call inside BB.
3170 TempsToInstrument.clear();
3171 if (CB->doesNotReturn())
3172 NoReturnCalls.push_back(CB);
3173 }
3174 if (CallInst *CI = dyn_cast<CallInst>(&Inst))
3176 }
3177 if (NumInsnsPerBB >= ClMaxInsnsToInstrumentPerBB) break;
3178 }
3179 }
3180
3181 bool UseCalls = (InstrumentationWithCallsThreshold >= 0 &&
3182 OperandsToInstrument.size() + IntrinToInstrument.size() >
3183 (unsigned)InstrumentationWithCallsThreshold);
3184 const DataLayout &DL = F.getDataLayout();
3185 ObjectSizeOffsetVisitor ObjSizeVis(DL, TLI, F.getContext());
3186
3187 // Instrument.
3188 int NumInstrumented = 0;
3189 for (auto &Operand : OperandsToInstrument) {
3190 if (!suppressInstrumentationSiteForDebug(NumInstrumented))
3191 instrumentMop(ObjSizeVis, Operand, UseCalls,
3192 F.getDataLayout(), RTCI);
3193 FunctionModified = true;
3194 }
3195 for (auto *Inst : IntrinToInstrument) {
3196 if (!suppressInstrumentationSiteForDebug(NumInstrumented))
3197 instrumentMemIntrinsic(Inst, RTCI);
3198 FunctionModified = true;
3199 }
3200
3201 FunctionStackPoisoner FSP(F, *this, RTCI);
3202 bool ChangedStack = FSP.runOnFunction();
3203
3204 // We must unpoison the stack before NoReturn calls (throw, _exit, etc).
3205 // See e.g. https://github.com/google/sanitizers/issues/37
3206 for (auto *CI : NoReturnCalls) {
3207 IRBuilder<> IRB(CI);
3208 RTCI.createRuntimeCall(IRB, AsanHandleNoReturnFunc, {});
3209 }
3210
3211 for (auto *Inst : PointerComparisonsOrSubtracts) {
3212 instrumentPointerComparisonOrSubtraction(Inst, RTCI);
3213 FunctionModified = true;
3214 }
3215
3216 if (ChangedStack || !NoReturnCalls.empty())
3217 FunctionModified = true;
3218
3219 LLVM_DEBUG(dbgs() << "ASAN done instrumenting: " << FunctionModified << " "
3220 << F << "\n");
3221
3222 return FunctionModified;
3223}
3224
3225// Workaround for bug 11395: we don't want to instrument stack in functions
3226// with large assembly blobs (32-bit only), otherwise reg alloc may crash.
3227// FIXME: remove once the bug 11395 is fixed.
3228bool AddressSanitizer::LooksLikeCodeInBug11395(Instruction *I) {
3229 if (LongSize != 32) return false;
3231 if (!CI || !CI->isInlineAsm()) return false;
3232 if (CI->arg_size() <= 5)
3233 return false;
3234 // We have inline assembly with quite a few arguments.
3235 return true;
3236}
3237
3238void FunctionStackPoisoner::initializeCallbacks(Module &) {
3239 IRBuilder<> IRB(*C);
3240 if (ASan.UseAfterReturn == AsanDetectStackUseAfterReturnMode::Always ||
3241 ASan.UseAfterReturn == AsanDetectStackUseAfterReturnMode::Runtime) {
3242 const char *MallocNameTemplate =
3243 ASan.UseAfterReturn == AsanDetectStackUseAfterReturnMode::Always
3246 for (int Index = 0; Index <= kMaxAsanStackMallocSizeClass; Index++) {
3247 std::string Suffix = itostr(Index);
3248 AsanStackMallocFunc[Index] = ASan.Inserter.insertFunction(
3249 MallocNameTemplate + Suffix, IntptrTy, IntptrTy);
3250 AsanStackFreeFunc[Index] =
3251 ASan.Inserter.insertFunction(kAsanStackFreeNameTemplate + Suffix,
3252 IRB.getVoidTy(), IntptrTy, IntptrTy);
3253 }
3254 }
3255 if (ASan.UseAfterScope) {
3256 AsanPoisonStackMemoryFunc = ASan.Inserter.insertFunction(
3257 kAsanPoisonStackMemoryName, IRB.getVoidTy(), IntptrTy, IntptrTy);
3258 AsanUnpoisonStackMemoryFunc = ASan.Inserter.insertFunction(
3259 kAsanUnpoisonStackMemoryName, IRB.getVoidTy(), IntptrTy, IntptrTy);
3260 }
3261
3262 for (size_t Val : {0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0xf1, 0xf2,
3263 0xf3, 0xf5, 0xf8}) {
3264 std::ostringstream Name;
3266 Name << std::setw(2) << std::setfill('0') << std::hex << Val;
3267 AsanSetShadowFunc[Val] = ASan.Inserter.insertFunction(
3268 Name.str(), IRB.getVoidTy(), IntptrTy, IntptrTy);
3269 }
3270
3271 AsanAllocaPoisonFunc = ASan.Inserter.insertFunction(
3272 kAsanAllocaPoison, IRB.getVoidTy(), IntptrTy, IntptrTy);
3273 AsanAllocasUnpoisonFunc = ASan.Inserter.insertFunction(
3274 kAsanAllocasUnpoison, IRB.getVoidTy(), IntptrTy, IntptrTy);
3275}
3276
3277void FunctionStackPoisoner::copyToShadowInline(ArrayRef<uint8_t> ShadowMask,
3278 ArrayRef<uint8_t> ShadowBytes,
3279 size_t Begin, size_t End,
3280 IRBuilder<> &IRB,
3281 Value *ShadowBase) {
3282 if (Begin >= End)
3283 return;
3284
3285 const size_t LargestStoreSizeInBytes =
3286 std::min<size_t>(sizeof(uint64_t), ASan.LongSize / 8);
3287
3288 const bool IsLittleEndian = F.getDataLayout().isLittleEndian();
3289
3290 // Poison given range in shadow using larges store size with out leading and
3291 // trailing zeros in ShadowMask. Zeros never change, so they need neither
3292 // poisoning nor up-poisoning. Still we don't mind if some of them get into a
3293 // middle of a store.
3294 for (size_t i = Begin; i < End;) {
3295 if (!ShadowMask[i]) {
3296 assert(!ShadowBytes[i]);
3297 ++i;
3298 continue;
3299 }
3300
3301 size_t StoreSizeInBytes = LargestStoreSizeInBytes;
3302 // Fit store size into the range.
3303 while (StoreSizeInBytes > End - i)
3304 StoreSizeInBytes /= 2;
3305
3306 // Minimize store size by trimming trailing zeros.
3307 for (size_t j = StoreSizeInBytes - 1; j && !ShadowMask[i + j]; --j) {
3308 while (j <= StoreSizeInBytes / 2)
3309 StoreSizeInBytes /= 2;
3310 }
3311
3312 uint64_t Val = 0;
3313 for (size_t j = 0; j < StoreSizeInBytes; j++) {
3314 if (IsLittleEndian)
3315 Val |= (uint64_t)ShadowBytes[i + j] << (8 * j);
3316 else
3317 Val = (Val << 8) | ShadowBytes[i + j];
3318 }
3319
3320 Value *Ptr = IRB.CreateAdd(ShadowBase, ConstantInt::get(IntptrTy, i));
3321 Value *Poison = IRB.getIntN(StoreSizeInBytes * 8, Val);
3323 Poison, IRB.CreateIntToPtr(Ptr, PointerType::getUnqual(Poison->getContext())),
3324 Align(1));
3325
3326 i += StoreSizeInBytes;
3327 }
3328}
3329
3330void FunctionStackPoisoner::copyToShadow(ArrayRef<uint8_t> ShadowMask,
3331 ArrayRef<uint8_t> ShadowBytes,
3332 IRBuilder<> &IRB, Value *ShadowBase) {
3333 copyToShadow(ShadowMask, ShadowBytes, 0, ShadowMask.size(), IRB, ShadowBase);
3334}
3335
3336void FunctionStackPoisoner::copyToShadow(ArrayRef<uint8_t> ShadowMask,
3337 ArrayRef<uint8_t> ShadowBytes,
3338 size_t Begin, size_t End,
3339 IRBuilder<> &IRB, Value *ShadowBase) {
3340 assert(ShadowMask.size() == ShadowBytes.size());
3341 size_t Done = Begin;
3342 for (size_t i = Begin, j = Begin + 1; i < End; i = j++) {
3343 if (!ShadowMask[i]) {
3344 assert(!ShadowBytes[i]);
3345 continue;
3346 }
3347 uint8_t Val = ShadowBytes[i];
3348 if (!AsanSetShadowFunc[Val])
3349 continue;
3350
3351 // Skip same values.
3352 for (; j < End && ShadowMask[j] && Val == ShadowBytes[j]; ++j) {
3353 }
3354
3355 if (j - i >= ASan.MaxInlinePoisoningSize) {
3356 copyToShadowInline(ShadowMask, ShadowBytes, Done, i, IRB, ShadowBase);
3357 RTCI.createRuntimeCall(
3358 IRB, AsanSetShadowFunc[Val],
3359 {IRB.CreateAdd(ShadowBase, ConstantInt::get(IntptrTy, i)),
3360 ConstantInt::get(IntptrTy, j - i)});
3361 Done = j;
3362 }
3363 }
3364
3365 copyToShadowInline(ShadowMask, ShadowBytes, Done, End, IRB, ShadowBase);
3366}
3367
3368// Fake stack allocator (asan_fake_stack.h) has 11 size classes
3369// for every power of 2 from kMinStackMallocSize to kMaxAsanStackMallocSizeClass
3370static int StackMallocSizeClass(uint64_t LocalStackSize) {
3371 assert(LocalStackSize <= kMaxStackMallocSize);
3372 uint64_t MaxSize = kMinStackMallocSize;
3373 for (int i = 0;; i++, MaxSize *= 2)
3374 if (LocalStackSize <= MaxSize) return i;
3375 llvm_unreachable("impossible LocalStackSize");
3376}
3377
3378void FunctionStackPoisoner::copyArgsPassedByValToAllocas() {
3379 Instruction *CopyInsertPoint = &F.front().front();
3380 if (CopyInsertPoint == ASan.LocalDynamicShadow) {
3381 // Insert after the dynamic shadow location is determined
3382 CopyInsertPoint = CopyInsertPoint->getNextNode();
3383 assert(CopyInsertPoint);
3384 }
3385 IRBuilder<> IRB(CopyInsertPoint);
3386 const DataLayout &DL = F.getDataLayout();
3387 for (Argument &Arg : F.args()) {
3388 if (Arg.hasByValAttr()) {
3389 Type *Ty = Arg.getParamByValType();
3390 const Align Alignment =
3391 DL.getValueOrABITypeAlignment(Arg.getParamAlign(), Ty);
3392
3393 AllocaInst *AI = IRB.CreateAlloca(
3394 Ty, nullptr,
3395 (Arg.hasName() ? Arg.getName() : "Arg" + Twine(Arg.getArgNo())) +
3396 ".byval");
3397 AI->setAlignment(Alignment);
3398 Arg.replaceAllUsesWith(AI);
3399
3400 uint64_t AllocSize = DL.getTypeAllocSize(Ty);
3401 IRB.CreateMemCpy(AI, Alignment, &Arg, Alignment, AllocSize);
3402 }
3403 }
3404}
3405
3406PHINode *FunctionStackPoisoner::createPHI(IRBuilder<> &IRB, Value *Cond,
3407 Value *ValueIfTrue,
3408 Instruction *ThenTerm,
3409 Value *ValueIfFalse) {
3410 PHINode *PHI = IRB.CreatePHI(ValueIfTrue->getType(), 2);
3411 BasicBlock *CondBlock = cast<Instruction>(Cond)->getParent();
3412 PHI->addIncoming(ValueIfFalse, CondBlock);
3413 BasicBlock *ThenBlock = ThenTerm->getParent();
3414 PHI->addIncoming(ValueIfTrue, ThenBlock);
3415 return PHI;
3416}
3417
3418Value *FunctionStackPoisoner::createAllocaForLayout(
3419 IRBuilder<> &IRB, const ASanStackFrameLayout &L, bool Dynamic) {
3420 AllocaInst *Alloca;
3421 if (Dynamic) {
3422 Alloca = IRB.CreateAlloca(IRB.getInt8Ty(),
3423 ConstantInt::get(IRB.getInt64Ty(), L.FrameSize),
3424 "MyAlloca");
3425 } else {
3426 Alloca = IRB.CreateAlloca(ArrayType::get(IRB.getInt8Ty(), L.FrameSize),
3427 nullptr, "MyAlloca");
3428 assert(Alloca->isStaticAlloca());
3429 }
3430 assert((ClRealignStack & (ClRealignStack - 1)) == 0);
3431 uint64_t FrameAlignment = std::max(L.FrameAlignment, uint64_t(ClRealignStack));
3432 Alloca->setAlignment(Align(FrameAlignment));
3433 return Alloca;
3434}
3435
3436void FunctionStackPoisoner::createDynamicAllocasInitStorage() {
3437 BasicBlock &FirstBB = *F.begin();
3438 IRBuilder<> IRB(dyn_cast<Instruction>(FirstBB.begin()));
3439 DynamicAllocaLayout = IRB.CreateAlloca(IntptrTy, nullptr);
3440 IRB.CreateStore(Constant::getNullValue(IntptrTy), DynamicAllocaLayout);
3441 DynamicAllocaLayout->setAlignment(Align(32));
3442}
3443
3444void FunctionStackPoisoner::processDynamicAllocas() {
3445 if (!ClInstrumentDynamicAllocas || DynamicAllocaVec.empty()) {
3446 assert(DynamicAllocaPoisonCallVec.empty());
3447 return;
3448 }
3449
3450 // Insert poison calls for lifetime intrinsics for dynamic allocas.
3451 for (const auto &APC : DynamicAllocaPoisonCallVec) {
3452 assert(APC.InsBefore);
3453 assert(APC.AI);
3454 assert(ASan.isInterestingAlloca(*APC.AI));
3455 assert(!APC.AI->isStaticAlloca());
3456
3457 IRBuilder<> IRB(APC.InsBefore);
3458 poisonAlloca(APC.AI, APC.Size, IRB, APC.DoPoison);
3459 // Dynamic allocas will be unpoisoned unconditionally below in
3460 // unpoisonDynamicAllocas.
3461 // Flag that we need unpoison static allocas.
3462 }
3463
3464 // Handle dynamic allocas.
3465 createDynamicAllocasInitStorage();
3466 for (auto &AI : DynamicAllocaVec)
3467 handleDynamicAllocaCall(AI);
3468 unpoisonDynamicAllocas();
3469}
3470
3471/// Collect instructions in the entry block after \p InsBefore which initialize
3472/// permanent storage for a function argument. These instructions must remain in
3473/// the entry block so that uninitialized values do not appear in backtraces. An
3474/// added benefit is that this conserves spill slots. This does not move stores
3475/// before instrumented / "interesting" allocas.
3477 AddressSanitizer &ASan, Instruction &InsBefore,
3478 SmallVectorImpl<Instruction *> &InitInsts) {
3479 Instruction *Start = InsBefore.getNextNode();
3480 for (Instruction *It = Start; It; It = It->getNextNode()) {
3481 // Argument initialization looks like:
3482 // 1) store <Argument>, <Alloca> OR
3483 // 2) <CastArgument> = cast <Argument> to ...
3484 // store <CastArgument> to <Alloca>
3485 // Do not consider any other kind of instruction.
3486 //
3487 // Note: This covers all known cases, but may not be exhaustive. An
3488 // alternative to pattern-matching stores is to DFS over all Argument uses:
3489 // this might be more general, but is probably much more complicated.
3490 if (isa<AllocaInst>(It) || isa<CastInst>(It))
3491 continue;
3492 if (auto *Store = dyn_cast<StoreInst>(It)) {
3493 // The store destination must be an alloca that isn't interesting for
3494 // ASan to instrument. These are moved up before InsBefore, and they're
3495 // not interesting because allocas for arguments can be mem2reg'd.
3496 auto *Alloca = dyn_cast<AllocaInst>(Store->getPointerOperand());
3497 if (!Alloca || ASan.isInterestingAlloca(*Alloca))
3498 continue;
3499
3500 Value *Val = Store->getValueOperand();
3501 bool IsDirectArgInit = isa<Argument>(Val);
3502 bool IsArgInitViaCast =
3503 isa<CastInst>(Val) &&
3504 isa<Argument>(cast<CastInst>(Val)->getOperand(0)) &&
3505 // Check that the cast appears directly before the store. Otherwise
3506 // moving the cast before InsBefore may break the IR.
3507 Val == It->getPrevNode();
3508 bool IsArgInit = IsDirectArgInit || IsArgInitViaCast;
3509 if (!IsArgInit)
3510 continue;
3511
3512 if (IsArgInitViaCast)
3513 InitInsts.push_back(cast<Instruction>(Val));
3514 InitInsts.push_back(Store);
3515 continue;
3516 }
3517
3518 // Do not reorder past unknown instructions: argument initialization should
3519 // only involve casts and stores.
3520 return;
3521 }
3522}
3523
3525 // Alloca could have been renamed for uniqueness. Its true name will have been
3526 // recorded as an annotation.
3527 if (AI->hasMetadata(LLVMContext::MD_annotation)) {
3528 MDTuple *AllocaAnnotations =
3529 cast<MDTuple>(AI->getMetadata(LLVMContext::MD_annotation));
3530 for (auto &Annotation : AllocaAnnotations->operands()) {
3531 if (!isa<MDTuple>(Annotation))
3532 continue;
3533 auto AnnotationTuple = cast<MDTuple>(Annotation);
3534 for (unsigned Index = 0; Index < AnnotationTuple->getNumOperands();
3535 Index++) {
3536 // All annotations are strings
3537 auto MetadataString =
3538 cast<MDString>(AnnotationTuple->getOperand(Index));
3539 if (MetadataString->getString() == "alloca_name_altered")
3540 return cast<MDString>(AnnotationTuple->getOperand(Index + 1))
3541 ->getString();
3542 }
3543 }
3544 }
3545 return AI->getName();
3546}
3547
3548void FunctionStackPoisoner::processStaticAllocas() {
3549 if (AllocaVec.empty()) {
3550 assert(StaticAllocaPoisonCallVec.empty());
3551 return;
3552 }
3553
3554 int StackMallocIdx = -1;
3555 DebugLoc EntryDebugLocation;
3556 if (auto SP = F.getSubprogram())
3557 EntryDebugLocation =
3558 DILocation::get(SP->getContext(), SP->getScopeLine(), 0, SP);
3559
3560 Instruction *InsBefore = AllocaVec[0];
3561 IRBuilder<> IRB(InsBefore);
3562
3563 // Make sure non-instrumented allocas stay in the entry block. Otherwise,
3564 // debug info is broken, because only entry-block allocas are treated as
3565 // regular stack slots.
3566 auto InsBeforeB = InsBefore->getParent();
3567 assert(InsBeforeB == &F.getEntryBlock());
3568 for (auto *AI : StaticAllocasToMoveUp)
3569 if (AI->getParent() == InsBeforeB)
3570 AI->moveBefore(InsBefore->getIterator());
3571
3572 // Move stores of arguments into entry-block allocas as well. This prevents
3573 // extra stack slots from being generated (to house the argument values until
3574 // they can be stored into the allocas). This also prevents uninitialized
3575 // values from being shown in backtraces.
3576 SmallVector<Instruction *, 8> ArgInitInsts;
3577 findStoresToUninstrumentedArgAllocas(ASan, *InsBefore, ArgInitInsts);
3578 for (Instruction *ArgInitInst : ArgInitInsts)
3579 ArgInitInst->moveBefore(InsBefore->getIterator());
3580
3581 // If we have a call to llvm.localescape, keep it in the entry block.
3582 if (LocalEscapeCall)
3583 LocalEscapeCall->moveBefore(InsBefore->getIterator());
3584
3586 SVD.reserve(AllocaVec.size());
3587 for (AllocaInst *AI : AllocaVec) {
3590 ASan.getAllocaSizeInBytes(*AI),
3591 0,
3592 AI->getAlign().value(),
3593 AI,
3594 0,
3595 0};
3596 SVD.push_back(D);
3597 }
3598
3599 // Minimal header size (left redzone) is 4 pointers,
3600 // i.e. 32 bytes on 64-bit platforms and 16 bytes in 32-bit platforms.
3601 uint64_t Granularity = 1ULL << Mapping.Scale;
3602 uint64_t MinHeaderSize = std::max((uint64_t)ASan.LongSize / 2, Granularity);
3603 const ASanStackFrameLayout &L =
3604 ComputeASanStackFrameLayout(SVD, Granularity, MinHeaderSize);
3605
3606 // Build AllocaToSVDMap for ASanStackVariableDescription lookup.
3608 for (auto &Desc : SVD)
3609 AllocaToSVDMap[Desc.AI] = &Desc;
3610
3611 // Update SVD with information from lifetime intrinsics.
3612 for (const auto &APC : StaticAllocaPoisonCallVec) {
3613 assert(APC.InsBefore);
3614 assert(APC.AI);
3615 assert(ASan.isInterestingAlloca(*APC.AI));
3616 assert(APC.AI->isStaticAlloca());
3617
3618 ASanStackVariableDescription &Desc = *AllocaToSVDMap[APC.AI];
3619 Desc.LifetimeSize = Desc.Size;
3620 if (const DILocation *FnLoc = EntryDebugLocation.get()) {
3621 if (const DILocation *LifetimeLoc = APC.InsBefore->getDebugLoc().get()) {
3622 if (LifetimeLoc->getFile() == FnLoc->getFile())
3623 if (unsigned Line = LifetimeLoc->getLine())
3624 Desc.Line = std::min(Desc.Line ? Desc.Line : Line, Line);
3625 }
3626 }
3627 }
3628
3629 auto DescriptionString = ComputeASanStackFrameDescription(SVD);
3630 LLVM_DEBUG(dbgs() << DescriptionString << " --- " << L.FrameSize << "\n");
3631 uint64_t LocalStackSize = L.FrameSize;
3632 bool DoStackMalloc =
3633 ASan.UseAfterReturn != AsanDetectStackUseAfterReturnMode::Never &&
3634 !ASan.CompileKernel && LocalStackSize <= kMaxStackMallocSize;
3635 bool DoDynamicAlloca = ClDynamicAllocaStack;
3636 // Don't do dynamic alloca or stack malloc if:
3637 // 1) There is inline asm: too often it makes assumptions on which registers
3638 // are available.
3639 // 2) There is a returns_twice call (typically setjmp), which is
3640 // optimization-hostile, and doesn't play well with introduced indirect
3641 // register-relative calculation of local variable addresses.
3642 DoDynamicAlloca &= !HasInlineAsm && !HasReturnsTwiceCall;
3643 DoStackMalloc &= !HasInlineAsm && !HasReturnsTwiceCall;
3644
3645 Type *PtrTy = F.getDataLayout().getAllocaPtrType(F.getContext());
3646 Value *StaticAlloca =
3647 DoDynamicAlloca ? nullptr : createAllocaForLayout(IRB, L, false);
3648
3649 Value *FakeStackPtr;
3650 Value *FakeStackInt;
3651 Value *LocalStackBase;
3652 Value *LocalStackBaseAlloca;
3653 uint8_t DIExprFlags = DIExpression::ApplyOffset;
3654
3655 if (DoStackMalloc) {
3656 LocalStackBaseAlloca =
3657 IRB.CreateAlloca(IntptrTy, nullptr, "asan_local_stack_base");
3658 if (ASan.UseAfterReturn == AsanDetectStackUseAfterReturnMode::Runtime) {
3659 // void *FakeStack = __asan_option_detect_stack_use_after_return
3660 // ? __asan_stack_malloc_N(LocalStackSize)
3661 // : nullptr;
3662 // void *LocalStackBase = (FakeStack) ? FakeStack :
3663 // alloca(LocalStackSize);
3664 Constant *OptionDetectUseAfterReturn = F.getParent()->getOrInsertGlobal(
3666 Value *UseAfterReturnIsEnabled = IRB.CreateICmpNE(
3667 IRB.CreateLoad(IRB.getInt32Ty(), OptionDetectUseAfterReturn),
3669 Instruction *Term =
3670 SplitBlockAndInsertIfThen(UseAfterReturnIsEnabled, InsBefore, false);
3671 IRBuilder<> IRBIf(Term);
3672 StackMallocIdx = StackMallocSizeClass(LocalStackSize);
3673 assert(StackMallocIdx <= kMaxAsanStackMallocSizeClass);
3674 Value *FakeStackValue =
3675 RTCI.createRuntimeCall(IRBIf, AsanStackMallocFunc[StackMallocIdx],
3676 ConstantInt::get(IntptrTy, LocalStackSize));
3677 IRB.SetInsertPoint(InsBefore);
3678 FakeStackInt = createPHI(IRB, UseAfterReturnIsEnabled, FakeStackValue,
3679 Term, ConstantInt::get(IntptrTy, 0));
3680 } else {
3681 // assert(ASan.UseAfterReturn == AsanDetectStackUseAfterReturnMode:Always)
3682 // void *FakeStack = __asan_stack_malloc_N(LocalStackSize);
3683 // void *LocalStackBase = (FakeStack) ? FakeStack :
3684 // alloca(LocalStackSize);
3685 StackMallocIdx = StackMallocSizeClass(LocalStackSize);
3686 FakeStackInt =
3687 RTCI.createRuntimeCall(IRB, AsanStackMallocFunc[StackMallocIdx],
3688 ConstantInt::get(IntptrTy, LocalStackSize));
3689 }
3690 FakeStackPtr = IRB.CreateIntToPtr(FakeStackInt, PtrTy);
3691 Value *NoFakeStack =
3692 IRB.CreateICmpEQ(FakeStackInt, Constant::getNullValue(IntptrTy));
3693 Instruction *Term =
3694 SplitBlockAndInsertIfThen(NoFakeStack, InsBefore, false);
3695 IRBuilder<> IRBIf(Term);
3696 Value *AllocaValue =
3697 DoDynamicAlloca ? createAllocaForLayout(IRBIf, L, true) : StaticAlloca;
3698
3699 IRB.SetInsertPoint(InsBefore);
3700 LocalStackBase =
3701 createPHI(IRB, NoFakeStack, AllocaValue, Term, FakeStackPtr);
3702 IRB.CreateStore(LocalStackBase, LocalStackBaseAlloca);
3703 DIExprFlags |= DIExpression::DerefBefore;
3704 } else {
3705 // void *FakeStack = nullptr;
3706 // void *LocalStackBase = alloca(LocalStackSize);
3707 FakeStackInt = Constant::getNullValue(IntptrTy);
3708 FakeStackPtr = Constant::getNullValue(PtrTy);
3709 LocalStackBase =
3710 DoDynamicAlloca ? createAllocaForLayout(IRB, L, true) : StaticAlloca;
3711 LocalStackBaseAlloca = LocalStackBase;
3712 }
3713
3714 // Replace Alloca instructions with base+offset.
3715 SmallVector<Value *> NewAllocaPtrs;
3716 for (const auto &Desc : SVD) {
3717 AllocaInst *AI = Desc.AI;
3718 replaceDbgDeclare(AI, LocalStackBaseAlloca, DIB, DIExprFlags, Desc.Offset);
3719 Value *NewAllocaPtr = IRB.CreatePtrAdd(
3720 LocalStackBase, ConstantInt::get(IntptrTy, Desc.Offset));
3721 AI->replaceAllUsesWith(NewAllocaPtr);
3722 NewAllocaPtrs.push_back(NewAllocaPtr);
3723 }
3724
3725 // The left-most redzone has enough space for at least 4 pointers.
3726 // Write the Magic value to redzone[0].
3727 IRB.CreateStore(ConstantInt::get(IntptrTy, kCurrentStackFrameMagic),
3728 LocalStackBase);
3729 // Write the frame description constant to redzone[1].
3730 Value *BasePlus1 = IRB.CreatePtrAdd(
3731 LocalStackBase, ConstantInt::get(IntptrTy, ASan.LongSize / 8));
3732 GlobalVariable *StackDescriptionGlobal =
3733 createPrivateGlobalForString(*F.getParent(), DescriptionString,
3734 /*AllowMerging*/ true, genName("stack"));
3735 Value *Description = IRB.CreatePointerCast(StackDescriptionGlobal, IntptrTy);
3736 IRB.CreateStore(Description, BasePlus1);
3737 // Write the PC to redzone[2].
3738 Value *BasePlus2 = IRB.CreatePtrAdd(
3739 LocalStackBase, ConstantInt::get(IntptrTy, 2 * ASan.LongSize / 8));
3740 IRB.CreateStore(IRB.CreatePointerCast(&F, IntptrTy), BasePlus2);
3741
3742 const auto &ShadowAfterScope = GetShadowBytesAfterScope(SVD, L);
3743
3744 // Poison the stack red zones at the entry.
3745 Value *ShadowBase =
3746 ASan.memToShadow(IRB.CreatePtrToInt(LocalStackBase, IntptrTy), IRB);
3747 // As mask we must use most poisoned case: red zones and after scope.
3748 // As bytes we can use either the same or just red zones only.
3749 copyToShadow(ShadowAfterScope, ShadowAfterScope, IRB, ShadowBase);
3750
3751 if (!StaticAllocaPoisonCallVec.empty()) {
3752 const auto &ShadowInScope = GetShadowBytes(SVD, L);
3753
3754 // Poison static allocas near lifetime intrinsics.
3755 for (const auto &APC : StaticAllocaPoisonCallVec) {
3756 const ASanStackVariableDescription &Desc = *AllocaToSVDMap[APC.AI];
3757 assert(Desc.Offset % L.Granularity == 0);
3758 size_t Begin = Desc.Offset / L.Granularity;
3759 size_t End = Begin + (APC.Size + L.Granularity - 1) / L.Granularity;
3760
3761 IRBuilder<> IRB(APC.InsBefore);
3762 copyToShadow(ShadowAfterScope,
3763 APC.DoPoison ? ShadowAfterScope : ShadowInScope, Begin, End,
3764 IRB, ShadowBase);
3765 }
3766 }
3767
3768 // Remove lifetime markers now that these are no longer allocas.
3769 for (Value *NewAllocaPtr : NewAllocaPtrs) {
3770 for (User *U : make_early_inc_range(NewAllocaPtr->users())) {
3771 auto *I = cast<Instruction>(U);
3772 if (I->isLifetimeStartOrEnd())
3773 I->eraseFromParent();
3774 }
3775 }
3776
3777 SmallVector<uint8_t, 64> ShadowClean(ShadowAfterScope.size(), 0);
3778 SmallVector<uint8_t, 64> ShadowAfterReturn;
3779
3780 // (Un)poison the stack before all ret instructions.
3781 for (Instruction *Ret : RetVec) {
3782 IRBuilder<> IRBRet(Ret);
3783 // Mark the current frame as retired.
3784 IRBRet.CreateStore(ConstantInt::get(IntptrTy, kRetiredStackFrameMagic),
3785 LocalStackBase);
3786 if (DoStackMalloc) {
3787 assert(StackMallocIdx >= 0);
3788 // if FakeStack != 0 // LocalStackBase == FakeStack
3789 // // In use-after-return mode, poison the whole stack frame.
3790 // if StackMallocIdx <= 4
3791 // // For small sizes inline the whole thing:
3792 // memset(ShadowBase, kAsanStackAfterReturnMagic, ShadowSize);
3793 // **SavedFlagPtr(FakeStack) = 0
3794 // else
3795 // __asan_stack_free_N(FakeStack, LocalStackSize)
3796 // else
3797 // <This is not a fake stack; unpoison the redzones>
3798 Value *Cmp =
3799 IRBRet.CreateICmpNE(FakeStackInt, Constant::getNullValue(IntptrTy));
3800 Instruction *ThenTerm, *ElseTerm;
3801 SplitBlockAndInsertIfThenElse(Cmp, Ret, &ThenTerm, &ElseTerm);
3802
3803 IRBuilder<> IRBPoison(ThenTerm);
3804 if (ASan.MaxInlinePoisoningSize != 0 && StackMallocIdx <= 4) {
3805 int ClassSize = kMinStackMallocSize << StackMallocIdx;
3806 ShadowAfterReturn.resize(ClassSize / L.Granularity,
3808 copyToShadow(ShadowAfterReturn, ShadowAfterReturn, IRBPoison,
3809 ShadowBase);
3810 Value *SavedFlagPtrPtr = IRBPoison.CreatePtrAdd(
3811 FakeStackPtr,
3812 ConstantInt::get(IntptrTy, ClassSize - ASan.LongSize / 8));
3813 Value *SavedFlagPtr = IRBPoison.CreateLoad(IntptrTy, SavedFlagPtrPtr);
3814 IRBPoison.CreateStore(
3815 Constant::getNullValue(IRBPoison.getInt8Ty()),
3816 IRBPoison.CreateIntToPtr(SavedFlagPtr, IRBPoison.getPtrTy()));
3817 } else {
3818 // For larger frames call __asan_stack_free_*.
3819 RTCI.createRuntimeCall(
3820 IRBPoison, AsanStackFreeFunc[StackMallocIdx],
3821 {FakeStackInt, ConstantInt::get(IntptrTy, LocalStackSize)});
3822 }
3823
3824 IRBuilder<> IRBElse(ElseTerm);
3825 copyToShadow(ShadowAfterScope, ShadowClean, IRBElse, ShadowBase);
3826 } else {
3827 copyToShadow(ShadowAfterScope, ShadowClean, IRBRet, ShadowBase);
3828 }
3829 }
3830
3831 // We are done. Remove the old unused alloca instructions.
3832 for (auto *AI : AllocaVec)
3833 AI->eraseFromParent();
3834}
3835
3836void FunctionStackPoisoner::poisonAlloca(Value *V, uint64_t Size,
3837 IRBuilder<> &IRB, bool DoPoison) {
3838 // For now just insert the call to ASan runtime.
3839 Value *AddrArg = IRB.CreatePointerCast(V, IntptrTy);
3840 Value *SizeArg = ConstantInt::get(IntptrTy, Size);
3841 RTCI.createRuntimeCall(
3842 IRB, DoPoison ? AsanPoisonStackMemoryFunc : AsanUnpoisonStackMemoryFunc,
3843 {AddrArg, SizeArg});
3844}
3845
3846// Handling llvm.lifetime intrinsics for a given %alloca:
3847// (1) collect all llvm.lifetime.xxx(%size, %value) describing the alloca.
3848// (2) if %size is constant, poison memory for llvm.lifetime.end (to detect
3849// invalid accesses) and unpoison it for llvm.lifetime.start (the memory
3850// could be poisoned by previous llvm.lifetime.end instruction, as the
3851// variable may go in and out of scope several times, e.g. in loops).
3852// (3) if we poisoned at least one %alloca in a function,
3853// unpoison the whole stack frame at function exit.
3854void FunctionStackPoisoner::handleDynamicAllocaCall(AllocaInst *AI) {
3855 IRBuilder<> IRB(AI);
3856
3857 const Align Alignment = std::max(Align(kAllocaRzSize), AI->getAlign());
3858 const uint64_t AllocaRedzoneMask = kAllocaRzSize - 1;
3859
3860 Value *Zero = Constant::getNullValue(IntptrTy);
3861 Value *AllocaRzSize = ConstantInt::get(IntptrTy, kAllocaRzSize);
3862 Value *AllocaRzMask = ConstantInt::get(IntptrTy, AllocaRedzoneMask);
3863
3864 // Since we need to extend alloca with additional memory to locate
3865 // redzones, and OldSize is number of allocated blocks with
3866 // ElementSize size, get allocated memory size in bytes by
3867 // OldSize * ElementSize.
3868 Value *OldSize = IRB.CreateAllocationSize(IntptrTy, AI);
3869
3870 // PartialSize = OldSize % 32
3871 Value *PartialSize = IRB.CreateAnd(OldSize, AllocaRzMask);
3872
3873 // Misalign = kAllocaRzSize - PartialSize;
3874 Value *Misalign = IRB.CreateSub(AllocaRzSize, PartialSize);
3875
3876 // PartialPadding = Misalign != kAllocaRzSize ? Misalign : 0;
3877 Value *Cond = IRB.CreateICmpNE(Misalign, AllocaRzSize);
3878 Value *PartialPadding = IRB.CreateSelect(Cond, Misalign, Zero);
3879
3880 // AdditionalChunkSize = Alignment + PartialPadding + kAllocaRzSize
3881 // Alignment is added to locate left redzone, PartialPadding for possible
3882 // partial redzone and kAllocaRzSize for right redzone respectively.
3883 Value *AdditionalChunkSize = IRB.CreateAdd(
3884 ConstantInt::get(IntptrTy, Alignment.value() + kAllocaRzSize),
3885 PartialPadding);
3886
3887 Value *NewSize = IRB.CreateAdd(OldSize, AdditionalChunkSize);
3888
3889 // Insert new alloca with new NewSize and Alignment params.
3890 AllocaInst *NewAlloca = IRB.CreateAlloca(IRB.getInt8Ty(), NewSize);
3891 NewAlloca->setAlignment(Alignment);
3892
3893 // NewAddress = Address + Alignment
3894 Value *NewAddress =
3895 IRB.CreateAdd(IRB.CreatePtrToInt(NewAlloca, IntptrTy),
3896 ConstantInt::get(IntptrTy, Alignment.value()));
3897
3898 // Insert __asan_alloca_poison call for new created alloca.
3899 RTCI.createRuntimeCall(IRB, AsanAllocaPoisonFunc, {NewAddress, OldSize});
3900
3901 // Store the last alloca's address to DynamicAllocaLayout. We'll need this
3902 // for unpoisoning stuff.
3903 IRB.CreateStore(IRB.CreatePtrToInt(NewAlloca, IntptrTy), DynamicAllocaLayout);
3904
3905 Value *NewAddressPtr = IRB.CreateIntToPtr(NewAddress, AI->getType());
3906
3907 // Remove lifetime markers now that this is no longer an alloca.
3908 for (User *U : make_early_inc_range(AI->users())) {
3909 auto *I = cast<Instruction>(U);
3910 if (I->isLifetimeStartOrEnd())
3911 I->eraseFromParent();
3912 }
3913
3914 // Replace all uses of AddressReturnedByAlloca with NewAddressPtr.
3915 AI->replaceAllUsesWith(NewAddressPtr);
3916
3917 // We are done. Erase old alloca from parent.
3918 AI->eraseFromParent();
3919}
3920
3921// isSafeAccess returns true if Addr is always inbounds with respect to its
3922// base object. For example, it is a field access or an array access with
3923// constant inbounds index.
3924bool AddressSanitizer::isSafeAccess(ObjectSizeOffsetVisitor &ObjSizeVis,
3925 Value *Addr, TypeSize TypeStoreSize) const {
3926 if (TypeStoreSize.isScalable())
3927 // TODO: We can use vscale_range to convert a scalable value to an
3928 // upper bound on the access size.
3929 return false;
3930
3931 SizeOffsetAPInt SizeOffset = ObjSizeVis.compute(Addr);
3932 if (!SizeOffset.bothKnown())
3933 return false;
3934
3935 uint64_t Size = SizeOffset.Size.getZExtValue();
3936 int64_t Offset = SizeOffset.Offset.getSExtValue();
3937
3938 // Three checks are required to ensure safety:
3939 // . Offset >= 0 (since the offset is given from the base ptr)
3940 // . Size >= Offset (unsigned)
3941 // . Size - Offset >= NeededSize (unsigned)
3942 return Offset >= 0 && Size >= uint64_t(Offset) &&
3943 Size - uint64_t(Offset) >= TypeStoreSize / 8;
3944}
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static cl::opt< bool > ClUseStackSafety("stack-tagging-use-stack-safety", cl::Hidden, cl::init(true), cl::desc("Use Stack Safety analysis results"))
Rewrite undef for PHI
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
static void findStoresToUninstrumentedArgAllocas(AddressSanitizer &ASan, Instruction &InsBefore, SmallVectorImpl< Instruction * > &InitInsts)
Collect instructions in the entry block after InsBefore which initialize permanent storage for a func...
static void doInstrumentAddress(AddressSanitizer *Pass, Instruction *I, Instruction *InsertBefore, Value *Addr, MaybeAlign Alignment, unsigned Granularity, TypeSize TypeStoreSize, bool IsWrite, Value *SizeArgument, bool UseCalls, uint32_t Exp, RuntimeCallInserter &RTCI)
static const uint64_t kDefaultShadowScale
const char kAMDGPUUnreachableName[]
constexpr size_t kAccessSizeIndexMask
static cl::opt< int > ClDebugMin("asan-debug-min", cl::desc("Debug min inst"), cl::Hidden, cl::init(-1))
static cl::opt< bool > ClUsePrivateAlias("asan-use-private-alias", cl::desc("Use private aliases for global variables"), cl::Hidden, cl::init(true))
static const uint64_t kPS_ShadowOffset64
static const uint64_t kFreeBSD_ShadowOffset32
constexpr size_t kIsWriteShift
static const uint64_t kSmallX86_64ShadowOffsetAlignMask
static bool isInterestingPointerSubtraction(Instruction *I)
const char kAMDGPUAddressSharedName[]
const char kAsanStackFreeNameTemplate[]
constexpr size_t kCompileKernelMask
static cl::opt< bool > ClForceDynamicShadow("asan-force-dynamic-shadow", cl::desc("Load shadow address into a local variable for each function"), cl::Hidden, cl::init(false))
const char kAsanOptionDetectUseAfterReturn[]
static cl::opt< std::string > ClMemoryAccessCallbackPrefix("asan-memory-access-callback-prefix", cl::desc("Prefix for memory access callbacks"), cl::Hidden, cl::init("__asan_"))
static const uint64_t kRISCV64_ShadowOffset64
static cl::opt< bool > ClInsertVersionCheck("asan-guard-against-version-mismatch", cl::desc("Guard against compiler/runtime version mismatch."), cl::Hidden, cl::init(true))
const char kAsanSetShadowPrefix[]
static cl::opt< AsanDtorKind > ClOverrideDestructorKind("asan-destructor-kind", cl::desc("Sets the ASan destructor kind. The default is to use the value " "provided to the pass constructor"), cl::values(clEnumValN(AsanDtorKind::None, "none", "No destructors"), clEnumValN(AsanDtorKind::Global, "global", "Use global destructors")), cl::init(AsanDtorKind::Invalid), cl::Hidden)
static Twine genName(StringRef suffix)
static cl::opt< bool > ClInstrumentWrites("asan-instrument-writes", cl::desc("instrument write instructions"), cl::Hidden, cl::init(true))
const char kAsanPtrCmp[]
static uint64_t GetCtorAndDtorPriority(Triple &TargetTriple)
const char kAsanStackMallocNameTemplate[]
static cl::opt< bool > ClInstrumentByval("asan-instrument-byval", cl::desc("instrument byval call arguments"), cl::Hidden, cl::init(true))
const char kAsanInitName[]
static cl::opt< bool > ClGlobals("asan-globals", cl::desc("Handle global objects"), cl::Hidden, cl::init(true))
static cl::opt< bool > ClRedzoneByvalArgs("asan-redzone-byval-args", cl::desc("Create redzones for byval " "arguments (extra copy " "required)"), cl::Hidden, cl::init(true))
static const uint64_t kWindowsShadowOffset64
const char kAsanGenPrefix[]
constexpr size_t kIsWriteMask
static uint64_t getRedzoneSizeForScale(int MappingScale)
static const uint64_t kDefaultShadowOffset64
static cl::opt< bool > ClOptimizeCallbacks("asan-optimize-callbacks", cl::desc("Optimize callbacks"), cl::Hidden, cl::init(false))
const char kAsanUnregisterGlobalsName[]
static const uint64_t kAsanCtorAndDtorPriority
const char kAsanUnpoisonGlobalsName[]
static cl::opt< bool > ClWithIfuncSuppressRemat("asan-with-ifunc-suppress-remat", cl::desc("Suppress rematerialization of dynamic shadow address by passing " "it through inline asm in prologue."), cl::Hidden, cl::init(true))
static cl::opt< int > ClDebugStack("asan-debug-stack", cl::desc("debug stack"), cl::Hidden, cl::init(0))
const char kAsanUnregisterElfGlobalsName[]
static bool isUnsupportedAMDGPUAddrspace(Value *Addr)
const char kAsanRegisterImageGlobalsName[]
static const uint64_t kWebAssemblyShadowOffset
static cl::opt< bool > ClOpt("asan-opt", cl::desc("Optimize instrumentation"), cl::Hidden, cl::init(true))
static const uint64_t kAllocaRzSize
const char kODRGenPrefix[]
static const uint64_t kSystemZ_ShadowOffset64
static const uint64_t kDefaultShadowOffset32
const char kAsanShadowMemoryDynamicAddress[]
static cl::opt< bool > ClUseOdrIndicator("asan-use-odr-indicator", cl::desc("Use odr indicators to improve ODR reporting"), cl::Hidden, cl::init(true))
static bool GlobalWasGeneratedByCompiler(GlobalVariable *G)
Check if G has been created by a trusted compiler pass.
const char kAsanStackMallocAlwaysNameTemplate[]
static cl::opt< int > ClShadowAddrSpace("asan-shadow-addr-space", cl::desc("Address space for pointers to the shadow map"), cl::Hidden, cl::init(0))
static cl::opt< bool > ClInvalidPointerCmp("asan-detect-invalid-pointer-cmp", cl::desc("Instrument <, <=, >, >= with pointer operands"), cl::Hidden, cl::init(false))
static const uint64_t kAsanEmscriptenCtorAndDtorPriority
static cl::opt< int > ClInstrumentationWithCallsThreshold("asan-instrumentation-with-call-threshold", cl::desc("If the function being instrumented contains more than " "this number of memory accesses, use callbacks instead of " "inline checks (-1 means never use callbacks)."), cl::Hidden, cl::init(7000))
static cl::opt< int > ClDebugMax("asan-debug-max", cl::desc("Debug max inst"), cl::Hidden, cl::init(-1))
static cl::opt< bool > ClInvalidPointerSub("asan-detect-invalid-pointer-sub", cl::desc("Instrument - operations with pointer operands"), cl::Hidden, cl::init(false))
static const uint64_t kFreeBSD_ShadowOffset64
static cl::opt< uint32_t > ClForceExperiment("asan-force-experiment", cl::desc("Force optimization experiment (for testing)"), cl::Hidden, cl::init(0))
const char kSanCovGenPrefix[]
static const uint64_t kFreeBSDKasan_ShadowOffset64
const char kAsanModuleDtorName[]
static const uint64_t kDynamicShadowSentinel
static bool isInterestingPointerComparison(Instruction *I)
static cl::list< unsigned > ClAddrSpaces("asan-instrument-address-spaces", cl::desc("Only instrument variables in the specified address spaces."), cl::Hidden, cl::CommaSeparated, cl::callback([](const unsigned &AddrSpace) { SrcAddrSpaces.insert(AddrSpace);}))
static cl::opt< bool > ClStack("asan-stack", cl::desc("Handle stack memory"), cl::Hidden, cl::init(true))
static const uint64_t kMIPS64_ShadowOffset64
static const uint64_t kLinuxKasan_ShadowOffset64
static int StackMallocSizeClass(uint64_t LocalStackSize)
static cl::opt< uint32_t > ClMaxInlinePoisoningSize("asan-max-inline-poisoning-size", cl::desc("Inline shadow poisoning for blocks up to the given size in bytes."), cl::Hidden, cl::init(64))
static cl::opt< bool > ClInstrumentAtomics("asan-instrument-atomics", cl::desc("instrument atomic instructions (rmw, cmpxchg)"), cl::Hidden, cl::init(true))
static cl::opt< bool > ClUseAfterScope("asan-use-after-scope", cl::desc("Check stack-use-after-scope"), cl::Hidden, cl::init(false))
constexpr size_t kAccessSizeIndexShift
static cl::opt< int > ClMappingScale("asan-mapping-scale", cl::desc("scale of asan shadow mapping"), cl::Hidden, cl::init(0))
const char kAsanPoisonStackMemoryName[]
static cl::opt< bool > ClEnableKasan("asan-kernel", cl::desc("Enable KernelAddressSanitizer instrumentation"), cl::Hidden, cl::init(false))
static cl::opt< std::string > ClDebugFunc("asan-debug-func", cl::Hidden, cl::desc("Debug func"))
static bool isSupportedAddrspace(const Triple &TargetTriple, Value *Addr)
static cl::opt< bool > ClUseGlobalsGC("asan-globals-live-support", cl::desc("Use linker features to support dead " "code stripping of globals"), cl::Hidden, cl::init(true))
static const size_t kNumberOfAccessSizes
const char kAsanUnpoisonStackMemoryName[]
static const uint64_t kLoongArch64_ShadowOffset64
const char kAsanRegisterGlobalsName[]
static cl::opt< bool > ClInstrumentDynamicAllocas("asan-instrument-dynamic-allocas", cl::desc("instrument dynamic allocas"), cl::Hidden, cl::init(true))
const char kAsanModuleCtorName[]
const char kAsanGlobalsRegisteredFlagName[]
static const size_t kMaxStackMallocSize
static cl::opt< bool > ClRecover("asan-recover", cl::desc("Enable recovery mode (continue-after-error)."), cl::Hidden, cl::init(false))
static cl::opt< bool > ClOptSameTemp("asan-opt-same-temp", cl::desc("Instrument the same temp just once"), cl::Hidden, cl::init(true))
static cl::opt< bool > ClDynamicAllocaStack("asan-stack-dynamic-alloca", cl::desc("Use dynamic alloca to represent stack variables"), cl::Hidden, cl::init(true))
static cl::opt< bool > ClOptStack("asan-opt-stack", cl::desc("Don't instrument scalar stack variables"), cl::Hidden, cl::init(false))
static const uint64_t kMIPS_ShadowOffsetN32
const char kAsanUnregisterImageGlobalsName[]
static cl::opt< AsanDetectStackUseAfterReturnMode > ClUseAfterReturn("asan-use-after-return", cl::desc("Sets the mode of detection for stack-use-after-return."), cl::values(clEnumValN(AsanDetectStackUseAfterReturnMode::Never, "never", "Never detect stack use after return."), clEnumValN(AsanDetectStackUseAfterReturnMode::Runtime, "runtime", "Detect stack use after return if " "binary flag 'ASAN_OPTIONS=detect_stack_use_after_return' is set."), clEnumValN(AsanDetectStackUseAfterReturnMode::Always, "always", "Always detect stack use after return.")), cl::Hidden, cl::init(AsanDetectStackUseAfterReturnMode::Runtime))
static cl::opt< bool > ClOptGlobals("asan-opt-globals", cl::desc("Don't instrument scalar globals"), cl::Hidden, cl::init(true))
static const uintptr_t kCurrentStackFrameMagic
static ShadowMapping getShadowMapping(const Triple &TargetTriple, int LongSize, bool IsKasan)
static const uint64_t kPPC64_ShadowOffset64
static cl::opt< AsanCtorKind > ClConstructorKind("asan-constructor-kind", cl::desc("Sets the ASan constructor kind"), cl::values(clEnumValN(AsanCtorKind::None, "none", "No constructors"), clEnumValN(AsanCtorKind::Global, "global", "Use global constructors")), cl::init(AsanCtorKind::Global), cl::Hidden)
static const int kMaxAsanStackMallocSizeClass
static const uint64_t kMIPS32_ShadowOffset32
static cl::opt< bool > ClAlwaysSlowPath("asan-always-slow-path", cl::desc("use instrumentation with slow path for all accesses"), cl::Hidden, cl::init(false))
static const uint64_t kNetBSD_ShadowOffset32
static const uint64_t kFreeBSDAArch64_ShadowOffset64
static const uint64_t kSmallX86_64ShadowOffsetBase
static cl::opt< bool > ClInitializers("asan-initialization-order", cl::desc("Handle C++ initializer order"), cl::Hidden, cl::init(true))
static const uint64_t kNetBSD_ShadowOffset64
const char kAsanPtrSub[]
static cl::opt< unsigned > ClRealignStack("asan-realign-stack", cl::desc("Realign stack to the value of this flag (power of two)"), cl::Hidden, cl::init(32))
static const uint64_t kWindowsShadowOffset32
static cl::opt< bool > ClInstrumentReads("asan-instrument-reads", cl::desc("instrument read instructions"), cl::Hidden, cl::init(true))
static size_t TypeStoreSizeToSizeIndex(uint32_t TypeSize)
const char kAsanAllocaPoison[]
constexpr size_t kCompileKernelShift
static SmallSet< unsigned, 8 > SrcAddrSpaces
static cl::opt< bool > ClWithIfunc("asan-with-ifunc", cl::desc("Access dynamic shadow through an ifunc global on " "platforms that support this"), cl::Hidden, cl::init(true))
static cl::opt< bool > ClKasanMemIntrinCallbackPrefix("asan-kernel-mem-intrinsic-prefix", cl::desc("Use prefix for memory intrinsics in KASAN mode"), cl::Hidden, cl::init(false))
const char kAsanVersionCheckNamePrefix[]
const char kAMDGPUAddressPrivateName[]
static const uint64_t kNetBSDKasan_ShadowOffset64
const char kAMDGPUBallotName[]
const char kAsanRegisterElfGlobalsName[]
static cl::opt< uint64_t > ClMappingOffset("asan-mapping-offset", cl::desc("offset of asan shadow mapping [EXPERIMENTAL]"), cl::Hidden, cl::init(0))
const char kAsanReportErrorTemplate[]
static cl::opt< bool > ClWithComdat("asan-with-comdat", cl::desc("Place ASan constructors in comdat sections"), cl::Hidden, cl::init(true))
static StringRef getAllocaName(AllocaInst *AI)
static cl::opt< bool > ClSkipPromotableAllocas("asan-skip-promotable-allocas", cl::desc("Do not instrument promotable allocas"), cl::Hidden, cl::init(true))
static cl::opt< int > ClMaxInsnsToInstrumentPerBB("asan-max-ins-per-bb", cl::init(10000), cl::desc("maximal number of instructions to instrument in any given BB"), cl::Hidden)
static const uintptr_t kRetiredStackFrameMagic
static cl::opt< bool > ClUseStackSafety("asan-use-stack-safety", cl::Hidden, cl::init(true), cl::Hidden, cl::desc("Use Stack Safety analysis results"), cl::Optional)
const char kAsanPoisonGlobalsName[]
const char kAsanHandleNoReturnName[]
static const size_t kMinStackMallocSize
static cl::opt< int > ClDebug("asan-debug", cl::desc("debug"), cl::Hidden, cl::init(0))
const char kAsanAllocasUnpoison[]
static const uint64_t kAArch64_ShadowOffset64
static cl::opt< bool > ClInvalidPointerPairs("asan-detect-invalid-pointer-pair", cl::desc("Instrument <, <=, >, >=, - with pointer operands"), cl::Hidden, cl::init(false))
Function Alias Analysis false
This file contains the simple types necessary to represent the attributes associated with functions a...
static bool isPointerOperand(Value *I, User *U)
static const Function * getParent(const Value *V)
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
static GCRegistry::Add< StatepointGC > D("statepoint-example", "an example strategy for statepoint")
#define clEnumValN(ENUMVAL, FLAGNAME, DESC)
This file contains the declarations for the subclasses of Constant, which represent the different fla...
DXIL Finalize Linkage
dxil translate DXIL Translate Metadata
This file defines the DenseMap class.
This file builds on the ADT/GraphTraits.h file to build generic depth first graph iterator.
static bool runOnFunction(Function &F, bool PostInlining)
This is the interface for a simple mod/ref and alias analysis over globals.
IRTranslator LLVM IR MI
Module.h This file contains the declarations for the Module class.
This defines the Use class.
std::pair< Instruction::BinaryOps, Value * > OffsetOp
Find all possible pairs (BinOp, RHS) that BinOp V, RHS can be simplified.
static bool isZero(Value *V, const DataLayout &DL, DominatorTree *DT, AssumptionCache *AC)
Definition Lint.cpp:539
#define F(x, y, z)
Definition MD5.cpp:54
#define I(x, y, z)
Definition MD5.cpp:57
#define G(x, y, z)
Definition MD5.cpp:55
print mir2vec MIR2Vec Vocabulary Printer Pass
Definition MIR2Vec.cpp:598
Machine Check Debug Module
This file contains the declarations for metadata subclasses.
uint64_t IntrinsicInst * II
FunctionAnalysisManager FAM
ModuleAnalysisManager MAM
if(PassOpts->AAPipeline)
const SmallVectorImpl< MachineOperand > & Cond
static void visit(BasicBlock &Start, std::function< bool(BasicBlock *)> op)
#define OP(OPC)
Definition Instruction.h:46
This file defines the SmallPtrSet class.
This file defines the SmallSet class.
This file defines the SmallVector class.
This file defines the 'Statistic' class, which is designed to be an easy way to expose various metric...
#define STATISTIC(VARNAME, DESC)
Definition Statistic.h:171
This file contains some functions that are useful when dealing with strings.
#define LLVM_DEBUG(...)
Definition Debug.h:119
static SymbolRef::Type getType(const Symbol *Sym)
Definition TapiFile.cpp:39
This pass exposes codegen information to IR-level passes.
uint64_t getZExtValue() const
Get zero extended value.
Definition APInt.h:1563
int64_t getSExtValue() const
Get sign extended value.
Definition APInt.h:1585
LLVM_ABI AddressSanitizerPass(const AddressSanitizerOptions &Options, bool UseGlobalGC=true, bool UseOdrIndicator=true, AsanDtorKind DestructorKind=AsanDtorKind::Global, AsanCtorKind ConstructorKind=AsanCtorKind::Global)
LLVM_ABI PreservedAnalyses run(Module &M, ModuleAnalysisManager &AM)
LLVM_ABI void printPipeline(raw_ostream &OS, function_ref< StringRef(StringRef)> MapClassName2PassName)
an instruction to allocate memory on the stack
bool isSwiftError() const
Return true if this alloca is used as a swifterror argument to a call.
LLVM_ABI bool isStaticAlloca() const
Return true if this alloca is in the entry block of the function and is a constant size.
Align getAlign() const
Return the alignment of the memory that is being allocated by the instruction.
PointerType * getType() const
Overload to return most specific pointer type.
Type * getAllocatedType() const
Return the type that is being allocated by the instruction.
bool isUsedWithInAlloca() const
Return true if this alloca is used as an inalloca argument to a call.
LLVM_ABI std::optional< TypeSize > getAllocationSize(const DataLayout &DL) const
Get allocation size in bytes.
void setAlignment(Align Align)
This class represents an incoming formal argument to a Function.
Definition Argument.h:32
Represent a constant reference to an array (0 or more elements consecutively in memory),...
Definition ArrayRef.h:40
size_t size() const
Get the array size.
Definition ArrayRef.h:141
Class to represent array types.
static LLVM_ABI ArrayType * get(Type *ElementType, uint64_t NumElements)
This static method is the primary way to construct an ArrayType.
An instruction that atomically checks whether a specified value is in a memory location,...
an instruction that atomically reads a memory location, combines it with another value,...
LLVM Basic Block Representation.
Definition BasicBlock.h:62
iterator begin()
Instruction iterator methods.
Definition BasicBlock.h:461
LLVM_ABI const_iterator getFirstInsertionPt() const
Returns an iterator to the first instruction in this block that is suitable for inserting a non-PHI i...
const Function * getParent() const
Return the enclosing method, or null if none.
Definition BasicBlock.h:213
static BasicBlock * Create(LLVMContext &Context, const Twine &Name="", Function *Parent=nullptr, BasicBlock *InsertBefore=nullptr)
Creates a new BasicBlock.
Definition BasicBlock.h:206
InstListType::iterator iterator
Instruction iterators...
Definition BasicBlock.h:170
const Instruction * getTerminator() const LLVM_READONLY
Returns the terminator instruction; assumes that the block is well-formed.
Definition BasicBlock.h:237
bool isInlineAsm() const
Check if this call is an inline asm statement.
void setCannotMerge()
static LLVM_ABI CallBase * addOperandBundle(CallBase *CB, uint32_t ID, OperandBundleDef OB, InsertPosition InsertPt=nullptr)
Create a clone of CB with operand bundle OB added.
bool doesNotReturn() const
Determine if the call cannot return.
unsigned arg_size() const
This class represents a function call, abstracting a target machine's calling convention.
static CallInst * Create(FunctionType *Ty, Value *F, const Twine &NameStr="", InsertPosition InsertBefore=nullptr)
@ Largest
The linker will choose the largest COMDAT.
Definition Comdat.h:39
@ SameSize
The data referenced by the COMDAT must be the same size.
Definition Comdat.h:41
@ Any
The linker may choose any COMDAT.
Definition Comdat.h:37
@ NoDeduplicate
No deduplication is performed.
Definition Comdat.h:40
@ ExactMatch
The data referenced by the COMDAT must be the same.
Definition Comdat.h:38
Conditional Branch instruction.
static CondBrInst * Create(Value *Cond, BasicBlock *IfTrue, BasicBlock *IfFalse, InsertPosition InsertBefore=nullptr)
ConstantArray - Constant Array Declarations.
Definition Constants.h:590
static LLVM_ABI Constant * get(ArrayType *T, ArrayRef< Constant * > V)
static LLVM_ABI Constant * getPointerCast(Constant *C, Type *Ty)
Create a BitCast, AddrSpaceCast, or a PtrToInt cast constant expression.
static LLVM_ABI Constant * getPtrToInt(Constant *C, Type *Ty, bool OnlyIfReduced=false)
static LLVM_ABI bool isValueValidForType(Type *Ty, uint64_t V)
This static method returns true if the type Ty is big enough to represent the value V.
static LLVM_ABI Constant * get(StructType *T, ArrayRef< Constant * > V)
This is an important base class in LLVM.
Definition Constant.h:43
static LLVM_ABI Constant * getAllOnesValue(Type *Ty)
static LLVM_ABI Constant * getNullValue(Type *Ty)
Constructor to create a '0' constant of arbitrary type.
LLVM_ABI Constant * getAggregateElement(unsigned Elt) const
For aggregates (struct/array/vector) return the constant that corresponds to the specified element if...
A parsed version of the target data layout string in and methods for querying it.
Definition DataLayout.h:64
A debug info location.
Definition DebugLoc.h:124
DILocation * get() const
Get the underlying DILocation.
Definition DebugLoc.h:218
A handy container for a FunctionType+Callee-pointer pair, which can be passed around as a single enti...
static LLVM_ABI FunctionType * get(Type *Result, ArrayRef< Type * > Params, bool isVarArg)
This static method is the primary way of constructing a FunctionType.
const BasicBlock & front() const
Definition Function.h:834
static Function * createWithDefaultAttr(FunctionType *Ty, LinkageTypes Linkage, unsigned AddrSpace, const Twine &N="", Module *M=nullptr)
Creates a function with some attributes recorded in llvm.module.flags and the LLVMContext applied.
Definition Function.cpp:373
bool hasPersonalityFn() const
Check whether this function has a personality function.
Definition Function.h:879
LLVMContext & getContext() const
getContext - Return a reference to the LLVMContext associated with this function.
Definition Function.cpp:353
const Constant * getAliasee() const
Definition GlobalAlias.h:87
static LLVM_ABI GlobalAlias * create(Type *Ty, unsigned AddressSpace, LinkageTypes Linkage, const Twine &Name, Constant *Aliasee, Module *Parent)
If a parent module is specified, the alias is automatically inserted into the end of the specified mo...
Definition Globals.cpp:621
LLVM_ABI void copyMetadata(const GlobalObject *Src, unsigned Offset)
Copy metadata from Src, adjusting offsets by Offset.
LLVM_ABI void setComdat(Comdat *C)
Definition Globals.cpp:223
LLVM_ABI void setSection(StringRef S)
Change the section for this global.
Definition Globals.cpp:284
VisibilityTypes getVisibility() const
void setUnnamedAddr(UnnamedAddr Val)
bool hasLocalLinkage() const
static StringRef dropLLVMManglingEscape(StringRef Name)
If the given string begins with the GlobalValue name mangling escape character '\1',...
ThreadLocalMode getThreadLocalMode() const
@ HiddenVisibility
The GV is hidden.
Definition GlobalValue.h:69
void setVisibility(VisibilityTypes V)
LinkageTypes
An enumeration for the kinds of linkage for global values.
Definition GlobalValue.h:52
@ PrivateLinkage
Like Internal, but omit from symbol table.
Definition GlobalValue.h:61
@ CommonLinkage
Tentative definitions.
Definition GlobalValue.h:63
@ InternalLinkage
Rename collisions when linking (static functions).
Definition GlobalValue.h:60
@ AvailableExternallyLinkage
Available for inspection, not emission.
Definition GlobalValue.h:54
@ ExternalWeakLinkage
ExternalWeak linkage description.
Definition GlobalValue.h:62
DLLStorageClassTypes getDLLStorageClass() const
const Constant * getInitializer() const
getInitializer - Return the initializer for this global variable.
LLVM_ABI void copyAttributesFrom(const GlobalVariable *Src)
copyAttributesFrom - copy all additional attributes (those not needed to create a GlobalVariable) fro...
Definition Globals.cpp:576
void setAlignment(Align Align)
Sets the alignment attribute of the GlobalVariable.
Analysis pass providing a never-invalidated alias analysis result.
This instruction compares its operands according to the predicate given to the constructor.
Common base class shared among various IRBuilders.
Definition IRBuilder.h:114
AllocaInst * CreateAlloca(Type *Ty, unsigned AddrSpace, Value *ArraySize=nullptr, const Twine &Name="")
Definition IRBuilder.h:1879
IntegerType * getInt1Ty()
Fetch the type representing a single bit.
Definition IRBuilder.h:519
LLVM_ABI Value * CreateAllocationSize(Type *DestTy, AllocaInst *AI)
Get allocation size of an alloca as a runtime Value* (handles both static and dynamic allocas and vsc...
LoadInst * CreateAlignedLoad(Type *Ty, Value *Ptr, MaybeAlign Align, const char *Name)
Definition IRBuilder.h:1923
CallInst * CreateMemCpy(Value *Dst, MaybeAlign DstAlign, Value *Src, MaybeAlign SrcAlign, uint64_t Size, bool isVolatile=false, const AAMDNodes &AAInfo=AAMDNodes())
Create and insert a memcpy between the specified pointers.
Definition IRBuilder.h:665
Value * CreatePointerCast(Value *V, Type *DestTy, const Twine &Name="")
Definition IRBuilder.h:2279
Value * CreateICmpSGE(Value *LHS, Value *RHS, const Twine &Name="")
Definition IRBuilder.h:2392
LLVM_ABI Value * CreateSelect(Value *C, Value *True, Value *False, const Twine &Name="", Instruction *MDFrom=nullptr)
BasicBlock::iterator GetInsertPoint() const
Definition IRBuilder.h:176
Value * CreateIntToPtr(Value *V, Type *DestTy, const Twine &Name="")
Definition IRBuilder.h:2227
Value * CreateLShr(Value *LHS, Value *RHS, const Twine &Name="", bool isExact=false)
Definition IRBuilder.h:1532
IntegerType * getInt32Ty()
Fetch the type representing a 32-bit integer.
Definition IRBuilder.h:534
Value * CreatePtrAdd(Value *Ptr, Value *Offset, const Twine &Name="", GEPNoWrapFlags NW=GEPNoWrapFlags::none())
Definition IRBuilder.h:2081
BasicBlock * GetInsertBlock() const
Definition IRBuilder.h:175
IntegerType * getInt64Ty()
Fetch the type representing a 64-bit integer.
Definition IRBuilder.h:539
Value * CreateICmpNE(Value *LHS, Value *RHS, const Twine &Name="")
Definition IRBuilder.h:2368
Value * CreateGEP(Type *Ty, Value *Ptr, ArrayRef< Value * > IdxList, const Twine &Name="", GEPNoWrapFlags NW=GEPNoWrapFlags::none())
Definition IRBuilder.h:2000
PHINode * CreatePHI(Type *Ty, unsigned NumReservedValues, const Twine &Name="")
Definition IRBuilder.h:2529
Value * CreateNot(Value *V, const Twine &Name="")
Definition IRBuilder.h:1854
Value * CreateICmpEQ(Value *LHS, Value *RHS, const Twine &Name="")
Definition IRBuilder.h:2364
Value * CreateSub(Value *LHS, Value *RHS, const Twine &Name="", bool HasNUW=false, bool HasNSW=false)
Definition IRBuilder.h:1439
ConstantInt * getIntN(unsigned N, uint64_t C)
Get a constant N-bit value, zero extended from a 64-bit value.
Definition IRBuilder.h:487
LoadInst * CreateLoad(Type *Ty, Value *Ptr, const char *Name)
Provided to resolve 'CreateLoad(Ty, Ptr, "...")' correctly, instead of converting the string to 'bool...
Definition IRBuilder.h:1906
Value * CreateAnd(Value *LHS, Value *RHS, const Twine &Name="")
Definition IRBuilder.h:1570
LLVM_ABI Value * CreateIntrinsic(Intrinsic::ID ID, ArrayRef< Type * > OverloadTypes, ArrayRef< Value * > Args, FMFSource FMFSource={}, const Twine &Name="", ArrayRef< OperandBundleDef > OpBundles={}, function_ref< void(CallInst *)> SetFn=[](CallInst *) {})
Variant to create a possibly constant-folded intrinsic.
StoreInst * CreateStore(Value *Val, Value *Ptr, bool isVolatile=false)
Definition IRBuilder.h:1919
Value * CreateAdd(Value *LHS, Value *RHS, const Twine &Name="", bool HasNUW=false, bool HasNSW=false)
Definition IRBuilder.h:1422
Value * CreatePtrToInt(Value *V, Type *DestTy, const Twine &Name="")
Definition IRBuilder.h:2222
Value * CreateIsNotNull(Value *Arg, const Twine &Name="")
Return a boolean value testing if Arg != 0.
Definition IRBuilder.h:2697
CallInst * CreateCall(FunctionType *FTy, Value *Callee, ArrayRef< Value * > Args={}, const Twine &Name="", MDNode *FPMathTag=nullptr)
Definition IRBuilder.h:2543
LLVM_ABI Value * CreateTypeSize(Type *Ty, TypeSize Size)
Create an expression which evaluates to the number of units in Size at runtime.
Value * CreateIntCast(Value *V, Type *DestTy, bool isSigned, const Twine &Name="")
Definition IRBuilder.h:2305
void SetInsertPoint(BasicBlock *TheBB)
This specifies that created instructions should be appended to the end of the specified block.
Definition IRBuilder.h:181
Type * getVoidTy()
Fetch the type representing void.
Definition IRBuilder.h:572
StoreInst * CreateAlignedStore(Value *Val, Value *Ptr, MaybeAlign Align, bool isVolatile=false)
Definition IRBuilder.h:1942
Value * CreateOr(Value *LHS, Value *RHS, const Twine &Name="", bool IsDisjoint=false)
Definition IRBuilder.h:1592
IntegerType * getInt8Ty()
Fetch the type representing an 8-bit integer.
Definition IRBuilder.h:524
Value * CreateAddrSpaceCast(Value *V, Type *DestTy, const Twine &Name="")
Definition IRBuilder.h:2237
Value * CreateMul(Value *LHS, Value *RHS, const Twine &Name="", bool HasNUW=false, bool HasNSW=false)
Definition IRBuilder.h:1456
This provides a uniform API for creating instructions and inserting them into a basic block: either a...
Definition IRBuilder.h:2848
static LLVM_ABI InlineAsm * get(FunctionType *Ty, StringRef AsmString, StringRef Constraints, bool hasSideEffects, bool isAlignStack=false, AsmDialect asmDialect=AD_ATT, bool canThrow=false)
InlineAsm::get - Return the specified uniqued inline asm string.
Definition InlineAsm.cpp:43
Base class for instruction visitors.
Definition InstVisitor.h:78
const DebugLoc & getDebugLoc() const
Return the debug location for this node as a DebugLoc.
bool hasMetadata() const
Return true if this instruction has any metadata attached to it.
LLVM_ABI void moveBefore(InstListType::iterator InsertPos)
Unlink this instruction from its current basic block and insert it into the basic block that MovePos ...
LLVM_ABI InstListType::iterator eraseFromParent()
This method unlinks 'this' from the containing basic block and deletes it.
MDNode * getMetadata(unsigned KindID) const
Get the metadata of given kind attached to this Instruction.
void setDebugLoc(DebugLoc Loc)
Set the debug location information for this instruction.
LLVM_ABI const DataLayout & getDataLayout() const
Get the data layout of the module this instruction belongs to.
static LLVM_ABI IntegerType * get(LLVMContext &C, unsigned NumBits)
This static method is the primary way of constructing an IntegerType.
Definition Type.cpp:350
A wrapper class for inspecting calls to intrinsic functions.
LLVM_ABI void emitError(const Instruction *I, const Twine &ErrorStr)
emitError - Emit an error message to the currently installed error handler with optional location inf...
An instruction for reading from memory.
static Error ParseSectionSpecifier(StringRef Spec, StringRef &Segment, StringRef &Section, unsigned &TAA, bool &TAAParsed, unsigned &StubSize)
Parse the section specifier indicated by "Spec".
LLVM_ABI MDNode * createUnlikelyBranchWeights()
Return metadata containing two branch weights, with significant bias towards false destination.
Definition MDBuilder.cpp:48
Metadata node.
Definition Metadata.h:1069
ArrayRef< MDOperand > operands() const
Definition Metadata.h:1431
static MDTuple * get(LLVMContext &Context, ArrayRef< Metadata * > MDs)
Definition Metadata.h:1561
Tuple of metadata.
Definition Metadata.h:1489
This is the common base class for memset/memcpy/memmove.
static MemoryEffectsBase argMemOnly(ModRefInfo MR=ModRefInfo::ModRef)
Definition ModRef.h:143
static MemoryEffectsBase otherMemOnly(ModRefInfo MR=ModRefInfo::ModRef)
Definition ModRef.h:159
Root of the metadata hierarchy.
Definition Metadata.h:64
A Module instance is used to store all the information related to an LLVM module.
Definition Module.h:67
Evaluate the size and offset of an object pointed to by a Value* statically.
LLVM_ABI SizeOffsetAPInt compute(Value *V)
Pass interface - Implemented by all 'passes'.
Definition Pass.h:99
static PointerType * getUnqual(Type *ElementType)
This constructs a pointer to an object of the specified type in the default address space (address sp...
static LLVM_ABI PointerType * get(Type *ElementType, unsigned AddressSpace)
This constructs a pointer to an object of the specified type in a numbered address space.
A set of analyses that are preserved following a run of a transformation pass.
Definition Analysis.h:112
static PreservedAnalyses none()
Convenience factory function for the empty preserved set.
Definition Analysis.h:115
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
Definition Analysis.h:118
PreservedAnalyses & abandon()
Mark an analysis as abandoned.
Definition Analysis.h:171
Return a value (possibly void), from a function.
static ReturnInst * Create(LLVMContext &C, Value *retVal=nullptr, InsertPosition InsertBefore=nullptr)
size_type count(ConstPtrType Ptr) const
count - Return 1 if the specified pointer is in the set, 0 otherwise.
std::pair< iterator, bool > insert(PtrType Ptr)
Inserts Ptr if and only if there is no element in the container equal to Ptr.
SmallPtrSet - This class implements a set which is optimized for holding SmallSize or less elements.
SmallSet - This maintains a set of unique values, optimizing for the case when the set is small (less...
Definition SmallSet.h:134
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
reference emplace_back(ArgTypes &&... Args)
void reserve(size_type N)
void resize(size_type N)
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
This pass performs the global (interprocedural) stack safety analysis (new pass manager).
LLVM_ABI bool stackAccessIsSafe(const Instruction &I) const
LLVM_ABI bool isSafe(const AllocaInst &AI) const
An instruction for storing to memory.
Represent a constant reference to a string, i.e.
Definition StringRef.h:56
bool starts_with(StringRef Prefix) const
Check if this string starts with the given Prefix.
Definition StringRef.h:258
constexpr bool empty() const
Check if the string is empty.
Definition StringRef.h:141
Class to represent struct types.
static LLVM_ABI StructType * get(LLVMContext &Context, ArrayRef< Type * > Elements, bool isPacked=false)
This static method is the primary way to create a literal StructType.
Definition Type.cpp:479
Analysis pass providing the TargetTransformInfo.
Analysis pass providing the TargetLibraryInfo.
Provides information about what library functions are available for the current target.
AttributeList getAttrList(LLVMContext *C, ArrayRef< unsigned > ArgNos, bool Signed, bool Ret=false, AttributeList AL=AttributeList()) const
This pass provides access to the codegen interfaces that are needed for IR-level transformations.
EltTy front() const
unsigned size() const
Triple - Helper class for working with autoconf configuration names.
Definition Triple.h:47
bool isThumb() const
Tests whether the target is Thumb (little and big endian).
Definition Triple.h:913
bool isDriverKit() const
Is this an Apple DriverKit triple.
Definition Triple.h:629
bool isBPF() const
Tests whether the target is eBPF.
Definition Triple.h:1151
bool isOSNetBSD() const
Definition Triple.h:666
bool isAndroid() const
Tests whether the target is Android.
Definition Triple.h:826
bool isABIN32() const
Definition Triple.h:1139
bool isMIPS64() const
Tests whether the target is MIPS 64-bit (little and big endian).
Definition Triple.h:1043
ArchType getArch() const
Get the parsed architecture type of this triple.
Definition Triple.h:436
bool isLoongArch64() const
Tests whether the target is 64-bit LoongArch.
Definition Triple.h:1032
bool isMIPS32() const
Tests whether the target is MIPS 32-bit (little and big endian).
Definition Triple.h:1038
bool isOSWindows() const
Tests whether the OS is Windows.
Definition Triple.h:699
@ UnknownObjectFormat
Definition Triple.h:332
bool isARM() const
Tests whether the target is ARM (little and big endian).
Definition Triple.h:918
bool isOSLinux() const
Tests whether the OS is Linux.
Definition Triple.h:746
bool isAMDGPU() const
Definition Triple.h:910
bool isMacOSX() const
Is this a Mac OS X triple.
Definition Triple.h:603
bool isOSFreeBSD() const
Definition Triple.h:670
bool isOSEmscripten() const
Tests whether the OS is Emscripten.
Definition Triple.h:761
bool isWatchOS() const
Is this an Apple watchOS triple.
Definition Triple.h:618
bool isiOS() const
Is this an iOS triple.
Definition Triple.h:612
bool isPS() const
Tests whether the target is the PS4 or PS5 platform.
Definition Triple.h:823
bool isWasm() const
Tests whether the target is wasm (32- and 64-bit).
Definition Triple.h:1125
bool isOSFuchsia() const
Definition Triple.h:672
bool isOSHaiku() const
Tests whether the OS is Haiku.
Definition Triple.h:693
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
Definition Twine.h:82
The instances of the Type class are immutable: once they are created, they are never changed.
Definition Type.h:46
LLVM_ABI unsigned getIntegerBitWidth() const
static LLVM_ABI IntegerType * getInt32Ty(LLVMContext &C)
Definition Type.cpp:309
LLVM_ABI unsigned getPointerAddressSpace() const
Get the address space of this pointer or pointer vector type.
static LLVM_ABI Type * getVoidTy(LLVMContext &C)
Definition Type.cpp:282
static LLVM_ABI IntegerType * getInt8Ty(LLVMContext &C)
Definition Type.cpp:307
Type * getScalarType() const
If this is a vector type, return the element type, otherwise return 'this'.
Definition Type.h:368
bool isSized(SmallPtrSetImpl< Type * > *Visited=nullptr) const
Return true if it makes sense to take the size of this type.
Definition Type.h:326
This function has undefined behavior.
A Use represents the edge between a Value definition and its users.
Definition Use.h:35
op_range operands()
Definition User.h:267
Value * getOperand(unsigned i) const
Definition User.h:207
static LLVM_ABI ValueAsMetadata * get(Value *V)
Definition Metadata.cpp:509
LLVM Value Representation.
Definition Value.h:75
Type * getType() const
All values are typed, get the type of this value.
Definition Value.h:255
LLVM_ABI void replaceAllUsesWith(Value *V)
Change all uses of this to point to a new Value.
Definition Value.cpp:553
iterator_range< user_iterator > users()
Definition Value.h:426
LLVM_ABI bool isSwiftError() const
Return true if this value is a swifterror value.
Definition Value.cpp:1155
LLVM_ABI StringRef getName() const
Return a constant reference to the value's name.
Definition Value.cpp:319
LLVM_ABI void takeName(Value *V)
Transfer the name from V to this value.
Definition Value.cpp:400
Base class of all SIMD vector types.
static LLVM_ABI VectorType * get(Type *ElementType, ElementCount EC)
This static method is the primary way to construct an VectorType.
constexpr ScalarTy getFixedValue() const
Definition TypeSize.h:200
constexpr bool isScalable() const
Returns whether the quantity is scaled by a runtime quantity (vscale).
Definition TypeSize.h:168
An efficient, type-erasing, non-owning reference to a callable.
const ParentTy * getParent() const
Definition ilist_node.h:34
self_iterator getIterator()
Definition ilist_node.h:123
NodeTy * getNextNode()
Get the next node, or nullptr for the list tail.
Definition ilist_node.h:348
This class implements an extremely fast bulk output stream that can only output to a stream.
Definition raw_ostream.h:53
CallInst * Call
Changed
This file contains the declaration of the Comdat class, which represents a single COMDAT in LLVM.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
void getInterestingMemoryOperands(Module &M, Instruction *I, SmallVectorImpl< InterestingMemoryOperand > &Interesting)
Get all the memory operands from the instruction that needs to be instrumented.
void instrumentAddress(Module &M, IRBuilder<> &IRB, Instruction *OrigIns, Instruction *InsertBefore, Value *Addr, Align Alignment, TypeSize TypeStoreSize, bool IsWrite, Value *SizeArgument, bool UseCalls, bool Recover, int AsanScale, int AsanOffset)
Instrument the memory operand Addr.
uint64_t getRedzoneSizeForGlobal(int AsanScale, uint64_t SizeInBytes)
Given SizeInBytes of the Value to be instrunmented, Returns the redzone size corresponding to it.
constexpr std::underlying_type_t< E > Mask()
Get a bitmask with 1s in all places up to the high-order bit of E's largest value.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
Definition CallingConv.h:24
@ C
The default llvm calling convention, compatible with C.
Definition CallingConv.h:34
@ BasicBlock
Various leaf nodes.
Definition ISDOpcodes.h:81
@ S_CSTRING_LITERALS
S_CSTRING_LITERALS - Section with literal C strings.
Definition MachO.h:131
@ OB
OB - OneByte - Set if this instruction has a one byte opcode.
ValuesClass values(OptsTy... Options)
Helper to build a ValuesClass by forwarding a variable number of arguments as an initializer list to ...
initializer< Ty > init(const Ty &Val)
cb< typename detail::callback_traits< F >::result_type, typename detail::callback_traits< F >::arg_type > callback(F CB)
LLVM_ABI uint64_t getAllocaSizeInBytes(const AllocaInst &AI)
Context & getContext() const
Definition BasicBlock.h:99
friend class Instruction
Iterator for Instructions in a `BasicBlock.
Definition BasicBlock.h:73
This is an optimization pass for GlobalISel generic memory operations.
LLVM_ABI void ReplaceInstWithInst(BasicBlock *BB, BasicBlock::iterator &BI, Instruction *I)
Replace the instruction specified by BI with the instruction specified by I.
@ Offset
Definition DWP.cpp:558
FunctionAddr VTableAddr Value
Definition InstrProf.h:137
bool all_of(R &&range, UnaryPredicate P)
Provide wrappers to std::all_of which take ranges instead of having to pass begin/end explicitly.
Definition STLExtras.h:1739
LLVM_ABI SmallVector< uint8_t, 64 > GetShadowBytesAfterScope(const SmallVectorImpl< ASanStackVariableDescription > &Vars, const ASanStackFrameLayout &Layout)
LLVM_ABI GlobalVariable * createPrivateGlobalForString(Module &M, StringRef Str, bool AllowMerging, Twine NamePrefix="")
LLVM_ABI AllocaInst * findAllocaForValue(Value *V, bool OffsetZero=false)
Returns unique alloca where the value comes from, or nullptr.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:643
@ Done
Definition Threading.h:60
FunctionAddr VTableAddr uintptr_t uintptr_t Int32Ty
Definition InstrProf.h:328
LLVM_ABI Function * createSanitizerCtor(Module &M, StringRef CtorName)
Creates sanitizer constructor function.
AsanDetectStackUseAfterReturnMode
Mode of ASan detect stack use after return.
@ Always
Always detect stack use after return.
@ Never
Never detect stack use after return.
@ Runtime
Detect stack use after return if not disabled runtime with (ASAN_OPTIONS=detect_stack_use_after_retur...
LLVM_ABI DenseMap< BasicBlock *, ColorVector > colorEHFunclets(Function &F)
If an EH funclet personality is in use (see isFuncletEHPersonality), this will recompute which blocks...
iterator_range< early_inc_iterator_impl< detail::IterOfRange< RangeT > > > make_early_inc_range(RangeT &&Range)
Make a range that does early increment to allow mutation of the underlying range without disrupting i...
Definition STLExtras.h:633
InnerAnalysisManagerProxy< FunctionAnalysisManager, Module > FunctionAnalysisManagerModuleProxy
Provide the FunctionAnalysisManager to Module proxy.
Op::Description Desc
LLVM_ABI bool isAllocaPromotable(const AllocaInst *AI)
Return true if this alloca is legal for promotion.
LLVM_ABI SmallString< 64 > ComputeASanStackFrameDescription(const SmallVectorImpl< ASanStackVariableDescription > &Vars)
LLVM_ABI SmallVector< uint8_t, 64 > GetShadowBytes(const SmallVectorImpl< ASanStackVariableDescription > &Vars, const ASanStackFrameLayout &Layout)
int countr_zero(T Val)
Count number of 0's from the least significant bit to the most stopping at the first 1.
Definition bit.h:204
auto dyn_cast_or_null(const Y &Val)
Definition Casting.h:753
LLVM_ABI FunctionCallee declareSanitizerInitFunction(Module &M, StringRef InitName, ArrayRef< Type * > InitArgTypes, bool Weak=false)
FunctionAddr VTableAddr uintptr_t uintptr_t Version
Definition InstrProf.h:334
LLVM_ABI std::string getUniqueModuleId(Module *M)
Produce a unique identifier for this module by taking the MD5 sum of the names of the module's strong...
constexpr bool isPowerOf2_32(uint32_t Value)
Return true if the argument is a power of two > 0.
Definition MathExtras.h:279
LLVM_ABI std::pair< Function *, FunctionCallee > createSanitizerCtorAndInitFunctions(Module &M, StringRef CtorName, StringRef InitName, ArrayRef< Type * > InitArgTypes, ArrayRef< Value * > InitArgs, StringRef VersionCheckName=StringRef(), bool Weak=false)
Creates sanitizer constructor function, and calls sanitizer's init function from it.
decltype(auto) get(const PointerIntPair< PointerTy, IntBits, IntType, PtrTraits, Info > &Pair)
LLVM_ABI void SplitBlockAndInsertIfThenElse(Value *Cond, BasicBlock::iterator SplitBefore, Instruction **ThenTerm, Instruction **ElseTerm, MDNode *BranchWeights=nullptr, DomTreeUpdater *DTU=nullptr, LoopInfo *LI=nullptr)
SplitBlockAndInsertIfThenElse is similar to SplitBlockAndInsertIfThen, but also creates the ElseBlock...
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
Definition Debug.cpp:209
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
Definition Error.cpp:163
bool isAlnum(char C)
Checks whether character C is either a decimal digit or an uppercase or lowercase letter as classifie...
class LLVM_GSL_OWNER SmallVector
Forward declaration of SmallVector so that calculateSmallVectorDefaultInlinedElements can reference s...
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
Definition Casting.h:547
AsanDtorKind
Types of ASan module destructors supported.
@ Invalid
Not a valid destructor Kind.
@ Global
Append to llvm.global_dtors.
@ None
Do not emit any destructors for ASan.
LLVM_ABI ASanStackFrameLayout ComputeASanStackFrameLayout(SmallVectorImpl< ASanStackVariableDescription > &Vars, uint64_t Granularity, uint64_t MinHeaderSize)
@ Ref
The access may reference the value stored in memory.
Definition ModRef.h:32
@ ModRef
The access may reference and may modify the value stored in memory.
Definition ModRef.h:36
@ Mod
The access may modify the value stored in memory.
Definition ModRef.h:34
@ ArgMem
Access to memory via argument pointers.
Definition ModRef.h:62
@ Other
Any other memory.
Definition ModRef.h:68
@ InaccessibleMem
Memory that is inaccessible via LLVM IR.
Definition ModRef.h:64
TargetTransformInfo TTI
void cantFail(Error Err, const char *Msg=nullptr)
Report a fatal error if Err is a failure value.
Definition Error.h:769
IRBuilder(LLVMContext &, FolderTy, InserterTy, MDNode *, ArrayRef< OperandBundleDef >) -> IRBuilder< FolderTy, InserterTy >
OperandBundleDefT< Value * > OperandBundleDef
Definition AutoUpgrade.h:34
LLVM_ABI void appendToCompilerUsed(Module &M, ArrayRef< GlobalValue * > Values)
Adds global values to the llvm.compiler.used list.
static const int kAsanStackUseAfterReturnMagic
LLVM_ABI void setGlobalVariableLargeSection(const Triple &TargetTriple, GlobalVariable &GV)
LLVM_ABI void removeASanIncompatibleFnAttributes(Function &F, bool ReadsArgMem)
Remove memory attributes that are incompatible with the instrumentation added by AddressSanitizer and...
@ Dynamic
Denotes mode unknown at compile time.
ArrayRef(const T &OneElt) -> ArrayRef< T >
bool isModAndRefSet(const ModRefInfo MRI)
Definition ModRef.h:46
LLVM_ABI void appendToGlobalCtors(Module &M, Function *F, int Priority, Constant *Data=nullptr)
Append F to the list of global ctors of module M with the given Priority.
TinyPtrVector< BasicBlock * > ColorVector
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:559
Align assumeAligned(uint64_t Value)
Treats the value 0 as a 1, so Align is always at least 1.
Definition Alignment.h:100
iterator_range< df_iterator< T > > depth_first(const T &G)
LLVM_ABI Instruction * SplitBlockAndInsertIfThen(Value *Cond, BasicBlock::iterator SplitBefore, bool Unreachable, MDNode *BranchWeights=nullptr, DomTreeUpdater *DTU=nullptr, LoopInfo *LI=nullptr, BasicBlock *ThenBlock=nullptr)
Split the containing block at the specified instruction - everything before SplitBefore stays in the ...
LLVM_ABI const Value * getUnderlyingObject(const Value *V, unsigned MaxLookup=MaxLookupSearchDepth)
This method strips off any GEP address adjustments, pointer casts or llvm.threadlocal....
AsanCtorKind
Types of ASan module constructors supported.
LLVM_ABI void maybeMarkSanitizerLibraryCallNoBuiltin(CallInst *CI, const TargetLibraryInfo *TLI)
Given a CallInst, check if it calls a string function known to CodeGen, and mark it with NoBuiltin if...
Definition Local.cpp:3886
LLVM_ABI void appendToUsed(Module &M, ArrayRef< GlobalValue * > Values)
Adds global values to the llvm.used list.
LLVM_ABI void appendToGlobalDtors(Module &M, Function *F, int Priority, Constant *Data=nullptr)
Same as appendToGlobalCtors(), but for global dtors.
LLVM_ABI bool checkIfAlreadyInstrumented(Module &M, StringRef Flag)
Check if module has flag attached, if not add the flag.
LLVM_ABI void getAddressSanitizerParams(const Triple &TargetTriple, int LongSize, bool IsKasan, uint64_t *ShadowBase, int *MappingScale, bool *OrShadowOffset)
DEMANGLE_ABI std::string demangle(std::string_view MangledName)
Attempt to demangle a string using different demangling schemes.
Definition Demangle.cpp:21
std::string itostr(int64_t X)
LLVM_ABI void SplitBlockAndInsertForEachLane(ElementCount EC, Type *IndexTy, BasicBlock::iterator InsertBefore, std::function< void(IRBuilderBase &, Value *)> Func)
Utility function for performing a given action on each lane of a vector with EC elements.
AnalysisManager< Module > ModuleAnalysisManager
Convenience typedef for the Module analysis manager.
Definition MIRParser.h:39
LLVM_ABI bool replaceDbgDeclare(Value *Address, Value *NewAddress, DIBuilder &Builder, uint8_t DIExprFlags, int Offset)
Replaces dbg.declare record when the address it describes is replaced with a new value.
Definition Local.cpp:1968
#define N
LLVM_ABI ASanAccessInfo(int32_t Packed)
const uint8_t AccessSizeIndex
This struct is a compact representation of a valid (non-zero power of two) alignment.
Definition Alignment.h:39
constexpr uint64_t value() const
This is a hole in the type system and should not be abused.
Definition Alignment.h:77
This struct is a compact representation of a valid (power of two) or undefined (0) alignment.
Definition Alignment.h:106
Align valueOrOne() const
For convenience, returns a valid alignment or 1 if undefined.
Definition Alignment.h:130
Information about a load/store intrinsic defined by the target.
SmallVector< InterestingMemoryOperand, 1 > InterestingOperands
A CRTP mix-in to automatically provide informational APIs needed for passes.
Definition PassManager.h:89
SizeOffsetAPInt - Used by ObjectSizeOffsetVisitor, which works with APInts.