comparison src/unexnt.c @ 100:4be1180a9e89 r20-1b2

Import from CVS: tag r20-1b2
author cvs
date Mon, 13 Aug 2007 09:15:11 +0200
parents
children 15872534500d
comparison
equal deleted inserted replaced
99:2d83cbd90d8d 100:4be1180a9e89
1 /* unexec for GNU Emacs on Windows NT.
2 Copyright (C) 1994 Free Software Foundation, Inc.
3
4 This file is part of XEmacs.
5
6 XEmacs is free software; you can redistribute it and/or modify it
7 under the terms of the GNU General Public License as published by the
8 Free Software Foundation; either version 2, or (at your option) any
9 later version.
10
11 XEmacs is distributed in the hope that it will be useful, but WITHOUT
12 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14 for more details.
15
16 You should have received a copy of the GNU General Public License
17 along with XEmacs; see the file COPYING. If not, write to the Free
18 Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
19 02111-1307, USA.
20
21 Geoff Voelker (voelker@cs.washington.edu) 8-12-94 */
22
23 /* Adapted for XEmacs by David Hobley <david@spook-le0.cia.com.au> */
24
25 #include <stdlib.h> /* _fmode */
26 #include <stdio.h>
27 #include <fcntl.h>
28 #include <windows.h>
29
30 #if 0
31 extern BOOL ctrl_c_handler (unsigned long type);
32 #endif
33
34 #include "ntheap.h"
35
36 /* A convenient type for keeping all the info about a mapped file together. */
37 typedef struct file_data {
38 char *name;
39 unsigned long size;
40 HANDLE file;
41 HANDLE file_mapping;
42 unsigned char *file_base;
43 } file_data;
44
45 /* Basically, our "initialized" flag. */
46 BOOL need_to_recreate_heap = FALSE;
47
48 /* So we can find our heap in the file to recreate it. */
49 unsigned long heap_index_in_executable = 0;
50
51 void open_input_file (file_data *p_file, char *name);
52 void open_output_file (file_data *p_file, char *name, unsigned long size);
53 void close_file_data (file_data *p_file);
54
55 void get_section_info (file_data *p_file);
56 void copy_executable_and_dump_data_section (file_data *, file_data *);
57 void dump_bss_and_heap (file_data *p_infile, file_data *p_outfile);
58
59 /* Cached info about the .data section in the executable. */
60 PUCHAR data_start_va = 0;
61 DWORD data_start_file = 0;
62 DWORD data_size = 0;
63
64 /* Cached info about the .bss section in the executable. */
65 PUCHAR bss_start = 0;
66 DWORD bss_size = 0;
67
68 #ifdef HAVE_NTGUI
69 HINSTANCE hinst = NULL;
70 HINSTANCE hprevinst = NULL;
71 LPSTR lpCmdLine = "";
72 int nCmdShow = 0;
73 #endif /* HAVE_NTGUI */
74
75 /* Startup code for running on NT. When we are running as the dumped
76 version, we need to bootstrap our heap and .bss section into our
77 address space before we can actually hand off control to the startup
78 code supplied by NT (primarily because that code relies upon malloc ()). */
79 void
80 _start (void)
81 {
82 extern void mainCRTStartup (void);
83
84 /* Cache system info, e.g., the NT page size. */
85 cache_system_info ();
86
87 /* If we're a dumped version of emacs then we need to recreate
88 our heap and play tricks with our .bss section. Do this before
89 start up. (WARNING: Do not put any code before this section
90 that relies upon malloc () and runs in the dumped version. It
91 won't work.) */
92 if (need_to_recreate_heap)
93 {
94 char executable_path[MAX_PATH];
95
96 if (GetModuleFileName (NULL, executable_path, MAX_PATH) == 0)
97 {
98 exit (1);
99 }
100 recreate_heap (executable_path);
101 need_to_recreate_heap = FALSE;
102 }
103
104 /* The default behavior is to treat files as binary and patch up
105 text files appropriately, in accordance with the MSDOS code. */
106 _fmode = O_BINARY;
107
108 #if 0
109 /* This prevents ctrl-c's in shells running while we're suspended from
110 having us exit. */
111 SetConsoleCtrlHandler ((PHANDLER_ROUTINE) ctrl_c_handler, TRUE);
112 #endif
113
114 /* Invoke the NT CRT startup routine now that our housecleaning
115 is finished. */
116 #ifdef HAVE_NTGUI
117 /* determine WinMain args like crt0.c does */
118 hinst = GetModuleHandle(NULL);
119 lpCmdLine = GetCommandLine();
120 nCmdShow = SW_SHOWDEFAULT;
121 #endif
122 mainCRTStartup ();
123 }
124
125 /* Dump out .data and .bss sections into a new executable. */
126 void
127 unexec (char *new_name, char *old_name, void *start_data, void *start_bss,
128 void *entry_address)
129 {
130 file_data in_file, out_file;
131 char out_filename[MAX_PATH], in_filename[MAX_PATH];
132 unsigned long size;
133 char *ptr;
134
135 /* Make sure that the input and output filenames have the
136 ".exe" extension...patch them up if they don't. */
137 strcpy (in_filename, old_name);
138 ptr = in_filename + strlen (in_filename) - 4;
139 if (strcmp (ptr, ".exe"))
140 strcat (in_filename, ".exe");
141
142 strcpy (out_filename, new_name);
143 ptr = out_filename + strlen (out_filename) - 4;
144 if (strcmp (ptr, ".exe"))
145 strcat (out_filename, ".exe");
146
147 printf ("Dumping from %s\n", in_filename);
148 printf (" to %s\n", out_filename);
149
150 /* We need to round off our heap to NT's allocation unit (64KB). */
151 round_heap (get_allocation_unit ());
152
153 /* Open the undumped executable file. */
154 open_input_file (&in_file, in_filename);
155
156 /* Get the interesting section info, like start and size of .bss... */
157 get_section_info (&in_file);
158
159 /* The size of the dumped executable is the size of the original
160 executable plus the size of the heap and the size of the .bss section. */
161 heap_index_in_executable = (unsigned long)
162 round_to_next ((unsigned char *) in_file.size, get_allocation_unit ());
163 size = heap_index_in_executable + get_committed_heap_size () + bss_size;
164 open_output_file (&out_file, out_filename, size);
165
166 /* Set the flag (before dumping). */
167 need_to_recreate_heap = TRUE;
168
169 copy_executable_and_dump_data_section (&in_file, &out_file);
170 dump_bss_and_heap (&in_file, &out_file);
171
172 close_file_data (&in_file);
173 close_file_data (&out_file);
174 }
175
176
177 /* File handling. */
178
179
180 void
181 open_input_file (file_data *p_file, char *filename)
182 {
183 HANDLE file;
184 HANDLE file_mapping;
185 void *file_base;
186 unsigned long size, upper_size;
187
188 file = CreateFile (filename, GENERIC_READ, FILE_SHARE_READ, NULL,
189 OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
190 if (file == INVALID_HANDLE_VALUE)
191 {
192 printf ("Failed to open %s (%d)...bailing.\n",
193 filename, GetLastError ());
194 exit (1);
195 }
196
197 size = GetFileSize (file, &upper_size);
198 file_mapping = CreateFileMapping (file, NULL, PAGE_READONLY,
199 0, size, NULL);
200 if (!file_mapping)
201 {
202 printf ("Failed to create file mapping of %s (%d)...bailing.\n",
203 filename, GetLastError ());
204 exit (1);
205 }
206
207 file_base = MapViewOfFile (file_mapping, FILE_MAP_READ, 0, 0, size);
208 if (file_base == 0)
209 {
210 printf ("Failed to map view of file of %s (%d)...bailing.\n",
211 filename, GetLastError ());
212 exit (1);
213 }
214
215 p_file->name = filename;
216 p_file->size = size;
217 p_file->file = file;
218 p_file->file_mapping = file_mapping;
219 p_file->file_base = file_base;
220 }
221
222 void
223 open_output_file (file_data *p_file, char *filename, unsigned long size)
224 {
225 HANDLE file;
226 HANDLE file_mapping;
227 void *file_base;
228 int i;
229
230 file = CreateFile (filename, GENERIC_READ | GENERIC_WRITE, 0, NULL,
231 CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0);
232 if (file == INVALID_HANDLE_VALUE)
233 {
234 i = GetLastError ();
235 printf ("open_output_file: Failed to open %s (%d).\n",
236 filename, i);
237 exit (1);
238 }
239
240 file_mapping = CreateFileMapping (file, NULL, PAGE_READWRITE,
241 0, size, NULL);
242 if (!file_mapping)
243 {
244 i = GetLastError ();
245 printf ("open_output_file: Failed to create file mapping of %s (%d).\n",
246 filename, i);
247 exit (1);
248 }
249
250 file_base = MapViewOfFile (file_mapping, FILE_MAP_WRITE, 0, 0, size);
251 if (file_base == 0)
252 {
253 i = GetLastError ();
254 printf ("open_output_file: Failed to map view of file of %s (%d).\n",
255 filename, i);
256 exit (1);
257 }
258
259 p_file->name = filename;
260 p_file->size = size;
261 p_file->file = file;
262 p_file->file_mapping = file_mapping;
263 p_file->file_base = file_base;
264 }
265
266 /* Close the system structures associated with the given file. */
267 static void
268 close_file_data (file_data *p_file)
269 {
270 UnmapViewOfFile (p_file->file_base);
271 CloseHandle (p_file->file_mapping);
272 CloseHandle (p_file->file);
273 }
274
275
276 /* Routines to manipulate NT executable file sections. */
277
278 static void
279 get_bss_info_from_map_file (file_data *p_infile, PUCHAR *p_bss_start,
280 DWORD *p_bss_size)
281 {
282 int n, start, len;
283 char map_filename[MAX_PATH];
284 char buffer[256];
285 FILE *map;
286
287 /* Overwrite the .exe extension on the executable file name with
288 the .map extension. */
289 strcpy (map_filename, p_infile->name);
290 n = strlen (map_filename) - 3;
291 strcpy (&map_filename[n], "map");
292
293 map = fopen (map_filename, "r");
294 if (!map)
295 {
296 printf ("Failed to open map file %s, error %d...bailing out.\n",
297 map_filename, GetLastError ());
298 exit (-1);
299 }
300
301 while (fgets (buffer, sizeof (buffer), map))
302 {
303 if (!(strstr (buffer, ".bss") && strstr (buffer, "DATA")))
304 continue;
305 n = sscanf (buffer, " %*d:%x %x", &start, &len);
306 if (n != 2)
307 {
308 printf ("Failed to scan the .bss section line:\n%s", buffer);
309 exit (-1);
310 }
311 break;
312 }
313 *p_bss_start = (PUCHAR) start;
314 *p_bss_size = (DWORD) len;
315 }
316
317 static unsigned long
318 get_section_size (PIMAGE_SECTION_HEADER p_section)
319 {
320 /* The section size is in different locations in the different versions. */
321 switch (get_nt_minor_version ())
322 {
323 case 10:
324 return p_section->SizeOfRawData;
325 default:
326 return p_section->Misc.VirtualSize;
327 }
328 }
329
330 /* Flip through the executable and cache the info necessary for dumping. */
331 static void
332 get_section_info (file_data *p_infile)
333 {
334 PIMAGE_DOS_HEADER dos_header;
335 PIMAGE_NT_HEADERS nt_header;
336 PIMAGE_SECTION_HEADER section, data_section;
337 unsigned char *ptr;
338 int i;
339
340 dos_header = (PIMAGE_DOS_HEADER) p_infile->file_base;
341 if (dos_header->e_magic != IMAGE_DOS_SIGNATURE)
342 {
343 printf ("Unknown EXE header in %s...bailing.\n", p_infile->name);
344 exit (1);
345 }
346 nt_header = (PIMAGE_NT_HEADERS) (((unsigned long) dos_header) +
347 dos_header->e_lfanew);
348 if (nt_header == NULL)
349 {
350 printf ("Failed to find IMAGE_NT_HEADER in %s...bailing.\n",
351 p_infile->name);
352 exit (1);
353 }
354
355 /* Check the NT header signature ... */
356 if (nt_header->Signature != IMAGE_NT_SIGNATURE)
357 {
358 printf ("Invalid IMAGE_NT_SIGNATURE 0x%x in %s...bailing.\n",
359 nt_header->Signature, p_infile->name);
360 }
361
362 /* Flip through the sections for .data and .bss ... */
363 section = (PIMAGE_SECTION_HEADER) IMAGE_FIRST_SECTION (nt_header);
364 for (i = 0; i < nt_header->FileHeader.NumberOfSections; i++)
365 {
366 if (!strcmp (section->Name, ".bss"))
367 {
368 /* The .bss section. */
369 ptr = (char *) nt_header->OptionalHeader.ImageBase +
370 section->VirtualAddress;
371 bss_start = ptr;
372 bss_size = get_section_size (section);
373 }
374 if (!strcmp (section->Name, ".data"))
375 {
376 /* From lastfile.c */
377 extern char my_edata[];
378
379 /* The .data section. */
380 data_section = section;
381 ptr = (char *) nt_header->OptionalHeader.ImageBase +
382 section->VirtualAddress;
383 data_start_va = ptr;
384 data_start_file = section->PointerToRawData;
385
386 /* We want to only write Emacs data back to the executable,
387 not any of the library data (if library data is included,
388 then a dumped Emacs won't run on system versions other
389 than the one Emacs was dumped on). */
390 data_size = my_edata - data_start_va;
391 }
392 section++;
393 }
394
395 if (!bss_start && !bss_size)
396 {
397 /* Starting with MSVC 4.0, the .bss section has been eliminated
398 and appended virtually to the end of the .data section. Our
399 only hint about where the .bss section starts in the address
400 comes from the SizeOfRawData field in the .data section
401 header. Unfortunately, this field is only approximate, as it
402 is a rounded number and is typically rounded just beyond the
403 start of the .bss section. To find the start and size of the
404 .bss section exactly, we have to peek into the map file. */
405 get_bss_info_from_map_file (p_infile, &ptr, &bss_size);
406 bss_start = ptr + nt_header->OptionalHeader.ImageBase
407 + data_section->VirtualAddress;
408 }
409 }
410
411
412 /* The dump routines. */
413
414 static void
415 copy_executable_and_dump_data_section (file_data *p_infile,
416 file_data *p_outfile)
417 {
418 unsigned char *data_file, *data_va;
419 unsigned long size, index;
420
421 /* Get a pointer to where the raw data should go in the executable file. */
422 data_file = (char *) p_outfile->file_base + data_start_file;
423
424 /* Get a pointer to the raw data in our address space. */
425 data_va = data_start_va;
426
427 size = (DWORD) data_file - (DWORD) p_outfile->file_base;
428 printf ("Copying executable up to data section...\n");
429 printf ("\t0x%08x Offset in input file.\n", 0);
430 printf ("\t0x%08x Offset in output file.\n", 0);
431 printf ("\t0x%08x Size in bytes.\n", size);
432 memcpy (p_outfile->file_base, p_infile->file_base, size);
433
434 size = data_size;
435 printf ("Dumping .data section...\n");
436 printf ("\t0x%08x Address in process.\n", data_va);
437 printf ("\t0x%08x Offset in output file.\n",
438 data_file - p_outfile->file_base);
439 printf ("\t0x%08x Size in bytes.\n", size);
440 memcpy (data_file, data_va, size);
441
442 index = (DWORD) data_file + size - (DWORD) p_outfile->file_base;
443 size = p_infile->size - index;
444 printf ("Copying rest of executable...\n");
445 printf ("\t0x%08x Offset in input file.\n", index);
446 printf ("\t0x%08x Offset in output file.\n", index);
447 printf ("\t0x%08x Size in bytes.\n", size);
448 memcpy ((char *) p_outfile->file_base + index,
449 (char *) p_infile->file_base + index, size);
450 }
451
452 static void
453 dump_bss_and_heap (file_data *p_infile, file_data *p_outfile)
454 {
455 unsigned char *heap_data, *bss_data;
456 unsigned long size, index;
457
458 printf ("Dumping heap into executable...\n");
459
460 index = heap_index_in_executable;
461 size = get_committed_heap_size ();
462 heap_data = get_heap_start ();
463
464 printf ("\t0x%08x Heap start in process.\n", heap_data);
465 printf ("\t0x%08x Heap offset in executable.\n", index);
466 printf ("\t0x%08x Heap size in bytes.\n", size);
467
468 memcpy ((PUCHAR) p_outfile->file_base + index, heap_data, size);
469
470 printf ("Dumping .bss into executable...\n");
471
472 index += size;
473 size = bss_size;
474 bss_data = bss_start;
475
476 printf ("\t0x%08x BSS start in process.\n", bss_data);
477 printf ("\t0x%08x BSS offset in executable.\n", index);
478 printf ("\t0x%08x BSS size in bytes.\n", size);
479 memcpy ((char *) p_outfile->file_base + index, bss_data, size);
480 }
481
482
483 /* Reload and remap routines. */
484
485
486 /* Load the dumped .bss section into the .bss area of our address space. */
487 void
488 read_in_bss (char *filename)
489 {
490 HANDLE file;
491 unsigned long size, index, n_read, total_read;
492 char buffer[512], *bss;
493 int i;
494
495 file = CreateFile (filename, GENERIC_READ, FILE_SHARE_READ, NULL,
496 OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
497 if (file == INVALID_HANDLE_VALUE)
498 {
499 i = GetLastError ();
500 exit (1);
501 }
502
503 /* Seek to where the .bss section is tucked away after the heap... */
504 index = heap_index_in_executable + get_committed_heap_size ();
505 if (SetFilePointer (file, index, NULL, FILE_BEGIN) == 0xFFFFFFFF)
506 {
507 i = GetLastError ();
508 exit (1);
509 }
510
511
512 /* Ok, read in the saved .bss section and initialize all
513 uninitialized variables. */
514 if (!ReadFile (file, bss_start, bss_size, &n_read, NULL))
515 {
516 i = GetLastError ();
517 exit (1);
518 }
519
520 CloseHandle (file);
521 }
522
523 /* Map the heap dumped into the executable file into our address space. */
524 void
525 map_in_heap (char *filename)
526 {
527 HANDLE file;
528 HANDLE file_mapping;
529 void *file_base;
530 unsigned long size, upper_size, n_read;
531 int i;
532
533 file = CreateFile (filename, GENERIC_READ, FILE_SHARE_READ, NULL,
534 OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
535 if (file == INVALID_HANDLE_VALUE)
536 {
537 i = GetLastError ();
538 exit (1);
539 }
540
541 size = GetFileSize (file, &upper_size);
542 file_mapping = CreateFileMapping (file, NULL, PAGE_WRITECOPY,
543 0, size, NULL);
544 if (!file_mapping)
545 {
546 i = GetLastError ();
547 exit (1);
548 }
549
550 size = get_committed_heap_size ();
551 file_base = MapViewOfFileEx (file_mapping, FILE_MAP_COPY, 0,
552 heap_index_in_executable, size,
553 get_heap_start ());
554 if (file_base != 0)
555 {
556 return;
557 }
558
559 /* If we don't succeed with the mapping, then copy from the
560 data into the heap. */
561
562 CloseHandle (file_mapping);
563
564 if (VirtualAlloc (get_heap_start (), get_committed_heap_size (),
565 MEM_RESERVE | MEM_COMMIT, PAGE_READWRITE) == NULL)
566 {
567 i = GetLastError ();
568 exit (1);
569 }
570
571 /* Seek to the location of the heap data in the executable. */
572 i = heap_index_in_executable;
573 if (SetFilePointer (file, i, NULL, FILE_BEGIN) == 0xFFFFFFFF)
574 {
575 i = GetLastError ();
576 exit (1);
577 }
578
579 /* Read in the data. */
580 if (!ReadFile (file, get_heap_start (),
581 get_committed_heap_size (), &n_read, NULL))
582 {
583 i = GetLastError ();
584 exit (1);
585 }
586
587 CloseHandle (file);
588 }