18a272653SPeter Holm#!/bin/sh 28a272653SPeter Holm 38a272653SPeter Holm# 48a272653SPeter Holm# Copyright (c) 2011 Peter Holm <pho@FreeBSD.org> 58a272653SPeter Holm# All rights reserved. 68a272653SPeter Holm# 78a272653SPeter Holm# Redistribution and use in source and binary forms, with or without 88a272653SPeter Holm# modification, are permitted provided that the following conditions 98a272653SPeter Holm# are met: 108a272653SPeter Holm# 1. Redistributions of source code must retain the above copyright 118a272653SPeter Holm# notice, this list of conditions and the following disclaimer. 128a272653SPeter Holm# 2. Redistributions in binary form must reproduce the above copyright 138a272653SPeter Holm# notice, this list of conditions and the following disclaimer in the 148a272653SPeter Holm# documentation and/or other materials provided with the distribution. 158a272653SPeter Holm# 168a272653SPeter Holm# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 178a272653SPeter Holm# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 188a272653SPeter Holm# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 198a272653SPeter Holm# ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 208a272653SPeter Holm# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 218a272653SPeter Holm# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 228a272653SPeter Holm# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 238a272653SPeter Holm# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 248a272653SPeter Holm# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 258a272653SPeter Holm# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 268a272653SPeter Holm# SUCH DAMAGE. 278a272653SPeter Holm# 288a272653SPeter Holm 298a272653SPeter Holm# readdir(3) fuzzing inspired by the iknowthis test suite 308a272653SPeter Holm# by Tavis Ormandy <taviso cmpxchg8b com> 318a272653SPeter Holm 328a272653SPeter Holm# "panic: kmem_malloc(1328054272): kmem_map too small" seen 338a272653SPeter Holm 348a272653SPeter Holm[ `id -u ` -ne 0 ] && echo "Must be root!" && exit 1 358a272653SPeter Holm 368a272653SPeter Holm. ../default.cfg 378a272653SPeter Holm 388a272653SPeter Holmhere=`pwd` 398a272653SPeter Holmcd /tmp 408a272653SPeter Holmsed '1,/^EOF/d' < $here/$0 > readdir.c 418a272653SPeter Holmmycc -o readdir -Wall -Wextra readdir.c || exit 1 428a272653SPeter Holmrm -f readdir.c 438a272653SPeter Holm 448a272653SPeter Holmmount | grep $mntpoint | grep -q /dev/md && umount -f $mntpoint 458a272653SPeter Holmmdconfig -l | grep -q md$mdstart && mdconfig -d -u $mdstart 468a272653SPeter Holm 478a272653SPeter Holmmount -t tmpfs tmpfs $mntpoint 48*088cc7d2SAlexander Ziaeeecho "Testing tmpfs(4)" 498a272653SPeter Holmcp -a /usr/include $mntpoint 508a272653SPeter Holm/tmp/readdir $mntpoint 518a272653SPeter Holmumount $mntpoint 528a272653SPeter Holm 53*088cc7d2SAlexander Ziaeeecho "Testing fdescfs(4)" 548a272653SPeter Holmkldstat -v | grep -q fdescfs || { kldload fdescfs.ko; loaded=1; } 558a272653SPeter Holmmount -t fdescfs null /dev/fd 568a272653SPeter Holm/tmp/readdir /dev/fd 578a272653SPeter Holmumount /dev/fd 588a272653SPeter Holm[ $unload ] && kldunload fdescfs.ko 598a272653SPeter Holm 60*088cc7d2SAlexander Ziaeeecho "Testing procfs(4)" 618a272653SPeter Holmmount -t procfs procfs $mntpoint 628a272653SPeter Holm/tmp/readdir $mntpoint 638a272653SPeter Holmumount $mntpoint 648a272653SPeter Holm 658a272653SPeter Holmif ping -c 2 `echo $nfs_export | sed 's/:.*//'` > /dev/null 2>&1; then 668a272653SPeter Holm echo "Testing nfs" 678a272653SPeter Holm mount -t nfs -o nfsv3,tcp,nolockd,retrycnt=3,soft,timeout=1 \ 688a272653SPeter Holm $nfs_export $mntpoint 698a272653SPeter Holm /tmp/readdir $mntpoint 708a272653SPeter Holm umount $mntpoint 718a272653SPeter Holmfi 728a272653SPeter Holm 738a272653SPeter Holmmdconfig -a -t swap -s 1g -u $mdstart || exit 1 74608c97bfSPeter Holmnewfs md$mdstart > /dev/null 75608c97bfSPeter Holmmount /dev/md$mdstart $mntpoint 768a272653SPeter Holmcp -a /usr/include $mntpoint 778a272653SPeter Holmecho "Testing UFS" 788a272653SPeter Holm/tmp/readdir $mntpoint 798a272653SPeter Holmumount $mntpoint 808a272653SPeter Holmmdconfig -d -u $mdstart 818a272653SPeter Holm 828a272653SPeter Holmmdconfig -a -t swap -s 1g -u $mdstart || exit 1 83608c97bfSPeter Holmnewfs $newfs_flags md$mdstart > /dev/null 84608c97bfSPeter Holmmount /dev/md$mdstart $mntpoint 858a272653SPeter Holmcp -a /usr/include $mntpoint 868a272653SPeter Holmecho "Testing FFS" 878a272653SPeter Holm/tmp/readdir $mntpoint 888a272653SPeter Holmumount $mntpoint 898a272653SPeter Holmmdconfig -d -u $mdstart 908a272653SPeter Holm 918a272653SPeter Holmmount -t nullfs /bin $mntpoint 92*088cc7d2SAlexander Ziaeeecho "Testing nullfs(4)" 938a272653SPeter Holm/tmp/readdir $mntpoint 948a272653SPeter Holmumount $mntpoint 958a272653SPeter Holm 968a272653SPeter Holmrm -f /tmp/readdir 978a272653SPeter Holmexit 0 988a272653SPeter HolmEOF 998a272653SPeter Holm#include <sys/types.h> 1008a272653SPeter Holm#include <sys/uio.h> 1018a272653SPeter Holm#include <sys/wait.h> 1028a272653SPeter Holm 1038a272653SPeter Holm#include <dirent.h> 1048a272653SPeter Holm#include <err.h> 1058a272653SPeter Holm#include <fcntl.h> 1068a272653SPeter Holm#include <signal.h> 1078a272653SPeter Holm#include <stdio.h> 1088a272653SPeter Holm#include <stdlib.h> 1098a272653SPeter Holm#include <strings.h> 1108a272653SPeter Holm#include <time.h> 1118a272653SPeter Holm#include <unistd.h> 1128a272653SPeter Holm 1138a272653SPeter Holm#define RUNTIME 120 1148a272653SPeter Holm 1158a272653SPeter Holm/* copy from /usr/src/lib/libc/gen/gen-private.h */ 1168a272653SPeter Holmstruct _telldir; /* see telldir.h */ 1178a272653SPeter Holmstruct pthread_mutex; 1188a272653SPeter Holm 1198a272653SPeter Holm/* 1208a272653SPeter Holm * Structure describing an open directory. 1218a272653SPeter Holm * 1228a272653SPeter Holm * NOTE. Change structure layout with care, at least dd_fd field has to 1238a272653SPeter Holm * remain unchanged to guarantee backward compatibility. 1248a272653SPeter Holm */ 1258a272653SPeter Holmstruct _dirdesc { 1268a272653SPeter Holm int dd_fd; /* file descriptor associated with directory */ 1278a272653SPeter Holm long dd_loc; /* offset in current buffer */ 1288a272653SPeter Holm long dd_size; /* amount of data returned by getdirentries */ 1298a272653SPeter Holm char *dd_buf; /* data buffer */ 1308a272653SPeter Holm int dd_len; /* size of data buffer */ 1318a272653SPeter Holm long dd_seek; /* magic cookie returned by getdirentries */ 1328a272653SPeter Holm long dd_rewind; /* magic cookie for rewinding */ 1338a272653SPeter Holm int dd_flags; /* flags for readdir */ 1348a272653SPeter Holm struct pthread_mutex *dd_lock; /* lock */ 1358a272653SPeter Holm struct _telldir *dd_td; /* telldir position recording */ 1368a272653SPeter Holm}; 1378a272653SPeter Holm/* End copy */ 1388a272653SPeter Holm 1398a272653SPeter Holmstatic void 1408a272653SPeter Holmhand(int i __unused) { /* handler */ 1418a272653SPeter Holm _exit(1); 1428a272653SPeter Holm} 1438a272653SPeter Holm 1448a272653SPeter Holmstatic void 1458a272653SPeter Holmtest(char *path) 1468a272653SPeter Holm{ 1478a272653SPeter Holm DIR *dirp, fuzz; 1488a272653SPeter Holm int i; 1498a272653SPeter Holm 1508a272653SPeter Holm signal(SIGSEGV, hand); 1518a272653SPeter Holm alarm(300); 1528a272653SPeter Holm for (i = 0; i < 2000; i++) { 1538a272653SPeter Holm if ((dirp = opendir(path)) == NULL) 1548a272653SPeter Holm break; 1558a272653SPeter Holm bcopy(dirp, &fuzz, sizeof(fuzz)); 1568a272653SPeter Holm fuzz.dd_len = arc4random(); 1578a272653SPeter Holm readdir(&fuzz); 1588a272653SPeter Holm closedir(dirp); 1598a272653SPeter Holm } 1608a272653SPeter Holm 1618a272653SPeter Holm _exit(0); 1628a272653SPeter Holm} 1638a272653SPeter Holm 1648a272653SPeter Holmint 1658a272653SPeter Holmmain(int argc __unused, char **argv) 1668a272653SPeter Holm{ 1678a272653SPeter Holm time_t start; 1688a272653SPeter Holm 1698a272653SPeter Holm start = time(NULL); 1708a272653SPeter Holm while (time(NULL) - start < RUNTIME) { 1718a272653SPeter Holm if (fork() == 0) 1728a272653SPeter Holm test(argv[1]); 1738a272653SPeter Holm wait(NULL); 1748a272653SPeter Holm } 1758a272653SPeter Holm 1768a272653SPeter Holm return (0); 1778a272653SPeter Holm} 178