]>
Commit | Line | Data |
---|---|---|
c906108c SS |
1 | /* Program to stuff files into a specially prepared space in kdb. |
2 | Copyright (C) 1986, 1989, 1991 Free Software Foundation, Inc. | |
3 | ||
c5aa993b | 4 | This file is part of GDB. |
c906108c | 5 | |
c5aa993b JM |
6 | This program is free software; you can redistribute it and/or modify |
7 | it under the terms of the GNU General Public License as published by | |
8 | the Free Software Foundation; either version 2 of the License, or | |
9 | (at your option) any later version. | |
c906108c | 10 | |
c5aa993b JM |
11 | This program is distributed in the hope that it will be useful, |
12 | but WITHOUT ANY WARRANTY; without even the implied warranty of | |
13 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
14 | GNU General Public License for more details. | |
c906108c | 15 | |
c5aa993b JM |
16 | You should have received a copy of the GNU General Public License |
17 | along with this program; if not, write to the Free Software | |
18 | Foundation, Inc., 59 Temple Place - Suite 330, | |
19 | Boston, MA 02111-1307, USA. */ | |
c906108c SS |
20 | |
21 | /* Written 13-Mar-86 by David Bridgham. */ | |
22 | ||
23 | #include <stdio.h> | |
24 | #include <a.out.h> | |
25 | #include <sys/types.h> | |
26 | #include "gdb_stat.h" | |
27 | #include <sys/file.h> | |
28 | #include <varargs.h> | |
29 | ||
30 | main (argc, argv) | |
31 | int argc; | |
32 | char *argv[]; | |
33 | { | |
34 | register char *cp; | |
35 | char *outfile; | |
36 | register int i; | |
37 | int offset; | |
38 | int out_fd, in_fd; | |
39 | struct stat stat_buf; | |
40 | int size, pad; | |
41 | char buf[1024]; | |
c5aa993b JM |
42 | static char zeros[4] = |
43 | {0}; | |
c906108c SS |
44 | |
45 | if (argc < 4) | |
c5aa993b JM |
46 | err ("Not enough arguments\nUsage: %s -o kdb file1 file2 ...\n", |
47 | argv[0]); | |
c906108c SS |
48 | |
49 | outfile = 0; | |
50 | for (i = 1; i < argc; i++) | |
51 | { | |
52 | if (STREQ (argv[i], "-o")) | |
53 | outfile = argv[++i]; | |
54 | } | |
55 | if (outfile == 0) | |
c5aa993b | 56 | err ("Output file not specified\n"); |
c906108c SS |
57 | |
58 | offset = get_offset (outfile, "_heap"); | |
59 | ||
60 | out_fd = open (outfile, O_WRONLY); | |
61 | if (out_fd < 0) | |
62 | err ("Error opening %s for write: %s\n", outfile, strerror (errno)); | |
63 | if (lseek (out_fd, offset, 0) < 0) | |
64 | err ("Error seeking to heap in %s: %s\n", outfile, strerror (errno)); | |
65 | ||
66 | /* For each file listed on the command line, write it into the | |
67 | * 'heap' of the output file. Make sure to skip the arguments | |
68 | * that name the output file. */ | |
69 | for (i = 1; i < argc; i++) | |
70 | { | |
71 | if (STREQ (argv[i], "-o")) | |
72 | continue; | |
73 | if ((in_fd = open (argv[i], O_RDONLY)) < 0) | |
74 | err ("Error opening %s for read: %s\n", argv[i], | |
75 | strerror (errno)); | |
76 | if (fstat (in_fd, &stat_buf) < 0) | |
77 | err ("Error stat'ing %s: %s\n", argv[i], strerror (errno)); | |
78 | size = strlen (argv[i]); | |
79 | pad = 4 - (size & 3); | |
80 | size += pad + stat_buf.st_size + sizeof (int); | |
81 | write (out_fd, &size, sizeof (int)); | |
82 | write (out_fd, argv[i], strlen (argv[i])); | |
83 | write (out_fd, zeros, pad); | |
84 | while ((size = read (in_fd, buf, sizeof (buf))) > 0) | |
85 | write (out_fd, buf, size); | |
86 | close (in_fd); | |
87 | } | |
88 | size = 0; | |
89 | write (out_fd, &size, sizeof (int)); | |
90 | close (out_fd); | |
91 | return (0); | |
92 | } | |
93 | ||
94 | /* Read symbol table from file and returns the offset into the file | |
95 | * where symbol sym_name is located. If error, print message and | |
96 | * exit. */ | |
fba45db2 | 97 | get_offset (char *file, char *sym_name) |
c906108c SS |
98 | { |
99 | int f; | |
100 | struct exec file_hdr; | |
101 | struct nlist *symbol_table; | |
102 | int size; | |
103 | char *strings; | |
104 | ||
105 | f = open (file, O_RDONLY); | |
106 | if (f < 0) | |
107 | err ("Error opening %s: %s\n", file, strerror (errno)); | |
108 | if (read (f, &file_hdr, sizeof (file_hdr)) < 0) | |
109 | err ("Error reading exec structure: %s\n", strerror (errno)); | |
110 | if (N_BADMAG (file_hdr)) | |
111 | err ("File %s not an a.out file\n", file); | |
112 | ||
113 | /* read in symbol table */ | |
c5aa993b | 114 | if ((symbol_table = (struct nlist *) malloc (file_hdr.a_syms)) == 0) |
c906108c SS |
115 | err ("Couldn't allocate space for symbol table\n"); |
116 | if (lseek (f, N_SYMOFF (file_hdr), 0) == -1) | |
117 | err ("lseek error: %s\n", strerror (errno)); | |
118 | if (read (f, symbol_table, file_hdr.a_syms) == -1) | |
119 | err ("Error reading symbol table from %s: %s\n", file, | |
120 | strerror (errno)); | |
121 | ||
122 | /* read in string table */ | |
123 | if (read (f, &size, 4) == -1) | |
124 | err ("reading string table size: %s\n", strerror (errno)); | |
c5aa993b | 125 | if ((strings = (char *) malloc (size)) == 0) |
c906108c SS |
126 | err ("Couldn't allocate memory for string table\n"); |
127 | if (read (f, strings, size - 4) == -1) | |
128 | err ("reading string table: %s\n", strerror (errno)); | |
129 | ||
130 | /* Find the core address at which the first byte of kdb text segment | |
131 | should be loaded into core when kdb is run. */ | |
132 | origin = find_symbol ("_etext", symbol_table, file_hdr.a_syms, strings) | |
133 | - file_hdr.a_text; | |
134 | /* Find the core address at which the heap will appear. */ | |
135 | coreaddr = find_symbol (sym_name, symbol_table, file_hdr.a_syms, strings); | |
136 | /* Return address in file of the heap data space. */ | |
137 | return (N_TXTOFF (file_hdr) + core_addr - origin); | |
138 | } | |
139 | ||
fba45db2 KB |
140 | find_symbol (char *sym_name, struct nlist *symbol_table, int length, |
141 | char *strings) | |
c906108c SS |
142 | { |
143 | register struct nlist *sym; | |
144 | ||
145 | /* Find symbol in question */ | |
146 | for (sym = symbol_table; | |
c5aa993b | 147 | sym != (struct nlist *) ((char *) symbol_table + length); |
c906108c | 148 | sym++) |
c5aa993b JM |
149 | { |
150 | if ((sym->n_type & N_TYPE) != N_DATA) | |
151 | continue; | |
152 | if (sym->n_un.n_strx == 0) | |
153 | continue; | |
154 | if (STREQ (sym_name, strings + sym->n_un.n_strx - 4)) | |
155 | return sym->n_value; | |
156 | } | |
157 | err ("Data symbol %s not found in %s\n", sym_name, file); | |
c906108c SS |
158 | } |
159 | ||
160 | /* VARARGS */ | |
161 | void | |
162 | err (va_alist) | |
163 | va_dcl | |
164 | { | |
165 | va_list args; | |
166 | char *string; | |
167 | ||
168 | va_start (args); | |
169 | string = va_arg (args, char *); | |
170 | vfprintf (gdb_stderr, string, args); | |
171 | va_end (args); | |
172 | exit (-1); | |
173 | } |