177 lines
3.7 KiB
C
177 lines
3.7 KiB
C
/*
|
|
* Copyright (C) 2003,2004 Red Hat, Inc. All rights reserved.
|
|
*
|
|
* The contents of this file may be used under the terms of the GNU
|
|
* General Public License version 2 (the "GPL")
|
|
*
|
|
* Author: Stephen C. Tweedie <sct@redhat.com>
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
|
|
* the GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
* Module: .c
|
|
*/
|
|
|
|
/*
|
|
* Change History:
|
|
*
|
|
* 2/2004 Marty Ridgeway (mridge@us.ibm.com) Changes to adapt to LTP
|
|
*
|
|
*/
|
|
|
|
#define _XOPEN_SOURCE 600
|
|
#define _GNU_SOURCE
|
|
#define MAX_ITERATIONS 250
|
|
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <sys/mman.h>
|
|
#include <sys/wait.h>
|
|
|
|
#define BIGSIZE 128*1024*1024
|
|
#define READSIZE 32*1024*1024
|
|
#define WRITESIZE 32*1024*1024
|
|
|
|
int pagesize;
|
|
char *iobuf;
|
|
int pass = 0;
|
|
|
|
void assert(const char *what, int assertion)
|
|
{
|
|
if (assertion)
|
|
return;
|
|
perror(what);
|
|
exit(1);
|
|
}
|
|
|
|
void do_buffered_writes(int fd, int pattern)
|
|
{
|
|
int rc;
|
|
int offset;
|
|
|
|
memset(iobuf, pattern, WRITESIZE);
|
|
for (offset = 0; offset + WRITESIZE <= BIGSIZE; offset += WRITESIZE) {
|
|
rc = pwrite(fd, iobuf, WRITESIZE, offset);
|
|
assert("pwrite", rc >= 0);
|
|
if (rc != WRITESIZE) {
|
|
fprintf(stderr, "Pass %d: short write (%d out of %d)\n",
|
|
pass, rc, WRITESIZE);
|
|
exit(1);
|
|
}
|
|
fsync(fd);
|
|
}
|
|
}
|
|
|
|
int do_direct_reads(char *filename)
|
|
{
|
|
int fd;
|
|
int offset;
|
|
int rc, i;
|
|
int *p;
|
|
|
|
fd = open(filename, O_DIRECT | O_RDONLY, 0);
|
|
assert("open", fd >= 0);
|
|
|
|
for (offset = 0; offset + READSIZE <= BIGSIZE; offset += READSIZE) {
|
|
rc = pread(fd, iobuf, READSIZE, offset);
|
|
assert("pread", rc >= 0);
|
|
if (rc != READSIZE) {
|
|
fprintf(stderr, "Pass: %d short read (%d out of %d)\n",
|
|
pass, rc, READSIZE);
|
|
exit(1);
|
|
}
|
|
for (i = 0, p = (int *)iobuf; i < READSIZE; i += 4) {
|
|
if (*p) {
|
|
fprintf(stderr,
|
|
"Pass: %d Found data (%08x) at offset %d+%d\n",
|
|
pass, *p, offset, i);
|
|
close(fd);
|
|
return 1;
|
|
}
|
|
p++;
|
|
}
|
|
}
|
|
close(fd);
|
|
return 0;
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
char *filename;
|
|
int fd;
|
|
int pid;
|
|
int err;
|
|
int bufsize;
|
|
|
|
if (argc != 2) {
|
|
fprintf(stderr, "Needs a filename as an argument.\n");
|
|
exit(1);
|
|
}
|
|
|
|
filename = argv[1];
|
|
|
|
pagesize = getpagesize();
|
|
bufsize = READSIZE;
|
|
if (WRITESIZE > READSIZE)
|
|
bufsize = WRITESIZE;
|
|
err = posix_memalign((void **)&iobuf, pagesize, bufsize);
|
|
if (err) {
|
|
fprintf(stderr, "Error allocating %d aligned bytes.\n",
|
|
bufsize);
|
|
exit(1);
|
|
}
|
|
|
|
fd = open(filename, O_CREAT | O_TRUNC | O_RDWR, 0666);
|
|
assert("open", fd >= 0);
|
|
|
|
do {
|
|
|
|
assert("ftruncate", ftruncate(fd, BIGSIZE) == 0);
|
|
fsync(fd);
|
|
|
|
pid = fork();
|
|
assert("fork", pid >= 0);
|
|
|
|
if (!pid) {
|
|
do_buffered_writes(fd, 0);
|
|
exit(0);
|
|
}
|
|
|
|
err = do_direct_reads(filename);
|
|
|
|
wait4(pid, NULL, WNOHANG, 0);
|
|
|
|
if (err)
|
|
break;
|
|
|
|
/* Fill the file with a known pattern so that the blocks
|
|
* on disk can be detected if they become exposed. */
|
|
do_buffered_writes(fd, 1);
|
|
fsync(fd);
|
|
|
|
assert("ftruncate", ftruncate(fd, 0) == 0);
|
|
fsync(fd);
|
|
} while (pass++ < MAX_ITERATIONS);
|
|
|
|
if (!err) {
|
|
fprintf(stdout, "ltp-diorh: Completed %d iterations OK \n",
|
|
pass);
|
|
}
|
|
|
|
return err;
|
|
}
|