| /* |
| * |
| * honggfuzz - architecture dependent code (LINUX/PERF) |
| * ----------------------------------------- |
| * |
| * Author: Robert Swiecki <swiecki@google.com> |
| * |
| * Copyright 2010-2015 by Google Inc. All Rights Reserved. |
| * |
| * 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. |
| * |
| */ |
| |
| #include "../common.h" |
| #include "perf.h" |
| |
| #include <asm/mman.h> |
| #include <errno.h> |
| #include <fcntl.h> |
| #include <inttypes.h> |
| #include <linux/hw_breakpoint.h> |
| #include <linux/perf_event.h> |
| #include <linux/sysctl.h> |
| #include <signal.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <sys/ioctl.h> |
| #include <sys/mman.h> |
| #include <sys/poll.h> |
| #include <sys/ptrace.h> |
| #include <sys/syscall.h> |
| #include <unistd.h> |
| |
| #include "../files.h" |
| #include "../log.h" |
| #include "../util.h" |
| #include "pt.h" |
| |
| #define _HF_PERF_MAP_SZ (1024 * 512) |
| #define _HF_PERF_AUX_SZ (1024 * 1024) |
| /* PERF_TYPE for Intel_PT/BTS -1 if none */ |
| static int32_t perfIntelPtPerfType = -1; |
| static int32_t perfIntelBtsPerfType = -1; |
| |
| #if defined(PERF_ATTR_SIZE_VER5) |
| static inline void arch_perfBtsCount(honggfuzz_t * hfuzz, fuzzer_t * fuzzer) |
| { |
| struct perf_event_mmap_page *pem = (struct perf_event_mmap_page *)fuzzer->linux.perfMmapBuf; |
| struct bts_branch { |
| uint64_t from; |
| uint64_t to; |
| uint64_t misc; |
| }; |
| |
| uint64_t aux_head = ATOMIC_GET(pem->aux_head); |
| struct bts_branch *br = (struct bts_branch *)fuzzer->linux.perfMmapAux; |
| for (; br < ((struct bts_branch *)(fuzzer->linux.perfMmapAux + aux_head)); br++) { |
| /* |
| * Kernel sometimes reports branches from the kernel (iret), we are not interested in that as it |
| * makes the whole concept of unique branch counting less predictable |
| */ |
| if (hfuzz->linux.kernelOnly == false |
| && (__builtin_expect(br->from > 0xFFFFFFFF00000000, false) |
| || __builtin_expect(br->to > 0xFFFFFFFF00000000, false))) { |
| LOG_D("Adding branch %#018" PRIx64 " - %#018" PRIx64, br->from, br->to); |
| continue; |
| } |
| if (br->from >= hfuzz->linux.dynamicCutOffAddr || br->to >= hfuzz->linux.dynamicCutOffAddr) { |
| continue; |
| } |
| |
| register size_t pos = br->to; |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BTS_EDGE) { |
| pos = ((br->from << 12) ^ (br->to & 0xFFF)); |
| } |
| pos &= _HF_PERF_BITMAP_BITSZ_MASK; |
| register uint8_t prev = ATOMIC_BTS(hfuzz->feedback->bbMapPc, pos); |
| if (!prev) { |
| fuzzer->linux.hwCnts.newBBCnt++; |
| } |
| } |
| } |
| #endif /* defined(PERF_ATTR_SIZE_VER5) */ |
| |
| static inline void arch_perfMmapParse(honggfuzz_t * hfuzz UNUSED, fuzzer_t * fuzzer UNUSED) |
| { |
| #if defined(PERF_ATTR_SIZE_VER5) |
| struct perf_event_mmap_page *pem = (struct perf_event_mmap_page *)fuzzer->linux.perfMmapBuf; |
| if (pem->aux_head == pem->aux_tail) { |
| return; |
| } |
| if (pem->aux_head < pem->aux_tail) { |
| LOG_F("The PERF AUX data has been overwritten. The AUX buffer is too small"); |
| } |
| |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BTS_BLOCK || hfuzz->dynFileMethod & _HF_DYNFILE_BTS_EDGE) { |
| arch_perfBtsCount(hfuzz, fuzzer); |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_IPT_BLOCK) { |
| arch_ptAnalyze(hfuzz, fuzzer); |
| } |
| #endif /* defined(PERF_ATTR_SIZE_VER5) */ |
| } |
| |
| static long perf_event_open(struct perf_event_attr *hw_event, pid_t pid, int cpu, int group_fd, |
| unsigned long flags) |
| { |
| return syscall(__NR_perf_event_open, hw_event, (uintptr_t) pid, (uintptr_t) cpu, |
| (uintptr_t) group_fd, (uintptr_t) flags); |
| } |
| |
| static bool arch_perfCreate(honggfuzz_t * hfuzz, fuzzer_t * fuzzer UNUSED, pid_t pid, |
| dynFileMethod_t method, int *perfFd) |
| { |
| LOG_D("Enabling PERF for PID=%d method=%x", pid, method); |
| |
| if (*perfFd != -1) { |
| LOG_F("The PERF FD is already initialized, possibly conflicting perf types enabled"); |
| } |
| |
| if (((method & _HF_DYNFILE_BTS_BLOCK) || method & _HF_DYNFILE_BTS_EDGE) |
| && perfIntelBtsPerfType == -1) { |
| LOG_F("Intel BTS events (new type) are not supported on this platform"); |
| } |
| if ((method & _HF_DYNFILE_IPT_BLOCK) |
| && perfIntelPtPerfType == -1) { |
| LOG_F("Intel PT events are not supported on this platform"); |
| } |
| |
| struct perf_event_attr pe; |
| memset(&pe, 0, sizeof(struct perf_event_attr)); |
| pe.size = sizeof(struct perf_event_attr); |
| if (hfuzz->linux.kernelOnly) { |
| pe.exclude_user = 1; |
| } else { |
| pe.exclude_kernel = 1; |
| } |
| if (hfuzz->linux.pid > 0 || hfuzz->persistent == true) { |
| pe.disabled = 0; |
| pe.enable_on_exec = 0; |
| } else { |
| pe.disabled = 1; |
| pe.enable_on_exec = 1; |
| } |
| pe.type = PERF_TYPE_HARDWARE; |
| |
| switch (method) { |
| case _HF_DYNFILE_INSTR_COUNT: |
| LOG_D("Using: PERF_COUNT_HW_INSTRUCTIONS for PID: %d", pid); |
| pe.config = PERF_COUNT_HW_INSTRUCTIONS; |
| pe.inherit = 1; |
| break; |
| case _HF_DYNFILE_BRANCH_COUNT: |
| LOG_D("Using: PERF_COUNT_HW_BRANCH_INSTRUCTIONS for PID: %d", pid); |
| pe.config = PERF_COUNT_HW_BRANCH_INSTRUCTIONS; |
| pe.inherit = 1; |
| break; |
| case _HF_DYNFILE_BTS_BLOCK: |
| LOG_D("Using: (Intel BTS) type=%" PRIu32 " for PID: %d", perfIntelBtsPerfType, pid); |
| pe.type = perfIntelBtsPerfType; |
| break; |
| case _HF_DYNFILE_BTS_EDGE: |
| LOG_D("Using: (Intel BTS) type=%" PRIu32 " for PID: %d", perfIntelBtsPerfType, pid); |
| pe.type = perfIntelBtsPerfType; |
| break; |
| case _HF_DYNFILE_IPT_BLOCK: |
| LOG_D("Using: (Intel PT) type=%" PRIu32 " for PID: %d", perfIntelPtPerfType, pid); |
| pe.type = perfIntelPtPerfType; |
| pe.config = (1U << 11); /* Disable RETCompression */ |
| break; |
| default: |
| LOG_E("Unknown perf mode: '%d' for PID: %d", method, pid); |
| return false; |
| break; |
| } |
| |
| #if !defined(PERF_FLAG_FD_CLOEXEC) |
| #define PERF_FLAG_FD_CLOEXEC 0 |
| #endif |
| *perfFd = perf_event_open(&pe, pid, -1, -1, PERF_FLAG_FD_CLOEXEC); |
| if (*perfFd == -1) { |
| PLOG_F("perf_event_open() failed"); |
| return false; |
| } |
| |
| if (method != _HF_DYNFILE_BTS_BLOCK && method != _HF_DYNFILE_BTS_EDGE |
| && method != _HF_DYNFILE_IPT_BLOCK) { |
| return true; |
| } |
| #if defined(PERF_ATTR_SIZE_VER5) |
| fuzzer->linux.perfMmapBuf = |
| mmap(NULL, _HF_PERF_MAP_SZ + getpagesize(), PROT_READ | PROT_WRITE, MAP_SHARED, *perfFd, 0); |
| if (fuzzer->linux.perfMmapBuf == MAP_FAILED) { |
| fuzzer->linux.perfMmapBuf = NULL; |
| PLOG_W("mmap(mmapBuf) failed, sz=%zu, try increasing the kernel.perf_event_mlock_kb " |
| "sysctl (up to even 300000000)", (size_t) _HF_PERF_MAP_SZ + getpagesize()); |
| close(*perfFd); |
| return false; |
| } |
| |
| struct perf_event_mmap_page *pem = (struct perf_event_mmap_page *)fuzzer->linux.perfMmapBuf; |
| pem->aux_offset = pem->data_offset + pem->data_size; |
| pem->aux_size = _HF_PERF_AUX_SZ; |
| fuzzer->linux.perfMmapAux = |
| mmap(NULL, pem->aux_size, PROT_READ, MAP_SHARED, *perfFd, pem->aux_offset); |
| |
| if (fuzzer->linux.perfMmapAux == MAP_FAILED) { |
| munmap(fuzzer->linux.perfMmapBuf, _HF_PERF_MAP_SZ + getpagesize()); |
| fuzzer->linux.perfMmapBuf = NULL; |
| PLOG_W("mmap(mmapAuxBuf) failed, try increasing the kernel.perf_event_mlock_kb " |
| "sysctl (up to even 300000000)"); |
| close(*perfFd); |
| return false; |
| } |
| #else /* defined(PERF_ATTR_SIZE_VER5) */ |
| LOG_F("Your <linux/perf_event.h> includes are too old to support Intel PT/BTS"); |
| #endif /* defined(PERF_ATTR_SIZE_VER5) */ |
| |
| return true; |
| } |
| |
| bool arch_perfOpen(pid_t pid, honggfuzz_t * hfuzz, fuzzer_t * fuzzer) |
| { |
| if (hfuzz->dynFileMethod == _HF_DYNFILE_NONE) { |
| return true; |
| } |
| |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_INSTR_COUNT) { |
| if (arch_perfCreate(hfuzz, fuzzer, pid, _HF_DYNFILE_INSTR_COUNT, &fuzzer->linux.cpuInstrFd) |
| == false) { |
| LOG_E("Cannot set up perf for PID=%d (_HF_DYNFILE_INSTR_COUNT)", pid); |
| goto out; |
| } |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BRANCH_COUNT) { |
| if (arch_perfCreate |
| (hfuzz, fuzzer, pid, _HF_DYNFILE_BRANCH_COUNT, &fuzzer->linux.cpuBranchFd) |
| == false) { |
| LOG_E("Cannot set up perf for PID=%d (_HF_DYNFILE_BRANCH_COUNT)", pid); |
| goto out; |
| } |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BTS_BLOCK) { |
| if (arch_perfCreate(hfuzz, fuzzer, pid, _HF_DYNFILE_BTS_BLOCK, &fuzzer->linux.cpuIptBtsFd) |
| == false) { |
| LOG_E("Cannot set up perf for PID=%d (_HF_DYNFILE_BTS_BLOCK)", pid); |
| goto out; |
| } |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BTS_EDGE) { |
| if (arch_perfCreate(hfuzz, fuzzer, pid, _HF_DYNFILE_BTS_EDGE, &fuzzer->linux.cpuIptBtsFd) == |
| false) { |
| LOG_E("Cannot set up perf for PID=%d (_HF_DYNFILE_BTS_EDGE)", pid); |
| goto out; |
| } |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_IPT_BLOCK) { |
| if (arch_perfCreate(hfuzz, fuzzer, pid, _HF_DYNFILE_IPT_BLOCK, &fuzzer->linux.cpuIptBtsFd) |
| == false) { |
| LOG_E("Cannot set up perf for PID=%d (_HF_DYNFILE_IPT_BLOCK)", pid); |
| goto out; |
| } |
| } |
| |
| return true; |
| |
| out: |
| close(fuzzer->linux.cpuInstrFd); |
| close(fuzzer->linux.cpuBranchFd); |
| close(fuzzer->linux.cpuIptBtsFd); |
| |
| return false; |
| } |
| |
| void arch_perfClose(honggfuzz_t * hfuzz, fuzzer_t * fuzzer) |
| { |
| if (hfuzz->dynFileMethod == _HF_DYNFILE_NONE) { |
| return; |
| } |
| |
| if (fuzzer->linux.perfMmapAux != NULL) { |
| munmap(fuzzer->linux.perfMmapAux, _HF_PERF_AUX_SZ); |
| fuzzer->linux.perfMmapAux = NULL; |
| } |
| if (fuzzer->linux.perfMmapBuf != NULL) { |
| munmap(fuzzer->linux.perfMmapBuf, _HF_PERF_MAP_SZ + getpagesize()); |
| fuzzer->linux.perfMmapBuf = NULL; |
| } |
| |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_INSTR_COUNT) { |
| close(fuzzer->linux.cpuInstrFd); |
| fuzzer->linux.cpuInstrFd = -1; |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BRANCH_COUNT) { |
| close(fuzzer->linux.cpuBranchFd); |
| fuzzer->linux.cpuBranchFd = -1; |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BTS_BLOCK) { |
| close(fuzzer->linux.cpuIptBtsFd); |
| fuzzer->linux.cpuIptBtsFd = -1; |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BTS_EDGE) { |
| close(fuzzer->linux.cpuIptBtsFd); |
| fuzzer->linux.cpuIptBtsFd = -1; |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_IPT_BLOCK) { |
| close(fuzzer->linux.cpuIptBtsFd); |
| fuzzer->linux.cpuIptBtsFd = -1; |
| } |
| } |
| |
| bool arch_perfEnable(honggfuzz_t * hfuzz, fuzzer_t * fuzzer) |
| { |
| if (hfuzz->dynFileMethod == _HF_DYNFILE_NONE) { |
| return true; |
| } |
| |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_INSTR_COUNT) { |
| ioctl(fuzzer->linux.cpuInstrFd, PERF_EVENT_IOC_ENABLE, 0); |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BRANCH_COUNT) { |
| ioctl(fuzzer->linux.cpuBranchFd, PERF_EVENT_IOC_ENABLE, 0); |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BTS_BLOCK) { |
| ioctl(fuzzer->linux.cpuIptBtsFd, PERF_EVENT_IOC_ENABLE, 0); |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BTS_EDGE) { |
| ioctl(fuzzer->linux.cpuIptBtsFd, PERF_EVENT_IOC_ENABLE, 0); |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_IPT_BLOCK) { |
| ioctl(fuzzer->linux.cpuIptBtsFd, PERF_EVENT_IOC_ENABLE, 0); |
| } |
| |
| return true; |
| } |
| |
| void arch_perfAnalyze(honggfuzz_t * hfuzz, fuzzer_t * fuzzer) |
| { |
| if (hfuzz->dynFileMethod == _HF_DYNFILE_NONE) { |
| return; |
| } |
| |
| uint64_t instrCount = 0; |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_INSTR_COUNT) { |
| ioctl(fuzzer->linux.cpuInstrFd, PERF_EVENT_IOC_DISABLE, 0); |
| if (files_readFromFd(fuzzer->linux.cpuInstrFd, (uint8_t *) & instrCount, sizeof(instrCount)) |
| != sizeof(instrCount)) { |
| PLOG_E("read(perfFd='%d') failed", fuzzer->linux.cpuInstrFd); |
| } |
| ioctl(fuzzer->linux.cpuInstrFd, PERF_EVENT_IOC_RESET, 0); |
| } |
| |
| uint64_t branchCount = 0; |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BRANCH_COUNT) { |
| ioctl(fuzzer->linux.cpuBranchFd, PERF_EVENT_IOC_DISABLE, 0); |
| if (files_readFromFd |
| (fuzzer->linux.cpuBranchFd, (uint8_t *) & branchCount, |
| sizeof(branchCount)) != sizeof(branchCount)) { |
| PLOG_E("read(perfFd='%d') failed", fuzzer->linux.cpuBranchFd); |
| } |
| ioctl(fuzzer->linux.cpuBranchFd, PERF_EVENT_IOC_RESET, 0); |
| } |
| |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BTS_BLOCK) { |
| ioctl(fuzzer->linux.cpuIptBtsFd, PERF_EVENT_IOC_DISABLE, 0); |
| arch_perfMmapParse(hfuzz, fuzzer); |
| ioctl(fuzzer->linux.cpuIptBtsFd, PERF_EVENT_IOC_RESET, 0); |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_BTS_EDGE) { |
| ioctl(fuzzer->linux.cpuIptBtsFd, PERF_EVENT_IOC_DISABLE, 0); |
| arch_perfMmapParse(hfuzz, fuzzer); |
| ioctl(fuzzer->linux.cpuIptBtsFd, PERF_EVENT_IOC_RESET, 0); |
| } |
| if (hfuzz->dynFileMethod & _HF_DYNFILE_IPT_BLOCK) { |
| ioctl(fuzzer->linux.cpuIptBtsFd, PERF_EVENT_IOC_DISABLE, 0); |
| arch_perfMmapParse(hfuzz, fuzzer); |
| ioctl(fuzzer->linux.cpuIptBtsFd, PERF_EVENT_IOC_RESET, 0); |
| } |
| |
| if (fuzzer->linux.perfMmapBuf != NULL) { |
| struct perf_event_mmap_page *pem = (struct perf_event_mmap_page *)fuzzer->linux.perfMmapBuf; |
| ATOMIC_SET(pem->data_head, 0); |
| ATOMIC_SET(pem->data_tail, 0); |
| #if defined(PERF_ATTR_SIZE_VER5) |
| ATOMIC_SET(pem->aux_head, 0); |
| ATOMIC_SET(pem->aux_tail, 0); |
| #endif /* defined(PERF_ATTR_SIZE_VER5) */ |
| } |
| |
| fuzzer->linux.hwCnts.cpuInstrCnt = instrCount; |
| fuzzer->linux.hwCnts.cpuBranchCnt = branchCount; |
| } |
| |
| bool arch_perfInit(honggfuzz_t * hfuzz UNUSED) |
| { |
| uint8_t buf[PATH_MAX + 1]; |
| ssize_t sz = |
| files_readFileToBufMax("/sys/bus/event_source/devices/intel_pt/type", buf, sizeof(buf) - 1); |
| if (sz > 0) { |
| buf[sz] = '\0'; |
| perfIntelPtPerfType = (int32_t) strtoul((char *)buf, NULL, 10); |
| LOG_D("perfIntelPtPerfType = %" PRIu32, perfIntelPtPerfType); |
| } |
| sz = files_readFileToBufMax("/sys/bus/event_source/devices/intel_bts/type", buf, |
| sizeof(buf) - 1); |
| if (sz > 0) { |
| buf[sz] = '\0'; |
| perfIntelBtsPerfType = (int32_t) strtoul((char *)buf, NULL, 10); |
| LOG_D("perfIntelBtsPerfType = %" PRIu32, perfIntelBtsPerfType); |
| } |
| return true; |
| } |