aboutsummaryrefslogtreecommitdiff
path: root/src/jmh/java/com/code_intelligence/jazzer/runtime/FuzzerCallbacksPanama.java
blob: ce3d62902b49f4b3461fd5a1ad270d0b1d50731a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
// Copyright 2022 Code Intelligence GmbH
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//      http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

package com.code_intelligence.jazzer.runtime;

import com.github.fmeum.rules_jni.RulesJni;
import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodType;
import jdk.incubator.foreign.CLinker;
import jdk.incubator.foreign.FunctionDescriptor;
import jdk.incubator.foreign.MemoryAddress;
import jdk.incubator.foreign.MemoryLayout;
import jdk.incubator.foreign.MemorySegment;
import jdk.incubator.foreign.ResourceScope;
import jdk.incubator.foreign.SymbolLookup;

/**
 * Pure-Java implementation of the fuzzer callbacks backed by Project Panama (requires JDK 16+).
 * To include the implementation in the benchmark on a supported JDK, uncomment the relevant lines
 * in BUILD.bazel.
 */
public class FuzzerCallbacksPanama {
  static {
    RulesJni.loadLibrary("fuzzer_callbacks", FuzzerCallbacks.class);
  }

  private static final MethodHandle traceCmp4 = CLinker.getInstance().downcallHandle(
      SymbolLookup.loaderLookup().lookup("__sanitizer_cov_trace_cmp4").get(),
      MethodType.methodType(void.class, int.class, int.class),
      FunctionDescriptor.ofVoid(CLinker.C_INT, CLinker.C_INT));
  private static final MethodHandle traceSwitch = CLinker.getInstance().downcallHandle(
      SymbolLookup.loaderLookup().lookup("__sanitizer_cov_trace_switch").get(),
      MethodType.methodType(void.class, long.class, MemoryAddress.class),
      FunctionDescriptor.ofVoid(CLinker.C_LONG, CLinker.C_POINTER));

  static void traceCmpInt(int arg1, int arg2, int pc) throws Throwable {
    traceCmp4.invokeExact(arg1, arg2);
  }

  static void traceCmpSwitch(long val, long[] cases, int pc) throws Throwable {
    try (ResourceScope scope = ResourceScope.newConfinedScope()) {
      MemorySegment nativeCopy = MemorySegment.allocateNative(
          MemoryLayout.sequenceLayout(cases.length, CLinker.C_LONG), scope);
      nativeCopy.copyFrom(MemorySegment.ofArray(cases));
      traceSwitch.invokeExact(val, nativeCopy.address());
    }
  }
}