1 /* Copyright (C) 1999, 2000, 2002 Red Hat, Inc.
2 This file is part of elfutils.
3 Written by Ulrich Drepper <drepper@redhat.com>, 1999.
4
5 This file is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 3 of the License, or
8 (at your option) any later version.
9
10 elfutils is distributed in the hope that it will be useful, but
11 WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
14
15 You should have received a copy of the GNU General Public License
16 along with this program. If not, see <http://www.gnu.org/licenses/>. */
17
18 #ifdef HAVE_CONFIG_H
19 # include <config.h>
20 #endif
21
22 #include <fcntl.h>
23 #include <gelf.h>
24 #include <stdio.h>
25 #include <stdlib.h>
26 #include <string.h>
27 #include <unistd.h>
28 #include <system.h>
29
30
31 int
main(int argc,char * argv[])32 main (int argc, char *argv[])
33 {
34 int fd;
35 Elf *elf;
36 Elf *subelf;
37 Elf_Cmd cmd;
38 off_t offset;
39 size_t todo;
40
41 if (argc < 4)
42 exit (1);
43
44 /* Open the archive. */
45 fd = open (argv[1], O_RDONLY);
46 if (fd == -1)
47 {
48 printf ("Cannot open input file: %m");
49 exit (1);
50 }
51
52 /* Set the ELF version. */
53 elf_version (EV_CURRENT);
54
55 /* Create an ELF descriptor. */
56 cmd = ELF_C_READ;
57 elf = elf_begin (fd, cmd, NULL);
58 if (elf == NULL)
59 {
60 printf ("Cannot create ELF descriptor: %s\n", elf_errmsg (-1));
61 exit (1);
62 }
63
64 /* If it is no archive punt. */
65 if (elf_kind (elf) != ELF_K_AR)
66 {
67 printf ("`%s' is no archive\n", argv[1]);
68 exit (1);
69 }
70
71 /* Get the elements of the archive one after the other. */
72 while ((subelf = elf_begin (fd, cmd, elf)) != NULL)
73 {
74 /* The the header for this element. */
75 Elf_Arhdr *arhdr = elf_getarhdr (subelf);
76
77 if (arhdr == NULL)
78 {
79 printf ("cannot get arhdr: %s\n", elf_errmsg (-1));
80 exit (1);
81 }
82
83 if (strcmp (arhdr->ar_name, argv[2]) == 0)
84 {
85 int outfd;
86
87 /* Get the offset of the file in the archive. */
88 offset = elf_getbase (subelf);
89 if (offset == -1)
90 {
91 printf ("\
92 Failed to get base address for the archive element: %s\n",
93 elf_errmsg (-1));
94 exit (1);
95 }
96
97 /* Open the output file. */
98 outfd = open (argv[3], O_CREAT | O_TRUNC | O_RDWR, DEFFILEMODE);
99 if (outfd == -1)
100 {
101 printf ("cannot open output file: %m");
102 exit (1);
103 }
104
105 /* Now write out the data. */
106 todo = arhdr->ar_size;
107 while (todo > 0)
108 {
109 char buf[1024];
110 ssize_t n = pread (fd, buf, MIN (sizeof buf, todo), offset);
111 if (n == 0)
112 break;
113
114 if (write (outfd, buf, n) != n)
115 {
116 puts ("Writing output failed");
117 exit (1);
118 }
119
120 offset += n;
121 todo -= n;
122 }
123
124 /* Check whether all the date was read and written out. */
125 if (todo != 0)
126 {
127 puts ("Reading archive member failed.");
128 exit (1);
129 }
130
131 /* Close the descriptors. */
132 if (elf_end (subelf) != 0 || elf_end (elf) != 0)
133 {
134 printf ("Freeing ELF descriptors failed: %s", elf_errmsg (-1));
135 exit (1);
136 }
137
138 close (outfd);
139 close (fd);
140
141 /* All went well. */
142 exit (0);
143 }
144
145 /* Get next archive element. */
146 cmd = elf_next (subelf);
147 if (elf_end (subelf) != 0)
148 {
149 printf ("error while freeing sub-ELF descriptor: %s\n",
150 elf_errmsg (-1));
151 exit (1);
152 }
153 }
154
155 /* When we reach this point we haven't found the given file in the
156 archive. */
157 printf ("File `%s' not found in archive\n", argv[2]);
158 exit (1);
159 }
160