blob: f895c1223b10a4e7fd4f6f381afa079b5b175680 [file] [log] [blame]
/*
*
* honggfuzz - architecture dependent code (NETBSD)
* -----------------------------------------
*
* Author: Kamil Rytarowski <n54@gmx.com>
*
* 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 <sys/param.h>
#include <sys/types.h>
// clang-format on
#include <sys/ptrace.h>
#include <sys/syscall.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/utsname.h>
#include <sys/wait.h>
#include <ctype.h>
#include <dlfcn.h>
#include <errno.h>
#include <fcntl.h>
#include <inttypes.h>
#include <locale.h>
#include <poll.h>
#include <setjmp.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <unistd.h>
#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;
}