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