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