| /* |
| * |
| * honggfuzz - utilities |
| * ----------------------------------------- |
| * |
| * 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 <fcntl.h> |
| #include <math.h> |
| #include <stdarg.h> |
| #include <stdint.h> |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <sys/stat.h> |
| #include <sys/time.h> |
| #include <sys/types.h> |
| #include <time.h> |
| #include <unistd.h> |
| |
| #include "common.h" |
| #include "files.h" |
| #include "log.h" |
| |
| #if defined(__builtin_bswap16) |
| #define SWAP16(x) __builtin_bswap16(x) |
| #else |
| #define SWAP16(x) ((x & 0xff) << 8) | ((x & 0xff00) >> 8) |
| #endif |
| |
| #if defined(__builtin_bswap32) |
| #define SWAP32(x) __builtin_bswap32(x) |
| #else |
| #define SWAP32(x) ((x & 0xff) << 24) | ((x & 0xff00) << 8) | ((x & 0xff0000) >> 8) | ((x & 0xff000000) >> 24) |
| #endif |
| |
| static int util_urandomFd = -1; |
| |
| uint64_t util_rndGet(uint64_t min, uint64_t max) |
| { |
| if (util_urandomFd == -1) { |
| if ((util_urandomFd = open("/dev/urandom", O_RDONLY)) == -1) { |
| LOGMSG_P(l_FATAL, "Couldn't open /dev/urandom for writing"); |
| } |
| } |
| |
| uint64_t rnd; |
| if (files_readFromFd(util_urandomFd, (uint8_t *) & rnd, sizeof(rnd)) == false) { |
| LOGMSG_P(l_FATAL, "Failed reading from /dev/urandom"); |
| } |
| |
| if (min > max) { |
| LOGMSG(l_FATAL, "min:%d > max:%d", min, max); |
| } |
| |
| return ((rnd % (max - min + 1)) + min); |
| } |
| |
| void util_rndBuf(uint8_t * buf, size_t sz) |
| { |
| /* MMIX LCG PRNG */ |
| uint64_t a = 6364136223846793005ULL; |
| uint64_t c = 1442695040888963407ULL; |
| uint64_t x = util_rndGet(0, 1ULL << 60); |
| |
| for (size_t i = 0; i < sz; i++) { |
| x = (a * x + c); |
| buf[i] = (uint8_t) ((x >> 32) & 0xFF); |
| } |
| |
| return; |
| } |
| |
| int util_vssnprintf(char *str, size_t size, const char *format, va_list ap) |
| { |
| char buf1[size]; |
| char buf2[size]; |
| |
| strncpy(buf1, str, size); |
| |
| vsnprintf(buf2, size, format, ap); |
| |
| return snprintf(str, size, "%s%s", buf1, buf2); |
| } |
| |
| int util_ssnprintf(char *str, size_t size, const char *format, ...) |
| { |
| char buf1[size]; |
| char buf2[size]; |
| |
| strncpy(buf1, str, size); |
| |
| va_list args; |
| va_start(args, format); |
| vsnprintf(buf2, size, format, args); |
| va_end(args); |
| |
| return snprintf(str, size, "%s%s", buf1, buf2); |
| } |
| |
| void util_getLocalTime(const char *fmt, char *buf, size_t len, time_t tm) |
| { |
| struct tm ltime; |
| localtime_r(&tm, <ime); |
| if (strftime(buf, len, fmt, <ime) < 1) { |
| snprintf(buf, len, "[date fetch error]"); |
| } |
| } |
| |
| void util_nullifyStdio(void) |
| { |
| int fd = open("/dev/null", O_RDWR); |
| |
| if (fd == -1) { |
| LOGMSG_P(l_ERROR, "Couldn't open '/dev/null'"); |
| return; |
| } |
| |
| dup2(fd, 0); |
| dup2(fd, 1); |
| dup2(fd, 2); |
| |
| if (fd > 2) { |
| close(fd); |
| } |
| |
| return; |
| } |
| |
| bool util_redirectStdin(char *inputFile) |
| { |
| int fd = open(inputFile, O_RDONLY); |
| |
| if (fd == -1) { |
| LOGMSG_P(l_ERROR, "Couldn't open '%s'", inputFile); |
| return false; |
| } |
| |
| dup2(fd, 0); |
| if (fd != 0) { |
| close(fd); |
| } |
| |
| return true; |
| } |
| |
| void util_recoverStdio(void) |
| { |
| int fd = open("/dev/tty", O_RDWR); |
| |
| if (fd == -1) { |
| LOGMSG_P(l_ERROR, "Couldn't open '/dev/tty'"); |
| return; |
| } |
| |
| dup2(fd, 0); |
| dup2(fd, 1); |
| dup2(fd, 2); |
| |
| if (tcsetpgrp(fd, getpid()) == -1) { |
| LOGMSG_P(l_WARN, "tcsetpgrp(%d) failed", getpid()); |
| } |
| |
| if (fd > 2) { |
| close(fd); |
| } |
| return; |
| } |
| |
| /* |
| * This is not a cryptographically secure hash |
| */ |
| extern uint64_t util_hash(const char *buf, size_t len) |
| { |
| uint64_t ret = 0; |
| |
| for (size_t i = 0; i < len; i++) { |
| ret += buf[i]; |
| ret += (ret << 10); |
| ret ^= (ret >> 6); |
| } |
| |
| return ret; |
| } |
| |
| extern int64_t util_timeNowMillis(void) |
| { |
| struct timeval tv; |
| if (gettimeofday(&tv, NULL) == -1) { |
| LOGMSG_P(l_FATAL, "gettimeofday()"); |
| } |
| |
| return (((int64_t) tv.tv_sec * 1000LL) + ((int64_t) tv.tv_usec / 1000LL)); |
| } |
| |
| extern uint16_t util_ToFromBE16(uint16_t val) |
| { |
| #if __BYTE_ORDER == __BIG_ENDIAN |
| return val; |
| #elif __BYTE_ORDER == __LITTLE_ENDIAN |
| return SWAP16(val); |
| #else |
| #error "Unknown ENDIANESS" |
| #endif |
| } |
| |
| extern uint16_t util_ToFromLE16(uint16_t val) |
| { |
| #if __BYTE_ORDER == __BIG_ENDIAN |
| return SWAP16(val); |
| #elif __BYTE_ORDER == __LITTLE_ENDIAN |
| return val; |
| #else |
| #error "Unknown ENDIANESS" |
| #endif |
| } |
| |
| extern uint32_t util_ToFromBE32(uint32_t val) |
| { |
| #if __BYTE_ORDER == __BIG_ENDIAN |
| return val; |
| #elif __BYTE_ORDER == __LITTLE_ENDIAN |
| return SWAP32(val); |
| #else |
| #error "Unknown ENDIANESS" |
| #endif |
| } |
| |
| extern uint32_t util_ToFromLE32(uint32_t val) |
| { |
| #if __BYTE_ORDER == __BIG_ENDIAN |
| return SWAP32(val); |
| #elif __BYTE_ORDER == __LITTLE_ENDIAN |
| return val; |
| #else |
| #error "Unknown ENDIANESS" |
| #endif |
| } |
| |
| extern void MX_LOCK(pthread_mutex_t * mutex) |
| { |
| if (pthread_mutex_lock(mutex)) { |
| LOGMSG_P(l_FATAL, "pthread_mutex_lock(%p)", mutex); |
| } |
| } |
| |
| extern void MX_UNLOCK(pthread_mutex_t * mutex) |
| { |
| if (pthread_mutex_unlock(mutex)) { |
| LOGMSG_P(l_FATAL, "pthread_mutex_unlock(%p)", mutex); |
| } |
| } |