summaryrefslogtreecommitdiff
path: root/src/commands/dd.c
blob: 6b973e2bfc2f8802c165a5f5ff09b0dbdb7dcf55 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
#include "../command.h"

static void help(void) {
    printf("Usage: dd [if=FILE] [of=FILE] [bs=N] [count=N]\n\n");
    printf("Copy a file with converting and formatting\n\n");
    printf("\tif=FILE\t\tRead from FILE instead of stdin\n");
    printf("\tof=FILE\t\tWrite to FILE instead of stdout\n");
    printf("\tbs=N\t\tRead and write N bytes at a time\n");
    printf("\tcount=N\t\tCopy only N input blocks\n");
}

COMMAND(dd) {

    FILE* in_file = stdin;
    FILE* out_file = stdout;
    int bs = 1024;
    int count = -1;

    parse_help(argc, argv, help);

    for (int i = 0; i < argc; i++) {
        if (prefix("if=", argv[i])) {
            char* path = argv[i] + 3;
            in_file = get_file(path, "rb");
        } else if (prefix("of=", argv[i])) {
            char* path = argv[i] + 3;
            out_file = get_file(path, "wb");
        } else if (prefix("bs=", argv[i])) {
            char* str = argv[i] + 3;
            bs = get_number(str);
            if (bs < 1) {
                error("error: block size must be greater than 0");
            }
        } else if (prefix("count=", argv[i])) {
            char* str = argv[i] + 6;
            count = get_number(str);
            if (count < 1) {
                error("error: count must be greather than 0");
            }
        } else {
            error("error: unkown option %s", argv[i]);
        }
    }
 
    char* buffer = malloc(bs);
    size_t read;
    while ((read = fread(buffer, 1, bs, in_file)) != 0) {
        fwrite(buffer, 1, read, out_file);
        if (--count, count == 0) break;
    }

    free(buffer);
    fclose(in_file);
    fclose(out_file);

    return EXIT_SUCCESS;
}