202 lines
6.2 KiB
C
202 lines
6.2 KiB
C
|
/*
|
||
|
* Copyright (C) 2007 Nokia Corporation.
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or
|
||
|
* modify it under the terms of the GNU General Public License
|
||
|
* version 2 as published by the Free Software Foundation.
|
||
|
*
|
||
|
* 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 St, Fifth Floor, Boston, MA
|
||
|
* 02110-1301 USA
|
||
|
*
|
||
|
* Author: Adrian Hunter
|
||
|
*/
|
||
|
|
||
|
#include <unistd.h>
|
||
|
#include <stdio.h>
|
||
|
#include <string.h>
|
||
|
#include <stdint.h>
|
||
|
#include <sys/types.h>
|
||
|
#include <sys/stat.h>
|
||
|
#include <fcntl.h>
|
||
|
|
||
|
#include "tests.h"
|
||
|
|
||
|
void test_2(void)
|
||
|
{
|
||
|
pid_t pid;
|
||
|
int create, full;
|
||
|
unsigned i, number_of_files;
|
||
|
unsigned growth;
|
||
|
unsigned size;
|
||
|
uint64_t big_file_size;
|
||
|
int fd;
|
||
|
off_t offset;
|
||
|
char dir_name[256];
|
||
|
|
||
|
/* Create a directory to test in */
|
||
|
pid = getpid();
|
||
|
tests_cat_pid(dir_name, "test_2_test_dir_", pid);
|
||
|
if (chdir(dir_name) == -1)
|
||
|
CHECK(mkdir(dir_name, 0777) != -1);
|
||
|
CHECK(chdir(dir_name) != -1);
|
||
|
/* Create up to 1000 files appending 400 bytes at a time to each file */
|
||
|
/* until the file system is full.*/
|
||
|
create = 1;
|
||
|
full = 0;
|
||
|
number_of_files = 1000;
|
||
|
while (!full) {
|
||
|
for (i = 0; i < number_of_files; ++i) {
|
||
|
growth = tests_append_to_fragment_file(i, 400, create);
|
||
|
if (!growth) {
|
||
|
full = 1;
|
||
|
if (create)
|
||
|
number_of_files = i;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
create = 0;
|
||
|
}
|
||
|
/* Check the files */
|
||
|
CHECK(tests_count_files_in_dir(".") == number_of_files);
|
||
|
for (i = 0; i < number_of_files; ++i)
|
||
|
tests_check_fragment_file(i);
|
||
|
/* Delete half of them */
|
||
|
for (i = 1; i < number_of_files; i += 2)
|
||
|
tests_delete_fragment_file(i);
|
||
|
/* Check them again */
|
||
|
CHECK(tests_count_files_in_dir(".") == (number_of_files + 1) / 2);
|
||
|
for (i = 0; i < number_of_files; i += 2)
|
||
|
tests_check_fragment_file(i);
|
||
|
CHECK(tests_count_files_in_dir(".") == (number_of_files + 1) / 2);
|
||
|
/* Create a big file that fills two thirds of the free space */
|
||
|
big_file_size = tests_get_big_file_size(2,3);
|
||
|
/* Check the big file */
|
||
|
tests_create_file("big_file", big_file_size);
|
||
|
CHECK(tests_count_files_in_dir(".") == 1 + (number_of_files + 1) / 2);
|
||
|
tests_check_filled_file("big_file");
|
||
|
/* Open the big file */
|
||
|
fd = open("big_file",O_RDWR | tests_maybe_sync_flag());
|
||
|
CHECK(fd != -1);
|
||
|
/* Delete the big file while it is still open */
|
||
|
tests_delete_file("big_file");
|
||
|
/* Check the big file again */
|
||
|
CHECK(tests_count_files_in_dir(".") == (number_of_files + 1) / 2);
|
||
|
tests_check_filled_file_fd(fd);
|
||
|
|
||
|
/* Write parts of the files and check them */
|
||
|
|
||
|
offset = 100; /* Offset to write at, in the small files */
|
||
|
size = 200; /* Number of bytes to write at the offset */
|
||
|
|
||
|
for (i = 0; i < number_of_files; i += 2)
|
||
|
tests_overwite_fragment_file(i, offset, size);
|
||
|
/* Rewrite the big file entirely */
|
||
|
tests_write_filled_file(fd, 0, big_file_size);
|
||
|
for (i = 0; i < number_of_files; i += 2)
|
||
|
tests_check_fragment_file(i);
|
||
|
tests_check_filled_file_fd(fd);
|
||
|
|
||
|
offset = 300; /* Offset to write at, in the small files */
|
||
|
size = 400; /* Number of bytes to write at the offset */
|
||
|
|
||
|
for (i = 0; i < number_of_files; i += 2)
|
||
|
tests_overwite_fragment_file(i, offset, size);
|
||
|
/* Rewrite the big file entirely */
|
||
|
tests_write_filled_file(fd, 0, big_file_size);
|
||
|
for (i = 0; i < number_of_files; i += 2)
|
||
|
tests_check_fragment_file(i);
|
||
|
tests_check_filled_file_fd(fd);
|
||
|
|
||
|
offset = 110; /* Offset to write at, in the small files */
|
||
|
size = 10; /* Number of bytes to write at the offset */
|
||
|
|
||
|
for (i = 0; i < number_of_files; i += 2)
|
||
|
tests_overwite_fragment_file(i, offset, size);
|
||
|
/* Rewrite the big file entirely */
|
||
|
tests_write_filled_file(fd, 0, big_file_size);
|
||
|
for (i = 0; i < number_of_files; i += 2)
|
||
|
tests_check_fragment_file(i);
|
||
|
tests_check_filled_file_fd(fd);
|
||
|
|
||
|
offset = 10; /* Offset to write at, in the small files */
|
||
|
size = 1000; /* Number of bytes to write at the offset */
|
||
|
|
||
|
for (i = 0; i < number_of_files; i += 2)
|
||
|
tests_overwite_fragment_file(i, offset, size);
|
||
|
/* Rewrite the big file entirely */
|
||
|
tests_write_filled_file(fd, 0, big_file_size);
|
||
|
for (i = 0; i < number_of_files; i += 2)
|
||
|
tests_check_fragment_file(i);
|
||
|
tests_check_filled_file_fd(fd);
|
||
|
|
||
|
offset = 0; /* Offset to write at, in the small files */
|
||
|
size = 100000; /* Number of bytes to write at the offset */
|
||
|
|
||
|
for (i = 0; i < number_of_files; i += 2)
|
||
|
tests_overwite_fragment_file(i, offset, size);
|
||
|
/* Rewrite the big file entirely */
|
||
|
tests_write_filled_file(fd, 0, big_file_size);
|
||
|
for (i = 0; i < number_of_files; i += 2)
|
||
|
tests_check_fragment_file(i);
|
||
|
tests_check_filled_file_fd(fd);
|
||
|
|
||
|
/* Close the big file*/
|
||
|
CHECK(close(fd) != -1);
|
||
|
/* Check the small files */
|
||
|
CHECK(tests_count_files_in_dir(".") == (number_of_files + 1) / 2);
|
||
|
for (i = 0; i < number_of_files; i += 2)
|
||
|
tests_check_fragment_file(i);
|
||
|
/* Delete the small files */
|
||
|
for (i = 0; i < number_of_files; i += 2)
|
||
|
tests_delete_fragment_file(i);
|
||
|
CHECK(tests_count_files_in_dir(".") == 0);
|
||
|
CHECK(chdir("..") != -1);
|
||
|
CHECK(rmdir(dir_name) != -1);
|
||
|
}
|
||
|
|
||
|
/* Title of this test */
|
||
|
|
||
|
const char *test_2_get_title(void)
|
||
|
{
|
||
|
return "Repeated write many small files and one big deleted file";
|
||
|
}
|
||
|
|
||
|
/* Description of this test */
|
||
|
|
||
|
const char *test_2_get_description(void)
|
||
|
{
|
||
|
return
|
||
|
"Create a directory named test_2_test_dir_pid, where " \
|
||
|
"pid is the process id. Within that directory, " \
|
||
|
"create about 1000 files. Append 400 bytes to each until " \
|
||
|
"the file system is full. Then delete half of them. Then " \
|
||
|
"create a big file that uses about 2/3 of the remaining free " \
|
||
|
"space. Get a file descriptor for the big file, and delete " \
|
||
|
"the big file. Then repeatedly write to the small files " \
|
||
|
"and the big file. " \
|
||
|
"Finally delete the big file and directory.";
|
||
|
}
|
||
|
|
||
|
int main(int argc, char *argv[])
|
||
|
{
|
||
|
int run_test;
|
||
|
|
||
|
/* Handle common arguments */
|
||
|
run_test = tests_get_args(argc, argv, test_2_get_title(),
|
||
|
test_2_get_description(), "s");
|
||
|
if (!run_test)
|
||
|
return 1;
|
||
|
/* Change directory to the file system and check it is ok for testing */
|
||
|
tests_check_test_file_system();
|
||
|
/* Do the actual test */
|
||
|
test_2();
|
||
|
return 0;
|
||
|
}
|