HelenOS sources
This source file includes following definitions.
- print_syntax
- newfile_abort_query
- newfile_progress
- newfile_io_error_query
- main
#include <capa.h>
#include <errno.h>
#include <fmgt.h>
#include <io/console.h>
#include <io/cons_event.h>
#include <io/kbd_event.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <str.h>
#include <str_error.h>
#define NAME "newfile"
static bool newfile_abort_query(void *);
static void newfile_progress(void *, fmgt_progress_t *);
static fmgt_error_action_t newfile_io_error_query(void *, fmgt_io_error_t *);
static bool prog_upd = false;
static bool nonint = false;
static bool quiet = false;
static console_ctrl_t *con;
static fmgt_cb_t newfile_fmgt_cb = {
.abort_query = newfile_abort_query,
.io_error_query = newfile_io_error_query,
.progress = newfile_progress,
};
static void print_syntax(void)
{
printf("Create new file.\n");
printf("Syntax: %s [<options] [<file-name>]\n", NAME);
printf("\t-h help\n");
printf("\t-n non-interactive\n");
printf("\t-p create sparse file\n");
printf("\t-q quiet\n");
printf("\t-size=<cap> file size (<number>[<kB>|<MB>|...])\n");
}
static bool newfile_abort_query(void *arg)
{
cons_event_t event;
kbd_event_t *ev;
errno_t rc;
usec_t timeout;
if (con == NULL)
return false;
timeout = 0;
rc = console_get_event_timeout(con, &event, &timeout);
if (rc != EOK)
return false;
if (event.type == CEV_KEY && event.ev.key.type == KEY_PRESS) {
ev = &event.ev.key;
if ((ev->mods & KM_ALT) == 0 &&
(ev->mods & KM_SHIFT) == 0 &&
(ev->mods & KM_CTRL) != 0) {
if (ev->key == KC_C)
return true;
}
}
return false;
}
static void newfile_progress(void *arg, fmgt_progress_t *progress)
{
(void)arg;
if (!quiet) {
printf("\rWritten %s of %s (%s done).", progress->curf_procb,
progress->curf_totalb, progress->curf_percent);
fflush(stdout);
prog_upd = true;
}
}
static fmgt_error_action_t newfile_io_error_query(void *arg,
fmgt_io_error_t *err)
{
cons_event_t event;
kbd_event_t *ev;
errno_t rc;
(void)arg;
if (nonint)
return fmgt_er_abort;
if (prog_upd)
putchar('\n');
fprintf(stderr, "I/O error %s file '%s' (%s).\n",
err->optype == fmgt_io_write ? "writing" : "reading",
err->fname, str_error(err->rc));
fprintf(stderr, "[A]bort or [R]etry?\n");
if (con == NULL)
return fmgt_er_abort;
while (true) {
rc = console_get_event(con, &event);
if (rc != EOK)
return fmgt_er_abort;
if (event.type == CEV_KEY && event.ev.key.type == KEY_PRESS) {
ev = &event.ev.key;
if ((ev->mods & KM_ALT) == 0 &&
(ev->mods & KM_CTRL) == 0) {
if (ev->c == 'r' || ev->c == 'R')
return fmgt_er_retry;
if (ev->c == 'a' || ev->c == 'A')
return fmgt_er_abort;
}
}
if (event.type == CEV_KEY && event.ev.key.type == KEY_PRESS) {
ev = &event.ev.key;
if ((ev->mods & KM_ALT) == 0 &&
(ev->mods & KM_SHIFT) == 0 &&
(ev->mods & KM_CTRL) != 0) {
if (ev->key == KC_C)
return fmgt_er_abort;
}
}
}
}
int main(int argc, char *argv[])
{
fmgt_t *fmgt = NULL;
errno_t rc;
int i;
bool sparse = false;
char *fsize = NULL;
char *fname = NULL;
capa_spec_t fcap;
uint64_t nbytes = 0;
con = console_init(stdin, stdout);
i = 1;
while (i < argc && argv[i][0] == '-') {
if (str_cmp(argv[i], "-h") == 0) {
print_syntax();
return 0;
} else if (str_cmp(argv[i], "-n") == 0) {
++i;
nonint = true;
} else if (str_cmp(argv[i], "-p") == 0) {
++i;
sparse = true;
} else if (str_cmp(argv[i], "-q") == 0) {
++i;
quiet = true;
} else if (str_lcmp(argv[i], "-size=",
str_length("-size=")) == 0) {
fsize = argv[i] + str_length("-size=");
++i;
} else {
printf("Invalid option '%s'.\n", argv[i]);
print_syntax();
goto error;
}
}
if (i < argc) {
fname = str_dup(argv[i++]);
if (fname == NULL) {
printf("Out of memory.\n");
goto error;
}
}
if (i < argc) {
printf("Unexpected argument.\n");
print_syntax();
goto error;
}
if (fname == NULL) {
rc = fmgt_new_file_suggest(&fname);
if (rc != EOK) {
printf("Out of memory.\n");
goto error;
}
}
(void)nonint;
(void)quiet;
(void)sparse;
if (fsize != NULL) {
rc = capa_parse(fsize, &fcap);
if (rc != EOK) {
printf("Invalid file size '%s'.\n", fsize);
goto error;
}
rc = capa_to_blocks(&fcap, cv_nom, 1, &nbytes);
if (rc != EOK) {
printf("File size too large '%s'.\n", fsize);
goto error;
}
}
rc = fmgt_create(&fmgt);
if (rc != EOK) {
printf("Out of memory.\n");
goto error;
}
fmgt_set_cb(fmgt, &newfile_fmgt_cb, NULL);
rc = fmgt_new_file(fmgt, fname, nbytes, sparse ? nf_sparse : nf_none);
if (prog_upd)
putchar('\n');
if (rc != EOK) {
printf("Error creating file: %s.\n", str_error(rc));
goto error;
}
free(fname);
fmgt_destroy(fmgt);
return 0;
error:
if (fname != NULL)
free(fname);
if (fmgt != NULL)
fmgt_destroy(fmgt);
return 1;
}
HelenOS homepage, sources at GitHub