/* * * honggfuzz - architecture dependent code (NETBSD) * ----------------------------------------- * * Author: Kamil Rytarowski * * Copyright 2010-2018 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 "arch.h" // clang-format off #include #include // clang-format on #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "fuzz.h" #include "libhfcommon/common.h" #include "libhfcommon/files.h" #include "libhfcommon/log.h" #include "libhfcommon/ns.h" #include "libhfcommon/util.h" #include "netbsd/trace.h" #include "subproc.h" extern char** environ; pid_t arch_fork(run_t* run HF_ATTR_UNUSED) { pid_t pid = fork(); if (pid == -1) { return pid; } if (pid == 0) { logMutexReset(); return pid; } return pid; } bool arch_launchChild(run_t* run) { /* alarms persist across execve(), so disable it here */ alarm(0); /* Wait for the ptrace to attach now */ if (raise(SIGSTOP) == -1) { LOG_F("Couldn't stop itself"); } execve(run->args[0], (char* const*)run->args, environ); int errno_cpy = errno; alarm(1); LOG_E("execve('%s'): %s", run->args[0], strerror(errno_cpy)); return false; } void arch_prepareParentAfterFork(run_t* run) { /* Parent */ if (run->global->exe.persistent) { if (fcntl(run->persistentSock, F_SETFL, O_ASYNC) == -1) { PLOG_F("fcntl(%d, F_SETFL, O_ASYNC)", run->persistentSock); } } if (!arch_traceAttach(run)) { LOG_F("Couldn't attach to pid=%d", (int)run->pid); } } void arch_prepareParent(run_t* run HF_ATTR_UNUSED) { } static bool arch_checkWait(run_t* run) { /* All queued wait events must be tested when SIGCHLD was delivered */ for (;;) { int status; /* Wait for the whole process group of run->pid */ pid_t pid = TEMP_FAILURE_RETRY(wait6(P_SID, run->pid, &status, WALLSIG | WALTSIG | WTRAPPED | WEXITED | WUNTRACED | WCONTINUED | WSTOPPED | WNOHANG, NULL, NULL)); if (pid == 0) { return false; } if (pid == -1 && errno == ECHILD) { LOG_D("No more processes to track"); return true; } if (pid == -1) { PLOG_F("wait6(pid/session=%d) failed", (int)run->pid); } arch_traceAnalyze(run, status, pid); LOG_D("pid=%d returned with status: %s", pid, subproc_StatusToStr(status)); if (pid == run->pid && (WIFEXITED(status) || WIFSIGNALED(status))) { if (run->global->exe.persistent) { if (!fuzz_isTerminating()) { LOG_W("Persistent mode: PID %d exited with status: %s", pid, subproc_StatusToStr(status)); } } return true; } } } void arch_reapChild(run_t* run) { for (;;) { if (subproc_persistentModeStateMachine(run)) { break; } subproc_checkTimeLimit(run); subproc_checkTermination(run); if (run->global->exe.persistent) { struct pollfd pfd = { .fd = run->persistentSock, .events = POLLIN, }; int r = poll(&pfd, 1, 250 /* 0.25s */); if (r == -1 && errno != EINTR) { PLOG_F("poll(fd=%d)", run->persistentSock); } } else { /* Return with SIGIO, SIGCHLD */ const struct timespec ts = { .tv_sec = 0ULL, .tv_nsec = (1000ULL * 1000ULL * 250ULL), }; int sig = sigtimedwait(&run->global->exe.waitSigSet, NULL, &ts /* 0.25s */); if (sig == -1 && (errno != EAGAIN && errno != EINTR)) { PLOG_F("sigtimedwait(SIGIO|SIGCHLD)"); } } if (arch_checkWait(run)) { run->pid = 0; break; } } } bool arch_archInit(honggfuzz_t* hfuzz) { /* Make %'d work */ setlocale(LC_NUMERIC, "en_US.UTF-8"); if (access(hfuzz->exe.cmdline[0], X_OK) == -1) { PLOG_E("File '%s' doesn't seem to be executable", hfuzz->exe.cmdline[0]); return false; } /* Updates the important signal array based on input args */ arch_traceSignalsInit(hfuzz); return true; } bool arch_archThreadInit(run_t* run HF_ATTR_UNUSED) { return true; }