blob: 48e8835c0282a43ef14c1285fc1b2b899fed937c [file] [log] [blame]
/*
*
* honggfuzz - file operations
* -----------------------------------------
*
* 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 "files.h"
#include <dirent.h>
#include <errno.h>
#include <fcntl.h>
#include <inttypes.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/mman.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include "log.h"
bool files_writeToFd(int fd, uint8_t * buf, size_t fileSz)
{
size_t writtenSz = 0;
while (writtenSz < fileSz) {
ssize_t sz = write(fd, &buf[writtenSz], fileSz - writtenSz);
if (sz < 0 && errno == EINTR)
continue;
if (sz < 0)
return false;
writtenSz += sz;
}
return true;
}
bool files_writeStrToFd(int fd, char *str)
{
return files_writeToFd(fd, (uint8_t *) str, strlen(str));
}
bool files_readFromFd(int fd, uint8_t * buf, size_t fileSz)
{
size_t readSz = 0;
while (readSz < fileSz) {
ssize_t sz = read(fd, &buf[readSz], fileSz - readSz);
if (sz < 0 && errno == EINTR)
continue;
if (sz < 0)
return false;
readSz += sz;
}
return true;
}
bool files_exists(char *fileName)
{
return (access(fileName, F_OK) != -1);
}
bool files_writePatternToFd(int fd, off_t size, unsigned char p)
{
void *buf = malloc(size);
if (!buf) {
LOGMSG_P(l_WARN, "Couldn't allocate memory");
return false;
}
memset(buf, p, (size_t) size);
int ret = files_writeToFd(fd, buf, size);
free(buf);
return ret;
}
void files_unmapFileCloseFd(void *ptr, size_t fileSz, int fd)
{
munmap(ptr, _HF_PAGE_ALIGN_UP(fileSz));
close(fd);
}
void files_unmapFileCloseFdMSync(void *ptr, size_t fileSz, int fd)
{
msync(ptr, fileSz, MS_SYNC);
files_unmapFileCloseFd(ptr, fileSz, fd);
}
uint8_t *files_mapFileToRead(char *fileName, size_t * fileSz, int *fd)
{
if ((*fd = open(fileName, O_RDONLY)) == -1) {
LOGMSG_P(l_WARN, "Couldn't open() '%s' file in R/O mode", fileName);
return NULL;
}
struct stat st;
if (fstat(*fd, &st) == -1) {
LOGMSG_P(l_WARN, "Couldn't stat() the '%s' file", fileName);
close(*fd);
return NULL;
}
uint8_t *buf;
if ((buf =
mmap(NULL, _HF_PAGE_ALIGN_UP(st.st_size), PROT_READ | PROT_WRITE, MAP_PRIVATE, *fd,
0)) == MAP_FAILED) {
LOGMSG_P(l_WARN, "Couldn't mmap() the '%s' file", fileName);
close(*fd);
return NULL;
}
LOGMSG(l_DEBUG, "mmap()'d '%llu' bytes for the file '%s' (original size: '%llu') at 0x%p",
(unsigned long long)_HF_PAGE_ALIGN_UP(st.st_size), fileName,
(unsigned long long)st.st_size, buf);
*fileSz = st.st_size;
return buf;
}
uint8_t *files_mapFileToWriteIni(char *fileName, size_t fileSz, int *fd, uint8_t * iniBuf)
{
if ((*fd = open(fileName, O_CREAT | O_EXCL | O_RDWR, 0644)) == -1) {
LOGMSG_P(l_WARN, "Couldn't open() '%s' file in R/W mode", fileName);
return NULL;
}
if (ftruncate(*fd, fileSz) == -1) {
LOGMSG_P(l_ERROR, "Couldn't ftruncate(fd='%d', size='%zu')", *fd, fileSz);
close(*fd);
return NULL;
}
uint8_t *buf;
if ((buf =
mmap(NULL, _HF_PAGE_ALIGN_UP(fileSz), PROT_READ | PROT_WRITE, MAP_SHARED, *fd,
0)) == MAP_FAILED) {
LOGMSG_P(l_WARN, "Couldn't mmap() the '%s' file, size '%zu'", fileName, fileSz);
close(*fd);
return NULL;
}
LOGMSG(l_DEBUG, "mmap()'d '%llu' bytes for the file '%s' (original size: '%llu') at 0x%p",
(unsigned long long)_HF_PAGE_ALIGN_UP(fileSz), fileName, (unsigned long long)fileSz,
buf);
memcpy(buf, iniBuf, fileSz);
return buf;
}
static bool files_readdir(honggfuzz_t * hfuzz)
{
DIR *dir = opendir(hfuzz->inputFile);
if (!dir) {
LOGMSG_P(l_ERROR, "Couldn't open dir '%s'", hfuzz->inputFile);
return false;
}
int count = 0;
for (;;) {
struct dirent de, *res;
if (readdir_r(dir, &de, &res) > 0) {
LOGMSG_P(l_ERROR, "Couldn't read the '%s' dir", hfuzz->inputFile);
closedir(dir);
return false;
}
if (res == NULL && count > 0) {
LOGMSG(l_INFO, "%d input files have been added to the list", hfuzz->fileCnt);
closedir(dir);
return true;
}
if (res == NULL && count == 0) {
LOGMSG(l_ERROR, "Directory '%s' doesn't contain any regular files", hfuzz->inputFile);
closedir(dir);
return false;
}
char path[PATH_MAX];
snprintf(path, sizeof(path), "%s/%s", hfuzz->inputFile, res->d_name);
struct stat st;
if (stat(path, &st) == -1) {
LOGMSG(l_WARN, "Couldn't stat() the '%s' file", path);
continue;
}
if (!S_ISREG(st.st_mode)) {
LOGMSG(l_DEBUG, "'%s' is not a regular file, skipping", path);
continue;
}
if (st.st_size == 0) {
LOGMSG(l_DEBUG, "'%s' is empty", path);
continue;
}
if (st.st_size > (off_t) hfuzz->maxFileSz) {
LOGMSG(l_WARN,
"File '%s' is bigger than maximal defined file size (-F): %" PRId64 " > %"
PRId64, path, (int64_t) st.st_size, (int64_t) hfuzz->maxFileSz);
continue;
}
if (!(hfuzz->files = realloc(hfuzz->files, sizeof(char *) * (count + 1)))) {
LOGMSG_P(l_ERROR, "Couldn't allocate memory");
closedir(dir);
return false;
}
hfuzz->files[count] = strdup(path);
if (!hfuzz->files[count]) {
LOGMSG_P(l_ERROR, "Couldn't allocate memory");
closedir(dir);
return false;
}
hfuzz->fileCnt = ++count;
LOGMSG(l_DEBUG, "Added '%s' to the list of input files", path);
}
abort(); /* NOTREACHED */
return false;
}
bool files_init(honggfuzz_t * hfuzz)
{
hfuzz->files = malloc(sizeof(char *));
if (hfuzz->dynFileMethod != _HF_DYNFILE_NONE && !hfuzz->inputFile) {
hfuzz->fileCnt = 1;
hfuzz->files[0] = "DYNAMIC_FILE";
return true;
}
if (hfuzz->externalCommand && !hfuzz->inputFile) {
hfuzz->fileCnt = 1;
hfuzz->files[0] = "CREATED";
LOGMSG(l_INFO,
"No input file corpus specified, the external command '%s' is responsible for creating the fuzz files",
hfuzz->externalCommand);
return true;
}
if (!hfuzz->files) {
LOGMSG_P(l_ERROR, "Couldn't allocate memory");
return false;
}
if (!hfuzz->inputFile) {
LOGMSG(l_ERROR, "No input file/dir specified");
return false;
}
struct stat st;
if (stat(hfuzz->inputFile, &st) == -1) {
LOGMSG_P(l_ERROR, "Couldn't stat the input file/dir '%s'", hfuzz->inputFile);
return false;
}
if (st.st_size > (off_t) hfuzz->maxFileSz) {
LOGMSG(l_ERROR,
"File '%s' is bigger than maximal defined file size (-F): %" PRId64 " > %" PRId64,
hfuzz->inputFile, (int64_t) st.st_size, (int64_t) hfuzz->maxFileSz);
return false;
}
if (S_ISDIR(st.st_mode)) {
return files_readdir(hfuzz);
}
if (!S_ISREG(st.st_mode)) {
LOGMSG(l_ERROR, "'%s' is not a regular file, nor a directory", hfuzz->inputFile);
return false;
}
hfuzz->files[0] = hfuzz->inputFile;
hfuzz->fileCnt = 1;
return true;
}
char *files_basename(char *path)
{
char *base = strrchr(path, '/');
return base ? base + 1 : path;
}