summaryrefslogtreecommitdiffstats
path: root/testsuite/usb01/test-file-system.c
diff options
context:
space:
mode:
Diffstat (limited to 'testsuite/usb01/test-file-system.c')
-rw-r--r--testsuite/usb01/test-file-system.c593
1 files changed, 0 insertions, 593 deletions
diff --git a/testsuite/usb01/test-file-system.c b/testsuite/usb01/test-file-system.c
deleted file mode 100644
index f2f74325..00000000
--- a/testsuite/usb01/test-file-system.c
+++ /dev/null
@@ -1,593 +0,0 @@
-/**
- * @file
- *
- * @brief File system test implementation.
- */
-
-/*
- * Copyright (c) 2010 embedded brains GmbH. All rights reserved.
- *
- * embedded brains GmbH
- * Obere Lagerstr. 30
- * 82178 Puchheim
- * Germany
- * <rtems@embedded-brains.de>
- *
- * The license and distribution terms for this file may be
- * found in the file LICENSE in this distribution or at
- * http://www.rtems.com/license/LICENSE.
- */
-
-#include <sys/types.h>
-#include <sys/stat.h>
-#include <assert.h>
-#include <stdio.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include <dirent.h>
-#include <unistd.h>
-#include <errno.h>
-#include <fcntl.h>
-
-#include <rtems.h>
-#include <rtems/libio.h>
-#include <rtems/dosfs.h>
-
-#include "test.h"
-
-#define ASSERT_SC(sc) assert((sc) == RTEMS_SUCCESSFUL)
-
-/**
- * @brief Test states.
- *
- * digraph {
- * INIT -> MOUNT;
- * MOUNT -> INIT_ROOT;
- * MOUNT -> FORMAT;
- * INIT_ROOT -> CHOOSE_DIR_ACTION
- * INIT_ROOT -> ERROR;
- * CHOOSE_DIR_ACTION -> DIR_OPEN;
- * CHOOSE_DIR_ACTION -> DIR_CLOSE;
- * CHOOSE_DIR_ACTION -> DIR_CREATE;
- * CHOOSE_DIR_ACTION -> DIR_DELETE;
- * CHOOSE_DIR_ACTION -> DIR_SLEEP;
- * DIR_OPEN -> CHOOSE_DIR_ACTION;
- * DIR_OPEN -> CHOOSE_FILE_ACTION;
- * DIR_OPEN -> ERROR;
- * DIR_CLOSE -> CHOOSE_DIR_ACTION;
- * DIR_CLOSE -> ERROR;
- * DIR_CREATE -> CHOOSE_DIR_ACTION;
- * DIR_CREATE -> DIR_DELETE;
- * DIR_CREATE -> ERROR;
- * DIR_DELETE -> CHOOSE_DIR_ACTION;
- * DIR_DELETE -> ERROR;
- * DIR_SLEEP -> CHOOSE_DIR_ACTION;
- * CHOOSE_FILE_ACTION -> FILE_CLOSE;
- * CHOOSE_FILE_ACTION -> FILE_APPEND;
- * CHOOSE_FILE_ACTION -> FILE_SLEEP;
- * FILE_CLOSE -> CHOOSE_DIR_ACTION;
- * FILE_CLOSE -> ERROR;
- * FILE_APPEND -> CHOOSE_FILE_ACTION;
- * FILE_APPEND -> ERROR;
- * FILE_SLEEP -> CHOOSE_FILE_ACTION;
- * ERROR -> FORMAT;
- * FORMAT -> MOUNT;
- * FORMAT -> FINISH;
- * }
- */
-typedef enum {
- CHOOSE_DIR_ACTION,
- CHOOSE_FILE_ACTION,
- DIR_CLOSE,
- DIR_CREATE,
- DIR_DELETE,
- DIR_OPEN,
- DIR_SLEEP,
- ERROR,
- FILE_APPEND,
- FILE_CLOSE,
- FILE_SLEEP,
- FINISH,
- FORMAT,
- INIT,
- INIT_ROOT,
- MOUNT
-} test_state;
-
-typedef struct {
- DIR *dir;
- unsigned level;
- unsigned content_count;
- int fd;
- int eno;
-} fs_state;
-
-static test_state do_format(unsigned index, fs_state *fs, const char *disk_path)
-{
- int rv = 0;
-
- printf("[%02u]: format: %s\n", index, disk_path);
-
- rv = msdos_format(disk_path, NULL);
- if (rv == 0) {
- return MOUNT;
- } else {
- fs->eno = errno;
-
- return FINISH;
- }
-}
-
-static unsigned get_bucket(unsigned count)
-{
- long unsigned unit = (1U << 31) / count;
- long unsigned bucket = (long unsigned) lrand48() / unit;
-
- if (bucket != count) {
- return bucket;
- } else {
- return bucket - 1;
- }
-}
-
-static test_state do_choose_dir_action(void)
-{
- switch (get_bucket(8)) {
- case 0:
- case 1:
- return DIR_CLOSE;
- case 2:
- case 3:
- return DIR_CREATE;
- case 4:
- case 5:
- return DIR_OPEN;
- case 6:
- return DIR_DELETE;
- case 7:
- return DIR_SLEEP;
- default:
- assert(false);
- break;
- }
-}
-
-static test_state do_choose_file_action(void)
-{
- switch (get_bucket(3)) {
- case 0:
- return FILE_CLOSE;
- case 1:
- return FILE_SLEEP;
- case 2:
- return FILE_APPEND;
- default:
- assert(false);
- break;
- }
-}
-
-static bool is_normal_entry(const char *entry_name)
-{
- return strcmp(entry_name, ".") != 0 && strcmp(entry_name, "..") != 0;
-}
-
-static bool open_dir(fs_state *fs, const char *dir_path)
-{
- int rv = 0;
- bool change_dir = true;
-
- if (dir_path == NULL) {
- if (fs->level > 1) {
- rv = chdir("..");
- if (rv != 0) {
- fs->eno = errno;
-
- return false;
- }
-
- --fs->level;
- } else {
- return true;
- }
- dir_path = ".";
- change_dir = false;
- }
-
- if (fs->dir != NULL) {
- rv = closedir(fs->dir);
- if (rv != 0) {
- fs->eno = errno;
-
- return false;
- }
- }
-
- fs->content_count = 0;
- fs->dir = opendir(dir_path);
-
- if (fs->dir != NULL) {
- struct dirent *de = NULL;
-
- rewinddir(fs->dir);
- while ((de = readdir(fs->dir)) != NULL) {
- if (is_normal_entry(de->d_name)) {
- ++fs->content_count;
- }
- }
- } else {
- fs->eno = errno;
-
- return false;
- }
-
- if (change_dir) {
- rv = chdir(dir_path);
- if (rv != 0) {
- fs->eno = errno;
-
- return false;
- }
-
- ++fs->level;
- }
-
- return true;
-}
-
-static char *get_dir_entry(fs_state *fs, bool *is_dir)
-{
- int rv = 0;
- char *entry_name = NULL;
-
- if (fs->content_count > 0) {
- struct dirent *de = NULL;
- unsigned bucket = get_bucket(fs->content_count);
- unsigned i = 0;
-
- rewinddir(fs->dir);
- while ((de = readdir(fs->dir)) != NULL) {
- if (is_normal_entry(de->d_name)) {
- if (i != bucket) {
- ++i;
- } else {
- break;
- }
- }
- }
-
- if (de != NULL) {
- struct stat st;
-
- rv = stat(de->d_name, &st);
- if (rv == 0) {
- *is_dir = S_ISDIR(st.st_mode);
-
- entry_name = strdup(de->d_name);
- }
- }
- }
-
- return entry_name;
-}
-
-
-static test_state do_init_root(unsigned index, fs_state *fs, const char *mount_path)
-{
- printf("[%02u]: init root: %s\n", index, mount_path);
-
- if (open_dir(fs, mount_path)) {
- return CHOOSE_DIR_ACTION;
- } else {
- return ERROR;
- }
-}
-
-static test_state do_dir_close(unsigned index, fs_state *fs)
-{
- if (fs->level > 1) {
- printf("[%02u]: close dir\n", index);
-
- if (open_dir(fs, NULL)) {
- return CHOOSE_DIR_ACTION;
- } else {
- return ERROR;
- }
- } else {
- return CHOOSE_DIR_ACTION;
- }
-}
-
-static test_state do_dir_create(unsigned index, fs_state *fs)
-{
- int rv = 0;
- test_state state = ERROR;
- long number = lrand48();
- char name [64];
-
- snprintf(name, sizeof(name), "%li", number);
-
- if ((number % 2) == 0) {
- printf("[%02u]: create file: %s\n", index, name);
-
- rv = open(name, O_RDONLY | O_CREAT, 0777);
-
- if (rv >= 0) {
- rv = close(rv);
-
- if (rv == 0) {
- state = CHOOSE_DIR_ACTION;
- }
- } else if (errno == ENOSPC) {
- state = DIR_DELETE;
- } else {
- fs->eno = errno;
- }
- } else {
- printf("[%02u]: create dir: %s\n", index, name);
-
- rv = mkdir(name, 0777);
-
- if (rv == 0) {
- ++fs->content_count;
-
- state = CHOOSE_DIR_ACTION;
- } else if (errno == EEXIST) {
- state = CHOOSE_DIR_ACTION;
- } else if (errno == ENOSPC) {
- state = DIR_DELETE;
- } else {
- fs->eno = errno;
- }
- }
-
- return state;
-}
-
-static test_state do_dir_delete(unsigned index, fs_state *fs)
-{
- int rv = 0;
- test_state state = ERROR;
-
- if (fs->content_count > 0) {
- bool is_dir = false;
- char *dir_entry_path = get_dir_entry(fs, &is_dir);
-
- if (dir_entry_path != NULL) {
- if (is_dir) {
- printf("[%02u]: remove dir: %s\n", index, dir_entry_path);
-
- rv = rmdir(dir_entry_path);
- } else {
- printf("[%02u]: remove file: %s\n", index, dir_entry_path);
-
- rv = unlink(dir_entry_path);
- }
-
- if (rv == 0) {
- --fs->content_count;
-
- state = CHOOSE_DIR_ACTION;
- } else if (errno == ENOTEMPTY) {
- state = CHOOSE_DIR_ACTION;
- } else {
- fs->eno = errno;
- }
-
- free(dir_entry_path);
- }
- } else {
- state = CHOOSE_DIR_ACTION;
- }
-
- return state;
-}
-
-static test_state do_dir_open(unsigned index, fs_state *fs)
-{
- test_state state = ERROR;
-
- if (fs->content_count > 0) {
- bool is_dir = false;
- char *dir_entry_path = get_dir_entry(fs, &is_dir);
-
- if (dir_entry_path != NULL) {
- if (is_dir) {
- printf("[%02u]: open dir: %s\n", index, dir_entry_path);
-
- if (open_dir(fs, dir_entry_path)) {
- state = CHOOSE_DIR_ACTION;
- }
- } else {
- printf("[%02u]: open file: %s\n", index, dir_entry_path);
-
- fs->fd = open(dir_entry_path, O_WRONLY | O_APPEND);
-
- if (fs->fd >= 0) {
- state = CHOOSE_FILE_ACTION;
- } else {
- fs->eno = errno;
- }
- }
-
- free(dir_entry_path);
- }
- } else {
- state = CHOOSE_DIR_ACTION;
- }
-
- return state;
-}
-
-static test_state do_file_append(unsigned index, fs_state *fs)
-{
- static const char buf [511];
- test_state state = ERROR;
- ssize_t n = 0;
-
- printf("[%02u]: append to file\n", index);
-
- n = write(fs->fd, buf, sizeof(buf));
-
- if (n == (ssize_t) sizeof(buf)) {
- state = CHOOSE_FILE_ACTION;
- } else if (n == -1) {
- fs->eno = errno;
- }
-
- return state;
-}
-
-static test_state do_file_close(unsigned index, fs_state *fs)
-{
- int rv = 0;
- test_state state = ERROR;
-
- printf("[%02u]: close file\n", index);
-
- rv = close(fs->fd);
- fs->fd = -1;
-
- if (rv == 0) {
- state = CHOOSE_DIR_ACTION;
- } else {
- fs->eno = errno;
- }
-
- return state;
-}
-
-static test_state do_sleep(unsigned index, test_state state)
-{
- rtems_status_code sc = RTEMS_SUCCESSFUL;
- rtems_interval ms = ((rtems_interval) get_bucket(10) + 1) * 100;
- rtems_interval interval = ms / rtems_configuration_get_milliseconds_per_tick();
-
- printf("[%02u]: sleep: %" PRIu32 " ms\n", index, ms);
-
- sc = rtems_task_wake_after(interval);
- ASSERT_SC(sc);
-
- return state;
-}
-
-static test_state do_mount(unsigned index, const char *disk_path, const char *mount_path)
-{
- int rv = 0;
-
- printf("[%02u]: mount: %s -> %s\n", index, disk_path, mount_path);
-
- rv = mount_and_make_target_path(
- disk_path,
- mount_path,
- RTEMS_FILESYSTEM_TYPE_DOSFS,
- RTEMS_FILESYSTEM_READ_WRITE,
- NULL
- );
- if (rv == 0) {
- return INIT_ROOT;
- } else {
- return FORMAT;
- }
-}
-
-static test_state do_error(unsigned index, fs_state *fs, const char *mount_path)
-{
- int rv = 0;
-
- if (fs->eno > 0) {
- printf("[%02u]: error: %s\n", index, strerror(fs->eno));
- } else {
- printf("[%02u]: error\n", index);
- }
- fs->eno = 0;
-
- if (fs->fd >= 0) {
- close(fs->fd);
- fs->fd = -1;
- }
-
- if (fs->dir != NULL) {
- closedir(fs->dir);
- fs->dir = NULL;
- }
-
- chdir("/");
- fs->level = 0;
-
- rv = unmount(mount_path);
- if (rv == 0) {
- return FORMAT;
- } else {
- return FINISH;
- }
-}
-
-void test_file_system(unsigned index, const char *disk_path, const char *mount_path)
-{
- test_state state = INIT;
- fs_state fs = {
- .dir = NULL,
- .level = 0,
- .content_count = 0,
- .fd = -1,
- .eno = 0
- };
-
- printf("[%02u]: start\n", index);
-
- while (state != FINISH) {
- switch (state) {
- case CHOOSE_DIR_ACTION:
- state = do_choose_dir_action();
- break;
- case CHOOSE_FILE_ACTION:
- state = do_choose_file_action();
- break;
- case DIR_CLOSE:
- state = do_dir_close(index, &fs);
- break;
- case DIR_CREATE:
- state = do_dir_create(index, &fs);
- break;
- case DIR_DELETE:
- state = do_dir_delete(index, &fs);
- break;
- case DIR_OPEN:
- state = do_dir_open(index, &fs);
- break;
- case DIR_SLEEP:
- state = do_sleep(index, CHOOSE_DIR_ACTION);
- break;
- case ERROR:
- state = do_error(index, &fs, mount_path);
- break;
- case FILE_APPEND:
- state = do_file_append(index, &fs);
- break;
- case FILE_CLOSE:
- state = do_file_close(index, &fs);
- break;
- case FILE_SLEEP:
- state = do_sleep(index, CHOOSE_FILE_ACTION);
- break;
- case FORMAT:
- state = do_format(index, &fs, disk_path);
- break;
- case INIT:
- state = MOUNT;
- break;
- case INIT_ROOT:
- state = do_init_root(index, &fs, mount_path);
- break;
- case MOUNT:
- state = do_mount(index, disk_path, mount_path);
- break;
- default:
- assert(false);
- break;
- }
- }
-
- printf("[%02u]: finish\n", index);
-}