blob: 4e44a87e9e816564acd55021434b3dbf4be75f3e [file] [log] [blame]
"Robert P. J. Day"63fc1a92006-07-02 19:47:05 +00001/* vi: set sw=4 ts=4: */
Glenn L McGrath5699b852003-11-15 23:19:05 +00002/*
Denys Vlasenko0ef64bd2010-08-16 20:14:46 +02003 * Licensed under GPLv2 or later, see file LICENSE in this source tree.
Glenn L McGrath5699b852003-11-15 23:19:05 +00004 */
5
Glenn L McGrath5699b852003-11-15 23:19:05 +00006#include "libbb.h"
Denys Vlasenkod184a722011-09-22 12:45:14 +02007#include "bb_archive.h"
Bernhard Reutner-Fischercfb53df2006-04-02 21:50:01 +00008
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +01009void FAST_FUNC init_transformer_aux_data(transformer_aux_data_t *aux)
10{
11 memset(aux, 0, sizeof(*aux));
12}
Denys Vlasenko59655072012-03-06 16:23:50 +010013
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +010014int FAST_FUNC check_signature16(transformer_aux_data_t *aux, int src_fd, unsigned magic16)
15{
16 if (aux && aux->check_signature) {
17 uint16_t magic2;
18 if (full_read(src_fd, &magic2, 2) != 2 || magic2 != magic16) {
19 bb_error_msg("invalid magic");
20#if 0 /* possible future extension */
21 if (aux->check_signature > 1)
22 xfunc_die();
Denys Vlasenko59655072012-03-06 16:23:50 +010023#endif
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +010024 return -1;
25 }
26 }
27 return 0;
28}
Denys Vlasenko59655072012-03-06 16:23:50 +010029
Denys Vlasenkofaac1d32012-03-06 16:33:42 +010030void check_errors_in_children(int signo)
31{
32 int status;
33
34 if (!signo) {
35 /* block waiting for any child */
36 if (wait(&status) < 0)
Denys Vlasenko577235d2013-02-28 16:38:25 +010037//FIXME: check EINTR?
Denys Vlasenkofaac1d32012-03-06 16:33:42 +010038 return; /* probably there are no children */
39 goto check_status;
40 }
41
42 /* Wait for any child without blocking */
43 for (;;) {
44 if (wait_any_nohang(&status) < 0)
Denys Vlasenko577235d2013-02-28 16:38:25 +010045//FIXME: check EINTR?
Denys Vlasenkofaac1d32012-03-06 16:33:42 +010046 /* wait failed?! I'm confused... */
47 return;
48 check_status:
Denys Vlasenko577235d2013-02-28 16:38:25 +010049 /*if (WIFEXITED(status) && WEXITSTATUS(status) == 0)*/
50 /* On Linux, the above can be checked simply as: */
51 if (status == 0)
Denys Vlasenkofaac1d32012-03-06 16:33:42 +010052 /* this child exited with 0 */
53 continue;
Denys Vlasenko577235d2013-02-28 16:38:25 +010054 /* Cannot happen:
55 if (!WIFSIGNALED(status) && !WIFEXITED(status)) ???;
56 */
Denys Vlasenkofaac1d32012-03-06 16:33:42 +010057 bb_got_signal = 1;
58 }
59}
60
Glenn L McGrath5699b852003-11-15 23:19:05 +000061/* transformer(), more than meets the eye */
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +010062#if BB_MMU
Denis Vlasenkob6052722008-07-10 17:43:01 +000063void FAST_FUNC open_transformer(int fd,
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +010064 int check_signature,
65 IF_DESKTOP(long long) int FAST_FUNC (*transformer)(transformer_aux_data_t *aux, int src_fd, int dst_fd)
66)
67#else
68void FAST_FUNC open_transformer(int fd, const char *transform_prog)
69#endif
Glenn L McGrath5699b852003-11-15 23:19:05 +000070{
Denis Vlasenko37188322008-02-16 13:20:56 +000071 struct fd_pair fd_pipe;
Glenn L McGrath5699b852003-11-15 23:19:05 +000072 int pid;
73
Denis Vlasenko37188322008-02-16 13:20:56 +000074 xpiped_pair(fd_pipe);
Pascal Bellard926031b2010-07-04 15:32:38 +020075 pid = BB_MMU ? xfork() : xvfork();
Glenn L McGrath5699b852003-11-15 23:19:05 +000076 if (pid == 0) {
Pascal Bellard926031b2010-07-04 15:32:38 +020077 /* Child */
Denis Vlasenkob6052722008-07-10 17:43:01 +000078 close(fd_pipe.rd); /* we don't want to read from the parent */
Denis Vlasenkoea620772006-10-14 02:23:43 +000079 // FIXME: error check?
Denis Vlasenko211f7f82007-09-05 11:48:32 +000080#if BB_MMU
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +010081 {
82 transformer_aux_data_t aux;
83 init_transformer_aux_data(&aux);
84 aux.check_signature = check_signature;
85 transformer(&aux, fd, fd_pipe.wr);
86 if (ENABLE_FEATURE_CLEAN_UP) {
87 close(fd_pipe.wr); /* send EOF */
88 close(fd);
89 }
90 /* must be _exit! bug was actually seen here */
91 _exit(EXIT_SUCCESS);
Denis Vlasenko7e0fbf92007-09-04 19:33:22 +000092 }
Denis Vlasenko211f7f82007-09-05 11:48:32 +000093#else
Denis Vlasenko059c9172007-11-12 02:13:12 +000094 {
95 char *argv[4];
Denis Vlasenkob6052722008-07-10 17:43:01 +000096 xmove_fd(fd, 0);
Denis Vlasenko37188322008-02-16 13:20:56 +000097 xmove_fd(fd_pipe.wr, 1);
Denis Vlasenko059c9172007-11-12 02:13:12 +000098 argv[0] = (char*)transform_prog;
99 argv[1] = (char*)"-cf";
100 argv[2] = (char*)"-";
101 argv[3] = NULL;
102 BB_EXECVP(transform_prog, argv);
Denis Vlasenkof9d4fc32009-04-21 20:40:51 +0000103 bb_perror_msg_and_die("can't execute '%s'", transform_prog);
Denis Vlasenko059c9172007-11-12 02:13:12 +0000104 }
Denis Vlasenko211f7f82007-09-05 11:48:32 +0000105#endif
Denis Vlasenkoea620772006-10-14 02:23:43 +0000106 /* notreached */
Glenn L McGrath5699b852003-11-15 23:19:05 +0000107 }
Glenn L McGrath20872be2003-11-18 21:31:19 +0000108
Glenn L McGrath5699b852003-11-15 23:19:05 +0000109 /* parent process */
Denis Vlasenkob6052722008-07-10 17:43:01 +0000110 close(fd_pipe.wr); /* don't want to write to the child */
111 xmove_fd(fd_pipe.rd, fd);
Glenn L McGrath5699b852003-11-15 23:19:05 +0000112}
Denys Vlasenko59655072012-03-06 16:23:50 +0100113
114
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100115#if SEAMLESS_COMPRESSION
116
Denys Vlasenko59655072012-03-06 16:23:50 +0100117/* Used by e.g. rpm which gives us a fd without filename,
118 * thus we can't guess the format from filename's extension.
119 */
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100120int FAST_FUNC setup_unzip_on_fd(int fd, int fail_if_not_detected)
Denys Vlasenko59655072012-03-06 16:23:50 +0100121{
Denys Vlasenko59655072012-03-06 16:23:50 +0100122 union {
123 uint8_t b[4];
124 uint16_t b16[2];
125 uint32_t b32[1];
126 } magic;
127 int offset = -2;
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100128 USE_FOR_MMU(IF_DESKTOP(long long) int FAST_FUNC (*xformer)(transformer_aux_data_t *aux, int src_fd, int dst_fd);)
129 USE_FOR_NOMMU(const char *xformer_prog;)
Denys Vlasenko59655072012-03-06 16:23:50 +0100130
131 /* .gz and .bz2 both have 2-byte signature, and their
132 * unpack_XXX_stream wants this header skipped. */
133 xread(fd, magic.b16, sizeof(magic.b16[0]));
134 if (ENABLE_FEATURE_SEAMLESS_GZ
135 && magic.b16[0] == GZIP_MAGIC
136 ) {
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100137 USE_FOR_MMU(xformer = unpack_gz_stream;)
138 USE_FOR_NOMMU(xformer_prog = "gunzip";)
Denys Vlasenko59655072012-03-06 16:23:50 +0100139 goto found_magic;
140 }
141 if (ENABLE_FEATURE_SEAMLESS_BZ2
142 && magic.b16[0] == BZIP2_MAGIC
143 ) {
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100144 USE_FOR_MMU(xformer = unpack_bz2_stream;)
145 USE_FOR_NOMMU(xformer_prog = "bunzip2";)
Denys Vlasenko59655072012-03-06 16:23:50 +0100146 goto found_magic;
147 }
148 if (ENABLE_FEATURE_SEAMLESS_XZ
149 && magic.b16[0] == XZ_MAGIC1
150 ) {
151 offset = -6;
152 xread(fd, magic.b32, sizeof(magic.b32[0]));
153 if (magic.b32[0] == XZ_MAGIC2) {
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100154 USE_FOR_MMU(xformer = unpack_xz_stream;)
155 USE_FOR_NOMMU(xformer_prog = "unxz";)
Denys Vlasenko59655072012-03-06 16:23:50 +0100156 goto found_magic;
157 }
158 }
159
160 /* No known magic seen */
161 if (fail_if_not_detected)
162 bb_error_msg_and_die("no gzip"
163 IF_FEATURE_SEAMLESS_BZ2("/bzip2")
164 IF_FEATURE_SEAMLESS_XZ("/xz")
165 " magic");
166 xlseek(fd, offset, SEEK_CUR);
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100167 return 1;
Denys Vlasenko59655072012-03-06 16:23:50 +0100168
169 found_magic:
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100170# if BB_MMU
171 open_transformer_with_no_sig(fd, xformer);
172# else
Denys Vlasenko59655072012-03-06 16:23:50 +0100173 /* NOMMU version of open_transformer execs
174 * an external unzipper that wants
175 * file position at the start of the file */
176 xlseek(fd, offset, SEEK_CUR);
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100177 open_transformer_with_sig(fd, xformer, xformer_prog);
Denys Vlasenko59655072012-03-06 16:23:50 +0100178# endif
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100179 return 0;
Denys Vlasenko59655072012-03-06 16:23:50 +0100180}
Denys Vlasenko59655072012-03-06 16:23:50 +0100181
182int FAST_FUNC open_zipped(const char *fname)
183{
Denys Vlasenko59655072012-03-06 16:23:50 +0100184 char *sfx;
185 int fd;
186
187 fd = open(fname, O_RDONLY);
188 if (fd < 0)
189 return fd;
190
191 sfx = strrchr(fname, '.');
192 if (sfx) {
193 sfx++;
194 if (ENABLE_FEATURE_SEAMLESS_LZMA && strcmp(sfx, "lzma") == 0)
195 /* .lzma has no header/signature, just trust it */
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100196 open_transformer_with_sig(fd, unpack_lzma_stream, "unlzma");
Denys Vlasenko59655072012-03-06 16:23:50 +0100197 else
198 if ((ENABLE_FEATURE_SEAMLESS_GZ && strcmp(sfx, "gz") == 0)
199 || (ENABLE_FEATURE_SEAMLESS_BZ2 && strcmp(sfx, "bz2") == 0)
200 || (ENABLE_FEATURE_SEAMLESS_XZ && strcmp(sfx, "xz") == 0)
201 ) {
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100202 setup_unzip_on_fd(fd, /*fail_if_not_detected:*/ 1);
Denys Vlasenko59655072012-03-06 16:23:50 +0100203 }
204 }
205
206 return fd;
Denys Vlasenko59655072012-03-06 16:23:50 +0100207}
208
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100209#endif /* SEAMLESS_COMPRESSION */
210
Denys Vlasenko59655072012-03-06 16:23:50 +0100211void* FAST_FUNC xmalloc_open_zipped_read_close(const char *fname, size_t *maxsz_p)
212{
213 int fd;
214 char *image;
215
216 fd = open_zipped(fname);
217 if (fd < 0)
218 return NULL;
219
220 image = xmalloc_read(fd, maxsz_p);
221 if (!image)
222 bb_perror_msg("read error from '%s'", fname);
223 close(fd);
224
225 return image;
226}