annotate src/unexhp9k800.c @ 273:411aac7253ef

Added tag r21-0b34 for changeset c5d627a313b1
author cvs
date Mon, 13 Aug 2007 10:28:54 +0200
parents c5d627a313b1
children 7df0dd720c89
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
1 /* Unexec for HP 9000 Series 800 machines.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
2 Bob Desinger <hpsemc!bd@hplabs.hp.com>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
3
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
4 This file is part of XEmacs.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
5
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
6 XEmacs is free software; you can redistribute it and/or modify it
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
7 under the terms of the GNU General Public License as published by the
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
8 Free Software Foundation; either version 2, or (at your option) any
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
9 later version.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
10
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
11 XEmacs is distributed in the hope that it will be useful, but WITHOUT
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
12 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
13 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
14 for more details.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
15
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
16 You should have received a copy of the GNU General Public License
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
17 along with XEmacs; see the file COPYING. If not, write to
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
18 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
19 Boston, MA 02111-1307, USA. */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
20
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
21 /* Synched up with: Not synched with FSF. */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
22
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
23 /*
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
24
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
25 Unexec creates a copy of the old a.out file, and replaces the old data
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
26 area with the current data area. When the new file is executed, the
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
27 process will see the same data structures and data values that the
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
28 original process had when unexec was called.
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
29
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
30 Unlike other versions of unexec, this one copies symbol table and
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
31 debug information to the new a.out file. Thus, the new a.out file
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
32 may be debugged with symbolic debuggers.
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
33
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
34 If you fix any bugs in this, I'd like to incorporate your fixes.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
35 Send them to uunet!hpda!hpsemc!jmorris or jmorris%hpsemc@hplabs.HP.COM.
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
36
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
37 CAVEATS:
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
38 This routine saves the current value of all static and external
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
39 variables. This means that any data structure that needs to be
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
40 initialized must be explicitly reset. Variables will not have their
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
41 expected default values.
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
42
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
43 Unfortunately, the HP-UX signal handler has internal initialization
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
44 flags which are not explicitly reset. Thus, for signals to work in
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
45 conjunction with this routine, the following code must executed when
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
46 the new process starts up.
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
47
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
48 void _sigreturn();
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
49 ...
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
50 sigsetreturn(_sigreturn);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
51 */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
52
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
53
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
54 #include <config.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
55 #include <stdio.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
56 #include <fcntl.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
57 #include <errno.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
58
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
59 #include <a.out.h>
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
60
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
61 /*
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
62 * Minor modification to enable dumping with shared libraries added by
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
63 * Dipankar Gupta (dg@hplb.hpl.hp.com). I studied Oliver Laumann's
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
64 * more elaborate dynamic loading scheme in ELK while implementing
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
65 * this, but don't use any of his machinery.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
66 *
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
67 * Stores the BRK value at dump time, and uses the RUN_TIME_REMAP hook
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
68 * to break back to the stored value when the dumped executable is restarted.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
69 *
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
70 * CAVEATS (addenda):
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
71 * 1. Text area of the shlibs are not stored. Thus, if a shared library is
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
72 * replaced between the time of dump and execution, all bets are off.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
73 *
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
74 * 2. Assumes that the data and bss area are adjacent, which is true of the
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
75 * current VM implementation.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
76 *
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
77 * 3. Any setup that defines HPUX_USE_SHLIBS *must* also define
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
78 * RUN_TIME_REMAP.
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
79 */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
80
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
81 #ifdef HPUX_USE_SHLIBS
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
82 #include <dl.h> /* User-space dynamic loader entry points */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
83 void Save_Shared_Data();
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
84 int run_time_remap();
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
85 #endif
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
86
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
87 #define roundup(x,n) ( ( (x)+(n-1) ) & ~(n-1) ) /* n is power of 2 */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
88 #define min(x,y) ( ((x)<(y))?(x):(y) )
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
89
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
90 void write_header(int file, struct header *hdr, struct som_exec_auxhdr *auxhdr);
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
91 void read_header (int file, struct header *hdr, struct som_exec_auxhdr *auxhdr);
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
92 void save_data_space (int file, struct header *hdr,
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
93 struct som_exec_auxhdr *auxhdr, int size);
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
94 void copy_rest (int old, int new);
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
95 void copy_file (int old, int new, int size);
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
96 void update_file_ptrs(int file, struct header *hdr,
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
97 struct som_exec_auxhdr *auxhdr,
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
98 unsigned int location, int offset);
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
99
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
100 /* Create a new a.out file, same as old but with current data space */
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
101 int
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
102 unexec(char new_name[], /* name of the new a.out file to be created */
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
103 char old_name[], /* name of the old a.out file */
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
104 char *new_end_of_text, /* ptr to new edata/etext; NOT USED YET */
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
105 int dummy1, int dummy2) /* not used by emacs */
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
106 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
107 int old, new;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
108 int old_size, new_size;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
109 struct header hdr;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
110 struct som_exec_auxhdr auxhdr;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
111 long i;
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
112
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
113 /* For the greatest flexibility, should create a temporary file in
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
114 the same directory as the new file. When everything is complete,
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
115 rename the temp file to the new name.
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
116 This way, a program could update its own a.out file even while
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
117 it is still executing. If problems occur, everything is still
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
118 intact. NOT implemented. */
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
119
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
120 /* Open the input and output a.out files */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
121 old = open (old_name, O_RDONLY);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
122 if (old < 0)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
123 { perror(old_name); exit(1); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
124 new = open (new_name, O_CREAT|O_RDWR|O_TRUNC, 0777);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
125 if (new < 0)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
126 { perror(new_name); exit(1); }
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
127
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
128 /* Read the old headers */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
129 read_header(old, &hdr, &auxhdr);
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
130
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
131 #ifdef HPUX_USE_SHLIBS
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
132 Save_Shared_Data(); /* Save break value (added: dg@hplb.hpl.hp.com) */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
133 #endif
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
134 /* Decide how large the new and old data areas are */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
135 old_size = auxhdr.exec_dsize;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
136 /* I suspect these two statements are separate
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
137 to avoid a compiler bug in hpux version 8. */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
138 i = (long) sbrk (0);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
139 new_size = i - auxhdr.exec_dmem;
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
140
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
141 /* Copy the old file to the new, up to the data space */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
142 lseek(old, 0, 0);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
143 copy_file(old, new, auxhdr.exec_dfile);
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
144
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
145 /* Skip the old data segment and write a new one */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
146 lseek(old, old_size, 1);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
147 save_data_space(new, &hdr, &auxhdr, new_size);
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
148
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
149 /* Copy the rest of the file */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
150 copy_rest(old, new);
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
151
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
152 /* Update file pointers since we probably changed size of data area */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
153 update_file_ptrs(new, &hdr, &auxhdr, auxhdr.exec_dfile, new_size-old_size);
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
154
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
155 /* Save the modified header */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
156 write_header(new, &hdr, &auxhdr);
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
157
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
158 /* Close the binary file */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
159 close (old);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
160 close (new);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
161 return 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
162 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
163
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
164 /* Save current data space in the file, update header. */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
165
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
166 void
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
167 save_data_space (int file, struct header *hdr,
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
168 struct som_exec_auxhdr *auxhdr, int size)
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
169 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
170 /* Write the entire data space out to the file */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
171 if (write(file, (void *)auxhdr->exec_dmem, size) != size)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
172 { perror("Can't save new data space"); exit(1); }
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
173
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
174 /* Update the header to reflect the new data size */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
175 auxhdr->exec_dsize = size;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
176 auxhdr->exec_bsize = 0;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
177 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
178
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
179 /* Update the values of file pointers when something is inserted. */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
180
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
181 void
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
182 update_file_ptrs(int file, struct header *hdr,
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
183 struct som_exec_auxhdr *auxhdr,
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
184 unsigned int location, int offset)
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
185 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
186 struct subspace_dictionary_record subspace;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
187 int i;
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
188
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
189 /* Increase the overall size of the module */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
190 hdr->som_length += offset;
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
191
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
192 /* Update the various file pointers in the header */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
193 #define update(ptr) if (ptr > location) ptr = ptr + offset
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
194 update(hdr->aux_header_location);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
195 update(hdr->space_strings_location);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
196 update(hdr->init_array_location);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
197 update(hdr->compiler_location);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
198 update(hdr->symbol_location);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
199 update(hdr->fixup_request_location);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
200 update(hdr->symbol_strings_location);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
201 update(hdr->unloadable_sp_location);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
202 update(auxhdr->exec_tfile);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
203 update(auxhdr->exec_dfile);
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
204
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
205 /* Do for each subspace dictionary entry */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
206 lseek(file, hdr->subspace_location, 0);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
207 for (i = 0; i < hdr->subspace_total; i++)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
208 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
209 if (read(file, &subspace, sizeof(subspace)) != sizeof(subspace))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
210 { perror("Can't read subspace record"); exit(1); }
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
211
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
212 /* If subspace has a file location, update it */
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
213 if (subspace.initialization_length > 0
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
214 && subspace.file_loc_init_value > location)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
215 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
216 subspace.file_loc_init_value += offset;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
217 lseek(file, -sizeof(subspace), 1);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
218 if (write(file, &subspace, sizeof(subspace)) != sizeof(subspace))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
219 { perror("Can't update subspace record"); exit(1); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
220 }
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
221 }
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
222
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
223 /* Do for each initialization pointer record */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
224 /* (I don't think it applies to executable files, only relocatables) */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
225 #undef update
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
226 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
227
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
228 /* Read in the header records from an a.out file. */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
229
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
230 void
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
231 read_header(int file, struct header *hdr, struct som_exec_auxhdr *auxhdr)
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
232 {
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
233
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
234 /* Read the header in */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
235 lseek(file, 0, 0);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
236 if (read(file, hdr, sizeof(*hdr)) != sizeof(*hdr))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
237 { perror("Couldn't read header from a.out file"); exit(1); }
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
238
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
239 if (hdr->a_magic != EXEC_MAGIC && hdr->a_magic != SHARE_MAGIC
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
240 && hdr->a_magic != DEMAND_MAGIC)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
241 {
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
242 fprintf(stderr, "a.out file doesn't have legal magic number\n");
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
243 exit(1);
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
244 }
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
245
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
246 lseek(file, hdr->aux_header_location, 0);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
247 if (read(file, auxhdr, sizeof(*auxhdr)) != sizeof(*auxhdr))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
248 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
249 perror("Couldn't read auxiliary header from a.out file");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
250 exit(1);
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
251 }
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
252 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
253
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
254 /* Write out the header records into an a.out file. */
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
255 void
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
256 write_header(int file, struct header *hdr, struct som_exec_auxhdr *auxhdr)
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
257 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
258 /* Update the checksum */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
259 hdr->checksum = calculate_checksum(hdr);
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
260
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
261 /* Write the header back into the a.out file */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
262 lseek(file, 0, 0);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
263 if (write(file, hdr, sizeof(*hdr)) != sizeof(*hdr))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
264 { perror("Couldn't write header to a.out file"); exit(1); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
265 lseek(file, hdr->aux_header_location, 0);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
266 if (write(file, auxhdr, sizeof(*auxhdr)) != sizeof(*auxhdr))
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
267 { perror("Couldn't write auxiliary header to a.out file"); exit(1); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
268 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
269
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
270 /* Calculate the checksum of a SOM header record. */
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
271 int
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
272 calculate_checksum(struct header *hdr)
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
273 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
274 int checksum, i, *ptr;
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
275
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
276 checksum = 0; ptr = (int *) hdr;
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
277
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
278 for (i=0; i<sizeof(*hdr)/sizeof(int)-1; i++)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
279 checksum ^= ptr[i];
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
280
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
281 return(checksum);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
282 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
283
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
284 /* Copy size bytes from the old file to the new one. */
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
285 void
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
286 copy_file (int old, int new, int size)
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
287 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
288 int len;
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
289 int buffer[8192]; /* word aligned will be faster */
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
290
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
291 for (; size > 0; size -= len)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
292 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
293 len = min(size, sizeof(buffer));
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
294 if (read(old, buffer, len) != len)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
295 { perror("Read failure on a.out file"); exit(1); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
296 if (write(new, buffer, len) != len)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
297 { perror("Write failure in a.out file"); exit(1); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
298 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
299 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
300
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
301 /* Copy the rest of the file, up to EOF. */
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
302 void
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
303 copy_rest (int old, int new)
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
304 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
305 int buffer[4096];
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
306 int len;
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
307
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
308 /* Copy bytes until end of file or error */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
309 while ( (len = read(old, buffer, sizeof(buffer))) > 0)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
310 if (write(new, buffer, len) != len) break;
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
311
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
312 if (len != 0)
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
313 { perror("Unable to copy the rest of the file"); exit(1); }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
314 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
315
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
316 #ifdef DEBUG
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
317 display_header(struct header *hdr, struct som_exec_auxhdr *auxhdr)
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
318 {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
319 /* Display the header information (debug) */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
320 printf("\n\nFILE HEADER\n");
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
321 printf("magic number %d \n", hdr->a_magic);
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
322 printf("text loc %.8x size %d \n", auxhdr->exec_tmem, auxhdr->exec_tsize);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
323 printf("data loc %.8x size %d \n", auxhdr->exec_dmem, auxhdr->exec_dsize);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
324 printf("entry %x \n", auxhdr->exec_entry);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
325 printf("Bss segment size %u\n", auxhdr->exec_bsize);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
326 printf("\n");
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
327 printf("data file loc %d size %d\n",
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
328 auxhdr->exec_dfile, auxhdr->exec_dsize);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
329 printf("som_length %d\n", hdr->som_length);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
330 printf("unloadable sploc %d size %d\n",
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
331 hdr->unloadable_sp_location, hdr->unloadable_sp_size);
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
332 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
333 #endif /* DEBUG */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
334
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
335 #ifdef HPUX_USE_SHLIBS
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
336 /* Added machinery for shared libs... see comments at the beginning of this file. */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
337
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
338 void *Brk_On_Dump = 0; /* Brk value to restore... stored as a global */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
339
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
340 void Save_Shared_Data () {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
341 Brk_On_Dump = sbrk( 0 );
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
342 }
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
343
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
344 void Restore_Shared_Data () {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
345 brk ( Brk_On_Dump );
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
346 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
347
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
348 int run_time_remap (int d) {
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
349 Restore_Shared_Data();
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
350 }
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
351
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
352 /* run_time_remap is the magic called by startup code in the dumped executable
272
c5d627a313b1 Import from CVS: tag r21-0b34
cvs
parents: 0
diff changeset
353 * if RUN_TIME_REMAP is set.
0
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
354 */
376386a54a3c Import from CVS: tag r19-14
cvs
parents:
diff changeset
355 #endif /* HPUX_USE_SHLIBS */