blob: a30f09b7e3f9b20772310f79d682f0bdb7e8aeec [file] [log] [blame]
/*
*
* honggfuzz - fuzzing routines
* -----------------------------------------
*
* Author:
* Robert Swiecki <swiecki@google.com>
* Felix Gröbert <groebert@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 "fuzz.h"
#include <errno.h>
#include <fcntl.h>
#include <inttypes.h>
#include <pthread.h>
#include <signal.h>
#include <stddef.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/mman.h>
#include <sys/param.h>
#include <sys/stat.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <time.h>
#include <unistd.h>
#include "arch.h"
#include "files.h"
#include "log.h"
#include "mangle.h"
#include "report.h"
#include "util.h"
static void fuzz_getFileName(honggfuzz_t * hfuzz, char *fileName)
{
struct timeval tv;
gettimeofday(&tv, NULL);
snprintf(fileName, PATH_MAX, ".honggfuzz.%d.%lu.%lu.%lu.%lu.%s",
(int)getpid(), (unsigned long int)tv.tv_sec,
(unsigned long int)tv.tv_usec,
(unsigned long int)util_rndGet(0, 1 << 30), (unsigned long int)util_rndGet(0, 1 << 30),
hfuzz->fileExtn);
return;
}
static bool fuzz_prepareFileDynamically(honggfuzz_t * hfuzz, fuzzer_t * fuzzer, int rnd_index)
{
while (pthread_mutex_lock(&hfuzz->dynamicFile_mutex)) ;
if (hfuzz->inputFile && hfuzz->branchBestCnt == 0) {
int srcfd = open(hfuzz->files[rnd_index], O_RDONLY);
if (srcfd == -1) {
LOGMSG_P(l_ERROR, "Couldn't open '%s' to read", hfuzz->files[rnd_index]);
while (pthread_mutex_unlock(&hfuzz->dynamicFile_mutex)) ;
return false;
}
struct stat sbuf;
if (fstat(srcfd, &sbuf) == -1) {
LOGMSG_P(l_ERROR, "Couldn't stat '%s' (fd='%d')", hfuzz->files[rnd_index], srcfd);
while (pthread_mutex_unlock(&hfuzz->dynamicFile_mutex)) ;
close(srcfd);
return false;
}
if ((size_t) sbuf.st_size > hfuzz->maxFileSz) {
while (pthread_mutex_unlock(&hfuzz->dynamicFile_mutex)) ;
close(srcfd);
LOGMSG(l_ERROR,
"File size (%zu) is bigger than the maximal allocated buffer/file size (-F) (%zu)",
sbuf.st_size, hfuzz->maxFileSz);
return false;
}
if (files_readFromFd(srcfd, hfuzz->dynamicFileBest, sbuf.st_size) == false) {
while (pthread_mutex_unlock(&hfuzz->dynamicFile_mutex)) ;
close(srcfd);
LOGMSG(l_ERROR, "Could read '%zu' bytes from '%s' (fd='%d')", sbuf.st_size,
hfuzz->files[rnd_index], srcfd);
return false;
}
hfuzz->dynamicFileBestSz = sbuf.st_size;
hfuzz->branchBestCnt = 1;
close(srcfd);
}
memcpy(fuzzer->dynamicFile, hfuzz->dynamicFileBest, hfuzz->dynamicFileBestSz);
fuzzer->dynamicFileSz = hfuzz->dynamicFileBestSz;
while (pthread_mutex_unlock(&hfuzz->dynamicFile_mutex)) ;
int dstfd;
uint8_t *buf = files_mapFileToWriteIni(fuzzer->fileName, fuzzer->dynamicFileSz, &dstfd,
fuzzer->dynamicFile);
if (buf == NULL) {
LOGMSG(l_ERROR, "Couldn't map file '%s' in R/W mode, size=%zu", fuzzer->fileName,
fuzzer->dynamicFile);
return false;
}
/* The first pass should be on an empty/initial file */
if (hfuzz->branchBestCnt > 0) {
if (mangle_Resize(hfuzz, NULL, &fuzzer->dynamicFileSz, -1) == false) {
return false;
}
mangle_mangleContent(hfuzz, fuzzer->dynamicFile, fuzzer->dynamicFileSz);
}
files_unmapFileCloseFdMSync(buf, fuzzer->dynamicFileSz, dstfd);
return true;
}
static bool fuzz_prepareFile(honggfuzz_t * hfuzz, char *fileName, int rnd_index)
{
int srcfd = open(hfuzz->files[rnd_index], O_RDONLY);
if (srcfd == -1) {
LOGMSG_P(l_ERROR, "Couldn't open '%s' for R/O", hfuzz->files[rnd_index]);
return false;
}
struct stat sbuf;
if (fstat(srcfd, &sbuf) == -1) {
LOGMSG_P(l_ERROR, "Couldn't fstat(fd='%s' fname='%s')", srcfd, hfuzz->files[rnd_index]);
close(srcfd);
return false;
}
size_t fileSz = (size_t) sbuf.st_size;
uint8_t *tmpbuf = malloc(fileSz);
if (tmpbuf == NULL) {
LOGMSG_P(l_ERROR, "Couldn't malloc(size='%zu')", fileSz);
close(srcfd);
return false;
}
if (files_readFromFd(srcfd, tmpbuf, fileSz) == false) {
LOGMSG(l_ERROR, "Couldn't read '%zu' bytes from '%s' (fd='%d')", fileSz,
hfuzz->files[rnd_index], srcfd);
close(srcfd);
free(tmpbuf);
return false;
}
close(srcfd);
int dstfd;
uint8_t *buf = files_mapFileToWriteIni(fileName, fileSz, &dstfd, tmpbuf);
if (buf == NULL) {
LOGMSG(l_ERROR, "Couldn't map '%s' for R/W, size='%zu'", fileName, fileSz);
free(tmpbuf);
return false;
}
free(tmpbuf);
if (mangle_Resize(hfuzz, &buf, &fileSz, dstfd) == false) {
files_unmapFileCloseFd(buf, fileSz, dstfd);
close(dstfd);
LOGMSG(l_ERROR, "File resizing failed");
return false;
}
mangle_mangleContent(hfuzz, buf, fileSz);
files_unmapFileCloseFdMSync(buf, fileSz, dstfd);
return true;
}
static bool fuzz_prepareFileExternally(honggfuzz_t * hfuzz, char *fileName, int rnd_index)
{
size_t fileSz;
int srcfd;
int dstfd = open(fileName, O_CREAT | O_EXCL | O_RDWR, 0644);
if (dstfd == -1) {
LOGMSG_P(l_ERROR, "Couldn't create a temporary file '%s' in the current directory",
fileName);
return false;
}
LOGMSG(l_DEBUG, "Created '%f' as an input file", fileName);
if (hfuzz->inputFile) {
uint8_t *buf = files_mapFileToRead(hfuzz->files[rnd_index], &fileSz, &srcfd);
if (buf == NULL) {
LOGMSG(l_ERROR, "Couldn't open and map '%s' in R/O mode", hfuzz->files[rnd_index]);
close(dstfd);
return false;
}
LOGMSG(l_DEBUG, "Mmaped '%s' in R/O mode, size: %d", hfuzz->files[rnd_index], fileSz);
bool ret = files_writeToFd(dstfd, buf, fileSz);
files_unmapFileCloseFd(buf, fileSz, srcfd);
if (!ret) {
close(dstfd);
return false;
}
}
close(dstfd);
pid_t pid = vfork();
if (pid == -1) {
LOGMSG_P(l_ERROR, "Couldn't vfork");
return false;
}
if (!pid) {
/*
* child performs the external file modifications
*/
execl(hfuzz->externalCommand, hfuzz->externalCommand, fileName, NULL);
LOGMSG_P(l_FATAL, "Couldn't execute '%s %s'", hfuzz->externalCommand, fileName);
return false;
}
/*
* parent waits until child is done fuzzing the input file
*/
int childStatus;
int flags = 0;
#if defined(__WNOTHREAD)
flags |= __WNOTHREAD;
#endif /* defined(__WNOTHREAD) */
while (wait4(pid, &childStatus, flags, NULL) != pid) ;
if (WIFEXITED(childStatus)) {
LOGMSG(l_DEBUG, "External command exited with status %d", WEXITSTATUS(childStatus));
return true;
}
if (WIFSIGNALED(childStatus)) {
LOGMSG(l_ERROR, "External command terminated with signal %d", WTERMSIG(childStatus));
return false;
}
LOGMSG(l_FATAL, "External command terminated abnormally, status: %d", childStatus);
return false;
abort(); /* NOTREACHED */
}
static int fuzz_numOfProc(honggfuzz_t * hfuzz)
{
int i;
sem_getvalue(hfuzz->sem, &i);
return hfuzz->threadsMax - i;
}
static void *fuzz_threadNew(void *arg)
{
honggfuzz_t *hfuzz = (honggfuzz_t *) arg;
fuzzer_t fuzzer = {
.pid = 0,
.timeStarted = time(NULL),
.pc = 0ULL,
.backtrace = 0ULL,
.access = 0ULL,
.exception = 0,
.dynamicFileSz = 0,
.dynamicFile = malloc(hfuzz->maxFileSz),
.branchCnt = 0,
.report = {'\0'}
};
if (fuzzer.dynamicFile == NULL) {
LOGMSG(l_FATAL, "malloc(%zu) failed", hfuzz->maxFileSz);
}
int rnd_index = util_rndGet(0, hfuzz->fileCnt - 1);
strncpy(fuzzer.origFileName, files_basename(hfuzz->files[rnd_index]), PATH_MAX);
fuzz_getFileName(hfuzz, fuzzer.fileName);
if (hfuzz->dynFileMethod != _HF_DYNFILE_NONE) {
if (!fuzz_prepareFileDynamically(hfuzz, &fuzzer, rnd_index)) {
exit(EXIT_FAILURE);
}
} else if (hfuzz->externalCommand != NULL) {
if (!fuzz_prepareFileExternally(hfuzz, fuzzer.fileName, rnd_index)) {
exit(EXIT_FAILURE);
}
} else {
if (!fuzz_prepareFile(hfuzz, fuzzer.fileName, rnd_index)) {
exit(EXIT_FAILURE);
}
}
#if defined(_HF_ARCH_LINUX)
#include <unistd.h>
#include <sys/syscall.h>
fuzzer.pid = syscall(__NR_fork);
#else /* defined(_HF_ARCH_LINUX) */
fuzzer.pid = fork();
#endif /* defined(_HF_ARCH_LINUX) */
if (fuzzer.pid == -1) {
LOGMSG_P(l_FATAL, "Couldn't fork");
exit(EXIT_FAILURE);
}
if (!fuzzer.pid) {
/*
* Ok, kill the parent if this fails
*/
if (!arch_launchChild(hfuzz, fuzzer.fileName)) {
LOGMSG(l_ERROR, "Error launching child process, killing parent");
exit(EXIT_FAILURE);
}
}
LOGMSG(l_INFO, "Launched new process, pid: %d, (%d/%d)", fuzzer.pid, fuzz_numOfProc(hfuzz),
hfuzz->threadsMax);
arch_reapChild(hfuzz, &fuzzer);
unlink(fuzzer.fileName);
if (hfuzz->dynFileMethod != _HF_DYNFILE_NONE) {
while (pthread_mutex_lock(&hfuzz->dynamicFile_mutex)) ;
if (fuzzer.branchCnt >= hfuzz->branchBestCnt) {
LOGMSG(l_INFO,
"New BEST feedback: New/Old: %zu/%zu', Branch events Curr/High: %" PRId64 "/%"
PRId64, fuzzer.dynamicFileSz, hfuzz->dynamicFileBestSz, fuzzer.branchCnt,
hfuzz->branchBestCnt);
memcpy(hfuzz->dynamicFileBest, fuzzer.dynamicFile, fuzzer.dynamicFileSz);
hfuzz->dynamicFileBestSz = fuzzer.dynamicFileSz;
hfuzz->branchBestCnt = fuzzer.branchCnt;
#define _HF_CURRENT_BEST "CURRENT_BEST"
#define _HF_CURRENT_BEST_TMP ".tmp.CURRENT_BEST"
int fd = open(_HF_CURRENT_BEST_TMP, O_CREAT | O_WRONLY | O_TRUNC, 0644);
if (fd != -1) {
if (files_writeToFd(fd, fuzzer.dynamicFile, fuzzer.dynamicFileSz)) {
rename(_HF_CURRENT_BEST_TMP, _HF_CURRENT_BEST);
}
unlink(_HF_CURRENT_BEST_TMP);
close(fd);
}
}
while (pthread_mutex_unlock(&hfuzz->dynamicFile_mutex)) ;
}
report_Report(hfuzz, fuzzer.report);
free(fuzzer.dynamicFile);
sem_post(hfuzz->sem);
return NULL;
}
static void *fuzz_threadPid(void *arg)
{
honggfuzz_t *hfuzz = (honggfuzz_t *) arg;
if (!arch_archInit(hfuzz)) {
LOGMSG(l_FATAL, "Couldn't prepare parent for fuzzing");
}
fuzzer_t fuzzer = {
.pid = hfuzz->pid,
.timeStarted = time(NULL),
.pc = 0ULL,
.backtrace = 0ULL,
.access = 0ULL,
.exception = 0,
.dynamicFileSz = 0,
.dynamicFile = malloc(hfuzz->maxFileSz),
.branchCnt = 0,
.report = {'\0'}
};
if (fuzzer.dynamicFile == NULL) {
LOGMSG(l_FATAL, "malloc(%zu) failed", hfuzz->maxFileSz);
}
char fileName[] = ".honggfuzz.empty.XXXXXX";
int fd;
if ((fd = mkstemp(fileName)) == -1) {
free(fuzzer.dynamicFile);
LOGMSG_P(l_ERROR, "Couldn't create a temporary file");
return NULL;
}
close(fd);
strncpy(fuzzer.origFileName, "PID_FUZZING", PATH_MAX);
strncpy(fuzzer.fileName, fileName, PATH_MAX);
arch_reapChild(hfuzz, &fuzzer);
unlink(fuzzer.fileName);
report_Report(hfuzz, fuzzer.report);
free(fuzzer.dynamicFile);
// There's no more hfuzz->pid to analyze. Just exit
LOGMSG(l_INFO, "PID: %d exited. Exiting", fuzzer.pid);
exit(EXIT_SUCCESS);
return NULL;
}
static void fuzz_runThread(honggfuzz_t * hfuzz, void *(*thread) (void *))
{
pthread_attr_t attr;
pthread_attr_init(&attr);
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);
pthread_attr_setstacksize(&attr, _HF_PTHREAD_STACKSIZE);
pthread_attr_setguardsize(&attr, (size_t) sysconf(_SC_PAGESIZE));
pthread_t t;
if (pthread_create(&t, &attr, thread, (void *)hfuzz) < 0) {
LOGMSG_P(l_FATAL, "Couldn't create a new thread");
}
return;
}
void fuzz_main(honggfuzz_t * hfuzz)
{
char semName[PATH_MAX];
snprintf(semName, sizeof(semName), "/honggfuzz.%d.%d.%" PRIx64, getpid(),
(int)time(NULL), util_rndGet(1, 1ULL << 62));
hfuzz->sem = sem_open(semName, O_CREAT, 0644, hfuzz->threadsMax);
if (hfuzz->sem == SEM_FAILED) {
LOGMSG_P(l_FATAL, "sem_open() failed");
}
// If we're doing a PID fuzzing, the parent of the PID will be a
// dedicated thread anyway
if (hfuzz->pid) {
fuzz_runThread(hfuzz, fuzz_threadPid);
} else {
if (!arch_archInit(hfuzz)) {
LOGMSG(l_FATAL, "Couldn't prepare parent for fuzzing");
}
}
for (;;) {
if (sem_wait(hfuzz->sem) == -1) {
LOGMSG_P(l_FATAL, "sem_wait() failed");
}
if (hfuzz->mutationsMax && (hfuzz->mutationsCnt >= hfuzz->mutationsMax)) {
/*
* Sleep a bit to let any running fuzzers terminate
*/
usleep(1.2 * hfuzz->tmOut * 1000000);
LOGMSG(l_INFO, "Finished fuzzing %ld times.", hfuzz->mutationsMax);
sem_destroy(hfuzz->sem);
exit(EXIT_SUCCESS);
}
hfuzz->mutationsCnt++;
fuzz_runThread(hfuzz, fuzz_threadNew);
}
}