shithub: lwext4

ref: 5b99cfc34279cd95c82510c4404b6625170dbdfb
dir: /fs_test/lwext4_server.c/

View raw version
/*
 * Copyright (c) 2014 Grzegorz Kostka (kostka.grzegorz@gmail.com)
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * - Redistributions of source code must retain the above copyright
 *   notice, this list of conditions and the following disclaimer.
 * - Redistributions in binary form must reproduce the above copyright
 *   notice, this list of conditions and the following disclaimer in the
 *   documentation and/or other materials provided with the distribution.
 * - The name of the author may not be used to endorse or promote products
 *   derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <stdint.h>
#include <stdbool.h>
#include <getopt.h>
#include <time.h>
#include <inttypes.h>
#include <sys/time.h>

#ifdef WIN32
#include <winsock2.h>
#include <ws2tcpip.h>
#include <windows.h>
#else
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/types.h>
#endif

#include <ext4.h>
#include "../blockdev/linux/file_dev.h"
#include "../blockdev/windows/file_windows.h"


static int winsock_init(void);
static void winsock_fini(void);
static char *entry_to_str(uint8_t type);

#define MAX_FILES 64
#define MAX_DIRS 64

#define MAX_RW_BUFFER (1024 * 1024)
#define RW_BUFFER_PATERN ('x')

/**@brief   Default connection port*/
static int connection_port = 1234;

/**@brief   Default filesystem filename.*/
static char *ext4_fname = "ext2";

/**@brief   Verbose mode*/
static bool verbose = false;

/**@brief   Winpart mode*/
static bool winpart = false;

/**@brief   Blockdev handle*/
static struct ext4_blockdev *bd;

static bool cache_wb = false;

static char read_buffer[MAX_RW_BUFFER];
static char write_buffer[MAX_RW_BUFFER];

static const char *usage = "                                    \n\
Welcome in lwext4_server.                                       \n\
Copyright (c) 2013 Grzegorz Kostka (kostka.grzegorz@gmail.com)  \n\
Usage:                                                          \n\
    --image     (-i) - ext2/3/4 image file                      \n\
    --port      (-p) - server port                              \n\
    --verbose   (-v) - verbose mode                             \n\
    --winpart   (-w) - windows_partition mode                   \n\
    --cache_wb  (-c) - cache writeback_mode                     \n\
\n";

/**@brief   Open file instance descriptor.*/
struct lwext4_files {
	char name[255];
	ext4_file fd;
};

/**@brief   Open directory instance descriptor.*/
struct lwext4_dirs {
	char name[255];
	ext4_dir fd;
};

/**@brief   Library call opcode.*/
struct lwext4_op_codes {
	char *func;
};

/**@brief   Library call wraper.*/
struct lwext4_call {
	int (*lwext4_call)(const char *p);
};

/**@brief  */
static struct lwext4_files file_tab[MAX_FILES];

/**@brief  */
static struct lwext4_dirs dir_tab[MAX_DIRS];

/**@brief  */
static struct lwext4_op_codes op_codes[] = {
    "device_register",
    "mount",
    "umount",
    "mount_point_stats",
    "cache_write_back",
    "fremove",
    "fopen",
    "fclose",
    "fread",
    "fwrite",
    "fseek",
    "ftell",
    "fsize",
    "dir_rm",
    "dir_mk",
    "dir_open",
    "dir_close",
    "dir_entry_get",

    "multi_fcreate",
    "multi_fwrite",
    "multi_fread",
    "multi_fremove",
    "multi_dcreate",
    "multi_dremove",
    "stats_save",
    "stats_check",
};

static int device_register(const char *p);
static int mount(const char *p);
static int umount(const char *p);
static int mount_point_stats(const char *p);
static int cache_write_back(const char *p);
static int fremove(const char *p);
static int file_open(const char *p);
static int file_close(const char *p);
static int file_read(const char *p);
static int file_write(const char *p);
static int file_seek(const char *p);
static int file_tell(const char *p);
static int file_size(const char *p);
static int dir_rm(const char *p);
static int dir_mk(const char *p);
static int dir_open(const char *p);
static int dir_close(const char *p);
static int dir_close(const char *p);
static int dir_entry_get(const char *p);

static int multi_fcreate(const char *p);
static int multi_fwrite(const char *p);
static int multi_fread(const char *p);
static int multi_fremove(const char *p);
static int multi_dcreate(const char *p);
static int multi_dremove(const char *p);
static int stats_save(const char *p);
static int stats_check(const char *p);

/**@brief  */
static struct lwext4_call op_call[] = {
    device_register,   /*PARAMS(3):   0 cache_mode dev_name   */
    mount,		/*PARAMS(2):   dev_name mount_point    */
    umount,		/*PARAMS(1):   mount_point             */
    mount_point_stats, /*PARAMS(2):   mount_point, 0          */
    cache_write_back,  /*PARAMS(2):   mount_point, en         */
    fremove,		/*PARAMS(1):   path                    */
    file_open,		/*PARAMS(2):   fid path flags          */
    file_close,		/*PARAMS(1):   fid                     */
    file_read,		/*PARAMS(4):   fid 0 len 0             */
    file_write,		/*PARAMS(4):   fid 0 len 0             */
    file_seek,		/*PARAMS(2):   fid off origin          */
    file_tell,		/*PARAMS(2):   fid exp                 */
    file_size,		/*PARAMS(2):   fid exp                 */
    dir_rm,		/*PARAMS(1):   path                    */
    dir_mk,		/*PARAMS(1):   path                    */
    dir_open,		/*PARAMS(2):   did, path               */
    dir_close,		/*PARAMS(1):   did                     */
    dir_entry_get,     /*PARAMS(2):   did, exp                */

    multi_fcreate, /*PARAMS(3):   path prefix cnt         */
    multi_fwrite,  /*PARAMS(4):   path prefix cnt size    */
    multi_fread,   /*PARAMS(4):   path prefix cnt size    */
    multi_fremove, /*PARAMS(2):   path prefix cnt         */
    multi_dcreate, /*PARAMS(3):   path prefix cnt         */
    multi_dremove, /*PARAMS(2):   path prefix             */
    stats_save,    /*PARAMS(1):   path                    */
    stats_check,   /*PARAMS(1):   path                    */
};

static clock_t get_ms(void)
{
	struct timeval t;
	gettimeofday(&t, NULL);
	return (t.tv_sec * 1000) + (t.tv_usec / 1000);
}

/**@brief  */
static int exec_op_code(const char *opcode)
{
	int i;
	int r = -1;

	for (i = 0; i < sizeof(op_codes) / sizeof(op_codes[0]); ++i) {

		if (strncmp(op_codes[i].func, opcode, strlen(op_codes[i].func)))
			continue;

		if (opcode[strlen(op_codes[i].func)] != ' ')
			continue;

		printf("%s\n", opcode);
		opcode += strlen(op_codes[i].func);
		/*Call*/

		clock_t t = get_ms();
		r = op_call[i].lwext4_call(opcode);

		printf("rc: %d, time: %ums\n", r, (unsigned int)(get_ms() - t));

		break;
	}

	return r;
}

static int server_open(void)
{
	int fd = 0;
	struct sockaddr_in serv_addr;

	memset(&serv_addr, 0, sizeof(serv_addr));

	if (winsock_init() < 0) {
		printf("winsock_init() error\n");
		exit(-1);
	}

	fd = socket(AF_INET, SOCK_STREAM, 0);
	if (fd < 0) {
		printf("socket() error: %s\n", strerror(errno));
		exit(-1);
	}

	int yes = 1;
	if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, (void *)&yes,
		       sizeof(int))) {
		printf("setsockopt() error: %s\n", strerror(errno));
		exit(-1);
	}

	serv_addr.sin_family = AF_INET;
	serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);
	serv_addr.sin_port = htons(connection_port);

	if (bind(fd, (struct sockaddr *)&serv_addr, sizeof(serv_addr))) {
		printf("bind() error: %s\n", strerror(errno));
		exit(-1);
	}

	if (listen(fd, 1)) {
		printf("listen() error: %s\n", strerror(errno));
		exit(-1);
	}

	return fd;
}

static bool parse_opt(int argc, char **argv)
{
	int option_index = 0;
	int c;

	static struct option long_options[] = {
	    {"image", required_argument, 0, 'i'},
	    {"port", required_argument, 0, 'p'},
	    {"verbose", no_argument, 0, 'v'},
	    {"winpart", no_argument, 0, 'w'},
	    {"cache_wb", no_argument, 0, 'c'},
	    {"version", no_argument, 0, 'x'},
	    {0, 0, 0, 0}};

	while (-1 != (c = getopt_long(argc, argv, "i:p:vcwx", long_options,
				      &option_index))) {

		switch (c) {
		case 'i':
			ext4_fname = optarg;
			break;
		case 'p':
			connection_port = atoi(optarg);
			break;
		case 'v':
			verbose = true;
			break;
		case 'c':
			cache_wb = true;
			break;
		case 'w':
			winpart = true;
			break;
		case 'x':
			puts(VERSION);
			exit(0);
			break;
		default:
			printf("%s", usage);
			return false;
		}
	}
	return true;
}

int main(int argc, char *argv[])
{
	int n;
	int listenfd;
	int connfd;
	char op_code[128];

	if (!parse_opt(argc, argv))
		return -1;

	listenfd = server_open();

	printf("lwext4_server: listening on port: %d\n", connection_port);

	memset(write_buffer, RW_BUFFER_PATERN, MAX_RW_BUFFER);
	while (1) {
		connfd = accept(listenfd, (struct sockaddr *)NULL, NULL);

		n = recv(connfd, op_code, sizeof(op_code), 0);

		if (n < 0) {
			printf("recv() error: %s fd = %d\n", strerror(errno),
			       connfd);
			break;
		}

		op_code[n] = 0;

		int r = exec_op_code(op_code);

		n = send(connfd, (void *)&r, sizeof(r), 0);
		if (n < 0) {
			printf("send() error: %s fd = %d\n", strerror(errno),
			       connfd);
			break;
		}

		close(connfd);
	}

	winsock_fini();
	return 0;
}

static int device_register(const char *p)
{
	int dev;
	int cache_mode;
	char dev_name[32];

	if (sscanf(p, "%d %d %s", &dev, &cache_mode, dev_name) != 3) {
		printf("Param list error\n");
		return -1;
	}

#ifdef WIN32
	if (winpart) {
		file_windows_name_set(ext4_fname);
		bd = file_windows_dev_get();

	} else
#endif
	{
		file_dev_name_set(ext4_fname);
		bd = file_dev_get();
	}

	ext4_device_unregister_all();

	return ext4_device_register(bd, dev_name);
}

static int mount(const char *p)
{
	char dev_name[32];
	char mount_point[32];
	int rc;

	if (sscanf(p, "%s %s", dev_name, mount_point) != 2) {
		printf("Param list error\n");
		return -1;
	}

	if (verbose)
		ext4_dmask_set(DEBUG_ALL);

	rc = ext4_mount(dev_name, mount_point, false);
	if (rc != EOK)
		return rc;

	rc = ext4_recover(mount_point);
	if (rc != EOK && rc != ENOTSUP)
		return rc;

	rc = ext4_journal_start(mount_point);
	if (rc != EOK)
		return rc;

	if (cache_wb)
		ext4_cache_write_back(mount_point, 1);
	return rc;
}

static int umount(const char *p)
{
	char mount_point[32];
	int rc;

	if (sscanf(p, "%s", mount_point) != 1) {
		printf("Param list error\n");
		return -1;
	}

	if (cache_wb)
		ext4_cache_write_back(mount_point, 0);

	rc = ext4_journal_stop(mount_point);
	if (rc != EOK)
		return rc;

	rc = ext4_umount(mount_point);
	if (rc != EOK)
		return rc;

	return rc;
}

static int mount_point_stats(const char *p)
{
	char mount_point[32];
	int d;
	int rc;
	struct ext4_mount_stats stats;

	if (sscanf(p, "%s %d", mount_point, &d) != 2) {
		printf("Param list error\n");
		return -1;
	}

	rc = ext4_mount_point_stats(mount_point, &stats);

	if (rc != EOK)
		return rc;

	if (verbose) {
		printf("\tinodes_count = %" PRIu32"\n", stats.inodes_count);
		printf("\tfree_inodes_count = %" PRIu32"\n",
				stats.free_inodes_count);
		printf("\tblocks_count = %" PRIu64"\n", stats.blocks_count);
		printf("\tfree_blocks_count = %" PRIu64"\n",
				stats.free_blocks_count);
		printf("\tblock_size = %" PRIu32"\n", stats.block_size);
		printf("\tblock_group_count = %" PRIu32"\n",
				stats.block_group_count);
		printf("\tblocks_per_group = %" PRIu32"\n",
				stats.blocks_per_group);
		printf("\tinodes_per_group = %" PRIu32"\n",
				stats.inodes_per_group);
		printf("\tvolume_name = %s\n", stats.volume_name);
	}

	return rc;
}

static int cache_write_back(const char *p)
{
	char mount_point[32];
	int en;

	if (sscanf(p, "%s %d", mount_point, &en) != 2) {
		printf("Param list error\n");
		return -1;
	}

	return ext4_cache_write_back(mount_point, en);
}

static int fremove(const char *p)
{
	char path[255];

	if (sscanf(p, "%s", path) != 1) {
		printf("Param list error\n");
		return -1;
	}

	return ext4_fremove(path);
}

static int file_open(const char *p)
{
	int fid = MAX_FILES;
	char path[256];
	char flags[8];
	int rc;

	if (sscanf(p, "%d %s %s", &fid, path, flags) != 3) {
		printf("Param list error\n");
		return -1;
	}

	if (!(fid < MAX_FILES)) {
		printf("File id too big\n");
		return -1;
	}

	rc = ext4_fopen(&file_tab[fid].fd, path, flags);

	if (rc == EOK)
		strcpy(file_tab[fid].name, path);

	return rc;
}

static int file_close(const char *p)
{
	int fid = MAX_FILES;
	int rc;

	if (sscanf(p, "%d", &fid) != 1) {
		printf("Param list error\n");
		return -1;
	}

	if (!(fid < MAX_FILES)) {
		printf("File id too big\n");
		return -1;
	}

	if (file_tab[fid].name[0] == 0) {
		printf("File id empty\n");
		return -1;
	}

	rc = ext4_fclose(&file_tab[fid].fd);

	if (rc == EOK)
		file_tab[fid].name[0] = 0;

	return rc;
}

static int file_read(const char *p)
{
	int fid = MAX_FILES;
	int len;
	int d;
	int rc;
	size_t rb;

	if (sscanf(p, "%d %d %d %d", &fid, &d, &len, &d) != 4) {
		printf("Param list error\n");
		return -1;
	}

	if (!(fid < MAX_FILES)) {
		printf("File id too big\n");
		return -1;
	}

	if (file_tab[fid].name[0] == 0) {
		printf("File id empty\n");
		return -1;
	}

	while (len) {
		d = len > MAX_RW_BUFFER ? MAX_RW_BUFFER : len;

		memset(read_buffer, 0, MAX_RW_BUFFER);
		rc = ext4_fread(&file_tab[fid].fd, read_buffer, d, &rb);

		if (rc != EOK)
			break;

		if (rb != d) {
			printf("Read count error\n");
			return -1;
		}

		if (memcmp(read_buffer, write_buffer, d)) {
			printf("Read compare error\n");
			return -1;
		}

		len -= d;
	}

	return rc;
}

static int file_write(const char *p)
{
	int fid = MAX_FILES;
	int d;
	int rc;

	int len;
	size_t wb;

	if (sscanf(p, "%d %d %d %d", &fid, &d, &len, &d) != 4) {
		printf("Param list error\n");
		return -1;
	}

	if (!(fid < MAX_FILES)) {
		printf("File id too big\n");
		return -1;
	}

	if (file_tab[fid].name[0] == 0) {
		printf("File id empty\n");
		return -1;
	}

	while (len) {
		d = len > MAX_RW_BUFFER ? MAX_RW_BUFFER : len;
		rc = ext4_fwrite(&file_tab[fid].fd, write_buffer, d, &wb);

		if (rc != EOK)
			break;

		if (wb != d) {
			printf("Write count error\n");
			return -1;
		}

		len -= d;
	}

	return rc;
}

static int file_seek(const char *p)
{
	int fid = MAX_FILES;
	int off;
	int origin;

	if (sscanf(p, "%d %d %d", &fid, &off, &origin) != 3) {
		printf("Param list error\n");
		return -1;
	}

	if (!(fid < MAX_FILES)) {
		printf("File id too big\n");
		return -1;
	}

	if (file_tab[fid].name[0] == 0) {
		printf("File id empty\n");
		return -1;
	}

	return ext4_fseek(&file_tab[fid].fd, off, origin);
}

static int file_tell(const char *p)
{
	int fid = MAX_FILES;
	uint32_t exp_pos;

	if (sscanf(p, "%d %u", &fid, &exp_pos) != 2) {
		printf("Param list error\n");
		return -1;
	}

	if (!(fid < MAX_FILES)) {
		printf("File id too big\n");
		return -1;
	}

	if (file_tab[fid].name[0] == 0) {
		printf("File id empty\n");
		return -1;
	}

	if (exp_pos != ext4_ftell(&file_tab[fid].fd)) {
		printf("Expected filepos error\n");
		return -1;
	}

	return EOK;
}

static int file_size(const char *p)
{
	int fid = MAX_FILES;
	uint32_t exp_size;

	if (sscanf(p, "%d %u", &fid, &exp_size) != 2) {
		printf("Param list error\n");
		return -1;
	}

	if (!(fid < MAX_FILES)) {
		printf("File id too big\n");
		return -1;
	}

	if (file_tab[fid].name[0] == 0) {
		printf("File id empty\n");
		return -1;
	}

	if (exp_size != ext4_fsize(&file_tab[fid].fd)) {
		printf("Expected filesize error\n");
		return -1;
	}

	return EOK;
}

static int dir_rm(const char *p)
{
	char path[255];

	if (sscanf(p, "%s", path) != 1) {
		printf("Param list error\n");
		return -1;
	}

	return ext4_dir_rm(path);
}

static int dir_mk(const char *p)
{
	char path[255];

	if (sscanf(p, "%s", path) != 1) {
		printf("Param list error\n");
		return -1;
	}

	return ext4_dir_mk(path);
}

static int dir_open(const char *p)
{
	int did = MAX_DIRS;
	char path[255];
	int rc;

	if (sscanf(p, "%d %s", &did, path) != 2) {
		printf("Param list error\n");
		return -1;
	}

	if (!(did < MAX_DIRS)) {
		printf("Dir id too big\n");
		return -1;
	}

	rc = ext4_dir_open(&dir_tab[did].fd, path);

	if (rc == EOK)
		strcpy(dir_tab[did].name, path);

	return rc;
}

static int dir_close(const char *p)
{
	int did = MAX_DIRS;
	int rc;

	if (sscanf(p, "%d", &did) != 1) {
		printf("Param list error\n");
		return -1;
	}

	if (!(did < MAX_DIRS)) {
		printf("Dir id too big\n");
		return -1;
	}

	if (dir_tab[did].name[0] == 0) {
		printf("Dir id empty\n");
		return -1;
	}

	rc = ext4_dir_close(&dir_tab[did].fd);

	if (rc == EOK)
		dir_tab[did].name[0] = 0;

	return rc;
}

static int dir_entry_get(const char *p)
{
	int did = MAX_DIRS;
	int exp;
	char name[256];

	if (sscanf(p, "%d %d", &did, &exp) != 2) {
		printf("Param list error\n");
		return -1;
	}

	if (!(did < MAX_DIRS)) {
		printf("Dir id too big\n");
		return -1;
	}

	if (dir_tab[did].name[0] == 0) {
		printf("Dir id empty\n");
		return -1;
	}

	int idx = 0;
	const ext4_direntry *d;

	while ((d = ext4_dir_entry_next(&dir_tab[did].fd)) != NULL) {

		idx++;
		memcpy(name, d->name, d->name_length);
		name[d->name_length] = 0;
		if (verbose) {
			printf("\t%s %s\n", entry_to_str(d->inode_type), name);
		}
	}

	if (idx < 2) {
		printf("Minumum dir entry error\n");
		return -1;
	}

	if ((idx - 2) != exp) {
		printf("Expected dir entry error\n");
		return -1;
	}

	return EOK;
}

static int multi_fcreate(const char *p)
{
	char path[256];
	char path1[256];
	char prefix[32];
	int cnt;
	int rc;
	int i;
	ext4_file fd;

	if (sscanf(p, "%s %s %d", path, prefix, &cnt) != 3) {
		printf("Param list error\n");
		return -1;
	}

	for (i = 0; i < cnt; ++i) {
		sprintf(path1, "%s%s%d", path, prefix, i);
		rc = ext4_fopen(&fd, path1, "wb+");

		if (rc != EOK)
			break;
	}

	return rc;
}

static int multi_fwrite(const char *p)
{
	char path[256];
	char path1[256];
	char prefix[32];
	int cnt, i;
	int len, ll;
	int rc;
	size_t d, wb;
	ext4_file fd;

	if (sscanf(p, "%s %s %d %d", path, prefix, &cnt, &ll) != 4) {
		printf("Param list error\n");
		return -1;
	}

	for (i = 0; i < cnt; ++i) {
		sprintf(path1, "%s%s%d", path, prefix, i);
		rc = ext4_fopen(&fd, path1, "rb+");

		if (rc != EOK)
			break;

		len = ll;
		while (len) {
			d = len > MAX_RW_BUFFER ? MAX_RW_BUFFER : len;
			rc = ext4_fwrite(&fd, write_buffer, d, &wb);

			if (rc != EOK)
				break;

			if (wb != d) {
				printf("Write count error\n");
				return -1;
			}

			len -= d;
		}
	}

	return rc;
}

static int multi_fread(const char *p)
{
	char path[256];
	char path1[256];
	char prefix[32];
	int cnt;
	int len, ll;
	int rc ,i, d;
	size_t rb;
	ext4_file fd;

	if (sscanf(p, "%s %s %d %d", path, prefix, &cnt, &ll) != 4) {
		printf("Param list error\n");
		return -1;
	}

	for (i = 0; i < cnt; ++i) {
		sprintf(path1, "%s%s%d", path, prefix, i);
		rc = ext4_fopen(&fd, path1, "rb+");

		if (rc != EOK)
			break;

		len = ll;
		while (len) {
			d = len > MAX_RW_BUFFER ? MAX_RW_BUFFER : len;

			memset(read_buffer, 0, MAX_RW_BUFFER);
			rc = ext4_fread(&fd, read_buffer, d, &rb);

			if (rc != EOK)
				break;

			if (rb != d) {
				printf("Read count error\n");
				return -1;
			}

			if (memcmp(read_buffer, write_buffer, d)) {
				printf("Read compare error\n");
				return -1;
			}

			len -= d;
		}
	}

	return rc;
}

static int multi_fremove(const char *p)
{
	char path[256];
	char path1[256];
	char prefix[32];
	int cnt, i, rc;

	if (sscanf(p, "%s %s %d", path, prefix, &cnt) != 3) {
		printf("Param list error\n");
		return -1;
	}

	for (i = 0; i < cnt; ++i) {
		sprintf(path1, "%s%s%d", path, prefix, i);
		rc = ext4_fremove(path1);
		if (rc != EOK)
			break;
	}

	return rc;
}

static int multi_dcreate(const char *p)
{
	char path[256];
	char path1[256];
	char prefix[32];
	int cnt, i, rc;

	if (sscanf(p, "%s %s %d", path, prefix, &cnt) != 3) {
		printf("Param list error\n");
		return -1;
	}

	for (i = 0; i < cnt; ++i) {
		sprintf(path1, "%s%s%d", path, prefix, i);
		rc = ext4_dir_mk(path1);
		if (rc != EOK)
			break;
	}

	return rc;
}

static int multi_dremove(const char *p)
{
	char path[256];
	char path1[256];
	char prefix[32];
	int cnt, i, rc;

	if (sscanf(p, "%s %s %d", path, prefix, &cnt) != 3) {
		printf("Param list error\n");
		return -1;
	}

	for (i = 0; i < cnt; ++i) {
		sprintf(path1, "%s%s%d", path, prefix, i);
		rc = ext4_dir_rm(path1);
		if (rc != EOK)
			break;
	}

	return rc;
}

struct ext4_mount_stats saved_stats;

static int stats_save(const char *p)
{
	char path[256];

	if (sscanf(p, "%s", path) != 1) {
		printf("Param list error\n");
		return -1;
	}

	return ext4_mount_point_stats(path, &saved_stats);
}

static int stats_check(const char *p)
{
	char path[256];
	int rc;

	struct ext4_mount_stats actual_stats;

	if (sscanf(p, "%s", path) != 1) {
		printf("Param list error\n");
		return -1;
	}

	rc = ext4_mount_point_stats(path, &actual_stats);

	if (rc != EOK)
		return rc;

	if (memcmp(&saved_stats, &actual_stats,
		   sizeof(struct ext4_mount_stats))) {
		if (verbose) {
			printf("\tMount point stats error:\n");
			printf("\tsaved_stats:\n");
			printf("\tinodes_count = %" PRIu32"\n",
			       saved_stats.inodes_count);
			printf("\tfree_inodes_count = %" PRIu32"\n",
			       saved_stats.free_inodes_count);
			printf("\tblocks_count = %" PRIu64"\n",
			       saved_stats.blocks_count);
			printf("\tfree_blocks_count = %" PRIu64"\n",
			       saved_stats.free_blocks_count);
			printf("\tblock_size = %" PRIu32"\n",
					saved_stats.block_size);
			printf("\tblock_group_count = %" PRIu32"\n",
			       saved_stats.block_group_count);
			printf("\tblocks_per_group = %" PRIu32"\n",
			       saved_stats.blocks_per_group);
			printf("\tinodes_per_group = %" PRIu32"\n",
			       saved_stats.inodes_per_group);
			printf("\tvolume_name = %s\n", saved_stats.volume_name);
			printf("\tactual_stats:\n");
			printf("\tinodes_count = %" PRIu32"\n",
			       actual_stats.inodes_count);
			printf("\tfree_inodes_count = %" PRIu32"\n",
			       actual_stats.free_inodes_count);
			printf("\tblocks_count = %" PRIu64"\n",
			       actual_stats.blocks_count);
			printf("\tfree_blocks_count = %" PRIu64"\n",
			       actual_stats.free_blocks_count);
			printf("\tblock_size = %d\n", actual_stats.block_size);
			printf("\tblock_group_count = %" PRIu32"\n",
			       actual_stats.block_group_count);
			printf("\tblocks_per_group = %" PRIu32"\n",
			       actual_stats.blocks_per_group);
			printf("\tinodes_per_group = %" PRIu32"\n",
			       actual_stats.inodes_per_group);
			printf("\tvolume_name = %s\n",
			       actual_stats.volume_name);
		}
		return -1;
	}

	return rc;
}

static char *entry_to_str(uint8_t type)
{
	switch (type) {
	case EXT4_DE_UNKNOWN:
		return "[unk] ";
	case EXT4_DE_REG_FILE:
		return "[fil] ";
	case EXT4_DE_DIR:
		return "[dir] ";
	case EXT4_DE_CHRDEV:
		return "[cha] ";
	case EXT4_DE_BLKDEV:
		return "[blk] ";
	case EXT4_DE_FIFO:
		return "[fif] ";
	case EXT4_DE_SOCK:
		return "[soc] ";
	case EXT4_DE_SYMLINK:
		return "[sym] ";
	default:
		break;
	}
	return "[???]";
}

static int winsock_init(void)
{
#if WIN32
	int rc;
	static WSADATA wsaData;
	rc = WSAStartup(MAKEWORD(2, 2), &wsaData);
	if (rc != 0) {
		return -1;
	}
#endif
	return 0;
}

static void winsock_fini(void)
{
#if WIN32
	WSACleanup();
#endif
}