xref: /aosp_15_r20/external/llvm/lib/Fuzzer/FuzzerTracePC.cpp (revision 9880d6810fe72a1726cb53787c6711e909410d58)
1*9880d681SAndroid Build Coastguard Worker //===- FuzzerTracePC.cpp - PC tracing--------------------------------------===//
2*9880d681SAndroid Build Coastguard Worker //
3*9880d681SAndroid Build Coastguard Worker //                     The LLVM Compiler Infrastructure
4*9880d681SAndroid Build Coastguard Worker //
5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source
6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details.
7*9880d681SAndroid Build Coastguard Worker //
8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
9*9880d681SAndroid Build Coastguard Worker // Trace PCs.
10*9880d681SAndroid Build Coastguard Worker // This module implements __sanitizer_cov_trace_pc, a callback required
11*9880d681SAndroid Build Coastguard Worker // for -fsanitize-coverage=trace-pc instrumentation.
12*9880d681SAndroid Build Coastguard Worker //
13*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
14*9880d681SAndroid Build Coastguard Worker 
15*9880d681SAndroid Build Coastguard Worker #include "FuzzerInternal.h"
16*9880d681SAndroid Build Coastguard Worker 
17*9880d681SAndroid Build Coastguard Worker namespace fuzzer {
18*9880d681SAndroid Build Coastguard Worker 
Reset()19*9880d681SAndroid Build Coastguard Worker void PcCoverageMap::Reset() { memset(Map, 0, sizeof(Map)); }
20*9880d681SAndroid Build Coastguard Worker 
Update(uintptr_t Addr)21*9880d681SAndroid Build Coastguard Worker void PcCoverageMap::Update(uintptr_t Addr) {
22*9880d681SAndroid Build Coastguard Worker   uintptr_t Idx = Addr % kMapSizeInBits;
23*9880d681SAndroid Build Coastguard Worker   uintptr_t WordIdx = Idx / kBitsInWord;
24*9880d681SAndroid Build Coastguard Worker   uintptr_t BitIdx = Idx % kBitsInWord;
25*9880d681SAndroid Build Coastguard Worker   Map[WordIdx] |= 1UL << BitIdx;
26*9880d681SAndroid Build Coastguard Worker }
27*9880d681SAndroid Build Coastguard Worker 
MergeFrom(const PcCoverageMap & Other)28*9880d681SAndroid Build Coastguard Worker size_t PcCoverageMap::MergeFrom(const PcCoverageMap &Other) {
29*9880d681SAndroid Build Coastguard Worker   uintptr_t Res = 0;
30*9880d681SAndroid Build Coastguard Worker   for (size_t i = 0; i < kMapSizeInWords; i++)
31*9880d681SAndroid Build Coastguard Worker     Res += __builtin_popcountl(Map[i] |= Other.Map[i]);
32*9880d681SAndroid Build Coastguard Worker   return Res;
33*9880d681SAndroid Build Coastguard Worker }
34*9880d681SAndroid Build Coastguard Worker 
35*9880d681SAndroid Build Coastguard Worker static PcCoverageMap CurrentMap;
36*9880d681SAndroid Build Coastguard Worker static thread_local uintptr_t Prev;
37*9880d681SAndroid Build Coastguard Worker 
PcMapResetCurrent()38*9880d681SAndroid Build Coastguard Worker void PcMapResetCurrent() {
39*9880d681SAndroid Build Coastguard Worker   if (Prev) {
40*9880d681SAndroid Build Coastguard Worker     Prev = 0;
41*9880d681SAndroid Build Coastguard Worker     CurrentMap.Reset();
42*9880d681SAndroid Build Coastguard Worker   }
43*9880d681SAndroid Build Coastguard Worker }
44*9880d681SAndroid Build Coastguard Worker 
PcMapMergeInto(PcCoverageMap * Map)45*9880d681SAndroid Build Coastguard Worker size_t PcMapMergeInto(PcCoverageMap *Map) {
46*9880d681SAndroid Build Coastguard Worker   if (!Prev)
47*9880d681SAndroid Build Coastguard Worker     return 0;
48*9880d681SAndroid Build Coastguard Worker   return Map->MergeFrom(CurrentMap);
49*9880d681SAndroid Build Coastguard Worker }
50*9880d681SAndroid Build Coastguard Worker 
HandlePC(uint32_t PC)51*9880d681SAndroid Build Coastguard Worker static void HandlePC(uint32_t PC) {
52*9880d681SAndroid Build Coastguard Worker   // We take 12 bits of PC and mix it with the previous PCs.
53*9880d681SAndroid Build Coastguard Worker   uintptr_t Next = (Prev << 5) ^ (PC & 4095);
54*9880d681SAndroid Build Coastguard Worker   CurrentMap.Update(Next);
55*9880d681SAndroid Build Coastguard Worker   Prev = Next;
56*9880d681SAndroid Build Coastguard Worker }
57*9880d681SAndroid Build Coastguard Worker 
58*9880d681SAndroid Build Coastguard Worker } // namespace fuzzer
59*9880d681SAndroid Build Coastguard Worker 
60*9880d681SAndroid Build Coastguard Worker extern "C" {
__sanitizer_cov_trace_pc()61*9880d681SAndroid Build Coastguard Worker void __sanitizer_cov_trace_pc() {
62*9880d681SAndroid Build Coastguard Worker   fuzzer::HandlePC(static_cast<uint32_t>(
63*9880d681SAndroid Build Coastguard Worker       reinterpret_cast<uintptr_t>(__builtin_return_address(0))));
64*9880d681SAndroid Build Coastguard Worker }
65*9880d681SAndroid Build Coastguard Worker 
__sanitizer_cov_trace_pc_indir(int *)66*9880d681SAndroid Build Coastguard Worker void __sanitizer_cov_trace_pc_indir(int *) {
67*9880d681SAndroid Build Coastguard Worker   // Stub to allow linking with code built with
68*9880d681SAndroid Build Coastguard Worker   // -fsanitize=indirect-calls,trace-pc.
69*9880d681SAndroid Build Coastguard Worker   // This isn't used currently.
70*9880d681SAndroid Build Coastguard Worker }
71*9880d681SAndroid Build Coastguard Worker }
72