stty: fix option parsing bug (spotted by Sascha Hauer <s.hauer@pengutronix.de>)
[oweals/busybox.git] / archival / cpio.c
1 /* vi: set sw=4 ts=4: */
2 /*
3  * Mini cpio implementation for busybox
4  *
5  * Copyright (C) 2001 by Glenn McGrath
6  *
7  * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
8  *
9  * Limitations:
10  *              Doesn't check CRC's
11  *              Only supports new ASCII and CRC formats
12  *
13  */
14 #include <fcntl.h>
15 #include <stdlib.h>
16 #include <string.h>
17 #include <unistd.h>
18 #include "unarchive.h"
19 #include "busybox.h"
20
21 #define CPIO_OPT_EXTRACT                        0x01
22 #define CPIO_OPT_TEST                           0x02
23 #define CPIO_OPT_UNCONDITIONAL          0x04
24 #define CPIO_OPT_VERBOSE                        0x08
25 #define CPIO_OPT_FILE                           0x10
26 #define CPIO_OPT_CREATE_LEADING_DIR     0x20
27 #define CPIO_OPT_PRESERVE_MTIME         0x40
28
29 int cpio_main(int argc, char **argv);
30 int cpio_main(int argc, char **argv)
31 {
32         archive_handle_t *archive_handle;
33         char *cpio_filename = NULL;
34         unsigned opt;
35
36         /* Initialise */
37         archive_handle = init_handle();
38         archive_handle->src_fd = STDIN_FILENO;
39         archive_handle->seek = seek_by_read;
40         archive_handle->flags = ARCHIVE_EXTRACT_NEWER | ARCHIVE_PRESERVE_DATE;
41
42         opt = getopt32(argc, argv, "ituvF:dm", &cpio_filename);
43
44         /* One of either extract or test options must be given */
45         if ((opt & (CPIO_OPT_TEST | CPIO_OPT_EXTRACT)) == 0) {
46                 bb_show_usage();
47         }
48
49         if (opt & CPIO_OPT_TEST) {
50                 /* if both extract and test options are given, ignore extract option */
51                 if (opt & CPIO_OPT_EXTRACT) {
52                         opt &= ~CPIO_OPT_EXTRACT;
53                 }
54                 archive_handle->action_header = header_list;
55         }
56         if (opt & CPIO_OPT_EXTRACT) {
57                 archive_handle->action_data = data_extract_all;
58         }
59         if (opt & CPIO_OPT_UNCONDITIONAL) {
60                 archive_handle->flags |= ARCHIVE_EXTRACT_UNCONDITIONAL;
61                 archive_handle->flags &= ~ARCHIVE_EXTRACT_NEWER;
62         }
63         if (opt & CPIO_OPT_VERBOSE) {
64                 if (archive_handle->action_header == header_list) {
65                         archive_handle->action_header = header_verbose_list;
66                 } else {
67                         archive_handle->action_header = header_list;
68                 }
69         }
70         if (cpio_filename) { /* CPIO_OPT_FILE */
71                 archive_handle->src_fd = xopen(cpio_filename, O_RDONLY);
72                 archive_handle->seek = seek_by_jump;
73         }
74         if (opt & CPIO_OPT_CREATE_LEADING_DIR) {
75                 archive_handle->flags |= ARCHIVE_CREATE_LEADING_DIRS;
76         }
77
78         while (optind < argc) {
79                 archive_handle->filter = filter_accept_list;
80                 llist_add_to(&(archive_handle->accept), argv[optind]);
81                 optind++;
82         }
83
84         while (get_header_cpio(archive_handle) == EXIT_SUCCESS);
85
86         return EXIT_SUCCESS;
87 }