github.com/ice-blockchain/go/src@v0.0.0-20240403114104-1564d284e521/runtime/libfuzzer.go (about) 1 // Copyright 2019 The Go Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style 3 // license that can be found in the LICENSE file. 4 5 //go:build libfuzzer 6 7 package runtime 8 9 import "unsafe" 10 11 func libfuzzerCallWithTwoByteBuffers(fn, start, end *byte) 12 func libfuzzerCallTraceIntCmp(fn *byte, arg0, arg1, fakePC uintptr) 13 func libfuzzerCall4(fn *byte, fakePC uintptr, s1, s2 unsafe.Pointer, result uintptr) 14 15 // Keep in sync with the definition of ret_sled in src/runtime/libfuzzer_amd64.s 16 const retSledSize = 512 17 18 // In libFuzzer mode, the compiler inserts calls to libfuzzerTraceCmpN and libfuzzerTraceConstCmpN 19 // (where N can be 1, 2, 4, or 8) for encountered integer comparisons in the code to be instrumented. 20 // This may result in these functions having callers that are nosplit. That is why they must be nosplit. 21 // 22 //go:nosplit 23 func libfuzzerTraceCmp1(arg0, arg1 uint8, fakePC uint) { 24 fakePC = fakePC % retSledSize 25 libfuzzerCallTraceIntCmp(&__sanitizer_cov_trace_cmp1, uintptr(arg0), uintptr(arg1), uintptr(fakePC)) 26 } 27 28 //go:nosplit 29 func libfuzzerTraceCmp2(arg0, arg1 uint16, fakePC uint) { 30 fakePC = fakePC % retSledSize 31 libfuzzerCallTraceIntCmp(&__sanitizer_cov_trace_cmp2, uintptr(arg0), uintptr(arg1), uintptr(fakePC)) 32 } 33 34 //go:nosplit 35 func libfuzzerTraceCmp4(arg0, arg1 uint32, fakePC uint) { 36 fakePC = fakePC % retSledSize 37 libfuzzerCallTraceIntCmp(&__sanitizer_cov_trace_cmp4, uintptr(arg0), uintptr(arg1), uintptr(fakePC)) 38 } 39 40 //go:nosplit 41 func libfuzzerTraceCmp8(arg0, arg1 uint64, fakePC uint) { 42 fakePC = fakePC % retSledSize 43 libfuzzerCallTraceIntCmp(&__sanitizer_cov_trace_cmp8, uintptr(arg0), uintptr(arg1), uintptr(fakePC)) 44 } 45 46 //go:nosplit 47 func libfuzzerTraceConstCmp1(arg0, arg1 uint8, fakePC uint) { 48 fakePC = fakePC % retSledSize 49 libfuzzerCallTraceIntCmp(&__sanitizer_cov_trace_const_cmp1, uintptr(arg0), uintptr(arg1), uintptr(fakePC)) 50 } 51 52 //go:nosplit 53 func libfuzzerTraceConstCmp2(arg0, arg1 uint16, fakePC uint) { 54 fakePC = fakePC % retSledSize 55 libfuzzerCallTraceIntCmp(&__sanitizer_cov_trace_const_cmp2, uintptr(arg0), uintptr(arg1), uintptr(fakePC)) 56 } 57 58 //go:nosplit 59 func libfuzzerTraceConstCmp4(arg0, arg1 uint32, fakePC uint) { 60 fakePC = fakePC % retSledSize 61 libfuzzerCallTraceIntCmp(&__sanitizer_cov_trace_const_cmp4, uintptr(arg0), uintptr(arg1), uintptr(fakePC)) 62 } 63 64 //go:nosplit 65 func libfuzzerTraceConstCmp8(arg0, arg1 uint64, fakePC uint) { 66 fakePC = fakePC % retSledSize 67 libfuzzerCallTraceIntCmp(&__sanitizer_cov_trace_const_cmp8, uintptr(arg0), uintptr(arg1), uintptr(fakePC)) 68 } 69 70 var pcTables []byte 71 72 func init() { 73 libfuzzerCallWithTwoByteBuffers(&__sanitizer_cov_8bit_counters_init, &__start___sancov_cntrs, &__stop___sancov_cntrs) 74 start := unsafe.Pointer(&__start___sancov_cntrs) 75 end := unsafe.Pointer(&__stop___sancov_cntrs) 76 77 // PC tables are arrays of ptr-sized integers representing pairs [PC,PCFlags] for every instrumented block. 78 // The number of PCs and PCFlags is the same as the number of 8-bit counters. Each PC table entry has 79 // the size of two ptr-sized integers. We allocate one more byte than what we actually need so that we can 80 // get a pointer representing the end of the PC table array. 81 size := (uintptr(end)-uintptr(start))*unsafe.Sizeof(uintptr(0))*2 + 1 82 pcTables = make([]byte, size) 83 libfuzzerCallWithTwoByteBuffers(&__sanitizer_cov_pcs_init, &pcTables[0], &pcTables[size-1]) 84 } 85 86 // We call libFuzzer's __sanitizer_weak_hook_strcmp function which takes the 87 // following four arguments: 88 // 89 // 1. caller_pc: location of string comparison call site 90 // 2. s1: first string used in the comparison 91 // 3. s2: second string used in the comparison 92 // 4. result: an integer representing the comparison result. 0 indicates 93 // equality (comparison will ignored by libfuzzer), non-zero indicates a 94 // difference (comparison will be taken into consideration). 95 // 96 //go:nosplit 97 func libfuzzerHookStrCmp(s1, s2 string, fakePC int) { 98 if s1 != s2 { 99 libfuzzerCall4(&__sanitizer_weak_hook_strcmp, uintptr(fakePC), cstring(s1), cstring(s2), uintptr(1)) 100 } 101 // if s1 == s2 we could call the hook with a last argument of 0 but this is unnecessary since this case will be then 102 // ignored by libfuzzer 103 } 104 105 // This function has now the same implementation as libfuzzerHookStrCmp because we lack better checks 106 // for case-insensitive string equality in the runtime package. 107 // 108 //go:nosplit 109 func libfuzzerHookEqualFold(s1, s2 string, fakePC int) { 110 if s1 != s2 { 111 libfuzzerCall4(&__sanitizer_weak_hook_strcmp, uintptr(fakePC), cstring(s1), cstring(s2), uintptr(1)) 112 } 113 } 114 115 //go:linkname __sanitizer_cov_trace_cmp1 __sanitizer_cov_trace_cmp1 116 //go:cgo_import_static __sanitizer_cov_trace_cmp1 117 var __sanitizer_cov_trace_cmp1 byte 118 119 //go:linkname __sanitizer_cov_trace_cmp2 __sanitizer_cov_trace_cmp2 120 //go:cgo_import_static __sanitizer_cov_trace_cmp2 121 var __sanitizer_cov_trace_cmp2 byte 122 123 //go:linkname __sanitizer_cov_trace_cmp4 __sanitizer_cov_trace_cmp4 124 //go:cgo_import_static __sanitizer_cov_trace_cmp4 125 var __sanitizer_cov_trace_cmp4 byte 126 127 //go:linkname __sanitizer_cov_trace_cmp8 __sanitizer_cov_trace_cmp8 128 //go:cgo_import_static __sanitizer_cov_trace_cmp8 129 var __sanitizer_cov_trace_cmp8 byte 130 131 //go:linkname __sanitizer_cov_trace_const_cmp1 __sanitizer_cov_trace_const_cmp1 132 //go:cgo_import_static __sanitizer_cov_trace_const_cmp1 133 var __sanitizer_cov_trace_const_cmp1 byte 134 135 //go:linkname __sanitizer_cov_trace_const_cmp2 __sanitizer_cov_trace_const_cmp2 136 //go:cgo_import_static __sanitizer_cov_trace_const_cmp2 137 var __sanitizer_cov_trace_const_cmp2 byte 138 139 //go:linkname __sanitizer_cov_trace_const_cmp4 __sanitizer_cov_trace_const_cmp4 140 //go:cgo_import_static __sanitizer_cov_trace_const_cmp4 141 var __sanitizer_cov_trace_const_cmp4 byte 142 143 //go:linkname __sanitizer_cov_trace_const_cmp8 __sanitizer_cov_trace_const_cmp8 144 //go:cgo_import_static __sanitizer_cov_trace_const_cmp8 145 var __sanitizer_cov_trace_const_cmp8 byte 146 147 //go:linkname __sanitizer_cov_8bit_counters_init __sanitizer_cov_8bit_counters_init 148 //go:cgo_import_static __sanitizer_cov_8bit_counters_init 149 var __sanitizer_cov_8bit_counters_init byte 150 151 // start, stop markers of counters, set by the linker 152 var __start___sancov_cntrs, __stop___sancov_cntrs byte 153 154 //go:linkname __sanitizer_cov_pcs_init __sanitizer_cov_pcs_init 155 //go:cgo_import_static __sanitizer_cov_pcs_init 156 var __sanitizer_cov_pcs_init byte 157 158 //go:linkname __sanitizer_weak_hook_strcmp __sanitizer_weak_hook_strcmp 159 //go:cgo_import_static __sanitizer_weak_hook_strcmp 160 var __sanitizer_weak_hook_strcmp byte