blob: 7256aae6bc0102221cc11318b04f42d0957dc60b [file] [log] [blame]
Glenn L McGrathf8736d22001-06-26 01:19:34 +00001/* vi: set sw=4 ts=4: */
2/*
3 * Mini rpm2cpio implementation for busybox
4 *
5 * Copyright (C) 2001 by Laurence Anderson
6 *
Denys Vlasenko0ef64bd2010-08-16 20:14:46 +02007 * Licensed under GPLv2 or later, see file LICENSE in this source tree.
Glenn L McGrathf8736d22001-06-26 01:19:34 +00008 */
Pere Orga1f4447b2011-03-27 22:40:30 +02009
10//usage:#define rpm2cpio_trivial_usage
11//usage: "package.rpm"
12//usage:#define rpm2cpio_full_usage "\n\n"
13//usage: "Output a cpio archive of the rpm file"
14
Denis Vlasenkob6adbf12007-05-26 19:00:18 +000015#include "libbb.h"
Denys Vlasenkod184a722011-09-22 12:45:14 +020016#include "bb_archive.h"
Denys Vlasenko27653ad2010-05-06 14:19:19 +000017#include "rpm.h"
Glenn L McGrathf8736d22001-06-26 01:19:34 +000018
Denys Vlasenkoe6c483e2009-08-28 21:15:24 +020019enum { rpm_fd = STDIN_FILENO };
20
21static unsigned skip_header(void)
Glenn L McGrathf8736d22001-06-26 01:19:34 +000022{
23 struct rpm_header header;
Denys Vlasenko0a130d52009-08-28 21:09:51 +020024 unsigned len;
Glenn L McGrathf8736d22001-06-26 01:19:34 +000025
Pascal Bellard7f214942009-08-28 06:20:33 +020026 xread(rpm_fd, &header, sizeof(header));
27// if (strncmp((char *) &header.magic, RPM_HEADER_MAGIC_STR, 3) != 0) {
28// bb_error_msg_and_die("invalid RPM header magic");
29// }
30// if (header.version != 1) {
31// bb_error_msg_and_die("unsupported RPM header version");
32// }
Denys Vlasenkoe6c483e2009-08-28 21:15:24 +020033 if (header.magic_and_ver != htonl(RPM_HEADER_MAGICnVER)) {
Pascal Bellard7f214942009-08-28 06:20:33 +020034 bb_error_msg_and_die("invalid RPM header magic or unsupported version");
Denys Vlasenkoe6c483e2009-08-28 21:15:24 +020035 // ": %x != %x", header.magic_and_ver, htonl(RPM_HEADER_MAGICnVER));
Glenn L McGrath7ca04f32002-09-25 02:47:48 +000036 }
Denys Vlasenko0a130d52009-08-28 21:09:51 +020037
38 /* Seek past index entries, and past store */
39 len = 16 * ntohl(header.entries) + ntohl(header.size);
40 seek_by_jump(rpm_fd, len);
41
42 return sizeof(header) + len;
Glenn L McGrathf8736d22001-06-26 01:19:34 +000043}
44
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +010045#if SEAMLESS_COMPRESSION
46static void handle_SIGCHLD(int signo UNUSED_PARAM)
47{
48 int status;
49
50 /* Wait for any child without blocking */
51 for (;;) {
52 if (wait_any_nohang(&status) < 0)
53 /* wait failed?! I'm confused... */
54 return;
55 if (WIFEXITED(status) && WEXITSTATUS(status) == 0)
56 /* this child exited with 0 */
57 continue;
58 /* Cannot happen?
59 if (!WIFSIGNALED(status) && !WIFEXITED(status)) ???; */
60 bb_got_signal = 1;
61 }
62}
63#endif
64
Glenn L McGrathf8736d22001-06-26 01:19:34 +000065/* No getopt required */
Denis Vlasenko9b49a5e2007-10-11 10:05:36 +000066int rpm2cpio_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
Pascal Bellard7f214942009-08-28 06:20:33 +020067int rpm2cpio_main(int argc UNUSED_PARAM, char **argv)
Glenn L McGrathf8736d22001-06-26 01:19:34 +000068{
69 struct rpm_lead lead;
Denys Vlasenko0a130d52009-08-28 21:09:51 +020070 unsigned pos;
Glenn L McGrathf8736d22001-06-26 01:19:34 +000071
Denys Vlasenkoe6c483e2009-08-28 21:15:24 +020072 if (argv[1]) {
73 xmove_fd(xopen(argv[1], O_RDONLY), rpm_fd);
Glenn L McGrathf8736d22001-06-26 01:19:34 +000074 }
Pascal Bellard7f214942009-08-28 06:20:33 +020075 xread(rpm_fd, &lead, sizeof(lead));
Glenn L McGrathf8736d22001-06-26 01:19:34 +000076
Pascal Bellard7f214942009-08-28 06:20:33 +020077 /* Just check the magic, the rest is irrelevant */
Denys Vlasenko27653ad2010-05-06 14:19:19 +000078 if (lead.magic != htonl(RPM_LEAD_MAGIC)) {
Pascal Bellard7f214942009-08-28 06:20:33 +020079 bb_error_msg_and_die("invalid RPM magic");
Glenn L McGrath7ca04f32002-09-25 02:47:48 +000080 }
81
Pascal Bellard7f214942009-08-28 06:20:33 +020082 /* Skip the signature header, align to 8 bytes */
Denys Vlasenkoe6c483e2009-08-28 21:15:24 +020083 pos = skip_header();
Denys Vlasenko27653ad2010-05-06 14:19:19 +000084 seek_by_jump(rpm_fd, (-(int)pos) & 7);
Glenn L McGrath7ca04f32002-09-25 02:47:48 +000085
Glenn L McGrathf8736d22001-06-26 01:19:34 +000086 /* Skip the main header */
Denys Vlasenkoe6c483e2009-08-28 21:15:24 +020087 skip_header();
Eric Andersenc7bda1c2004-03-15 08:29:22 +000088
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +010089#if SEAMLESS_COMPRESSION
90 /* We need to know whether child (gzip/bzip/etc) exits abnormally */
91 signal(SIGCHLD, handle_SIGCHLD);
92#endif
93
Denys Vlasenko27653ad2010-05-06 14:19:19 +000094 /* This works, but doesn't report uncompress errors (they happen in child) */
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +010095 setup_unzip_on_fd(rpm_fd, /*fail_if_not_detected:*/ 1);
Denys Vlasenko27653ad2010-05-06 14:19:19 +000096 if (bb_copyfd_eof(rpm_fd, STDOUT_FILENO) < 0)
Pascal Bellard7f214942009-08-28 06:20:33 +020097 bb_error_msg_and_die("error unpacking");
Glenn L McGrath26a0d9a2001-07-13 06:49:18 +000098
Pascal Bellard7f214942009-08-28 06:20:33 +020099 if (ENABLE_FEATURE_CLEAN_UP) {
100 close(rpm_fd);
101 }
Glenn L McGrath7ca04f32002-09-25 02:47:48 +0000102
Denys Vlasenko8a6a2f92012-03-06 16:27:48 +0100103#if SEAMLESS_COMPRESSION
104 return bb_got_signal;
105#else
106 return EXIT_SUCCESS;
107#endif
Glenn L McGrathf8736d22001-06-26 01:19:34 +0000108}