54844
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
1 /* simulate sbrk() with an array in .bss, for unexec() support for Cygwin;
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
2 complete rewrite of xemacs Cygwin unexec() code
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
3
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
4 Copyright (C) 2004
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
5 Free Software Foundation, Inc.
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
6
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
7 This file is part of GNU Emacs.
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
8
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
9 GNU Emacs is free software; you can redistribute it and/or modify
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
10 it under the terms of the GNU General Public License as published by
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
11 the Free Software Foundation; either version 2, or (at your option)
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
12 any later version.
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
13
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
14 GNU Emacs is distributed in the hope that it will be useful,
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
15 but WITHOUT ANY WARRANTY; without even the implied warranty of
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
16 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
17 GNU General Public License for more details.
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
18
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
19 You should have received a copy of the GNU General Public License
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
20 along with GNU Emacs; see the file COPYING. If not, write to
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
21 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
22 Boston, MA 02111-1307, USA. */
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
23
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
24 #include <config.h>
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
25 #include <stdio.h>
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
26 #include "lisp.h"
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
27
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
28 #include <unistd.h>
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
29
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
30 #ifdef HAVE_X_WINDOWS
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
31 #define STATIC_HEAP_SIZE (7 * 1024 * 1024)
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
32 #else
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
33 #define STATIC_HEAP_SIZE (7 * 1024 * 1024)
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
34 #endif
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
35
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
36 int debug_sheap = 0;
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
37
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
38 #define BLOCKSIZE 4096
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
39
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
40 char bss_sbrk_buffer[STATIC_HEAP_SIZE];
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
41 char *bss_sbrk_ptr;
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
42 int bss_sbrk_did_unexec;
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
43
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
44 void *
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
45 bss_sbrk (ptrdiff_t request_size)
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
46 {
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
47 if (!bss_sbrk_ptr)
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
48 {
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
49 bss_sbrk_ptr = bss_sbrk_buffer;
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
50 #ifdef CYGWIN
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
51 sbrk (BLOCKSIZE); /* force space for fork to work */
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
52 #endif
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
53 }
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
54
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
55 if (!(int) request_size)
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
56 {
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
57 return (bss_sbrk_ptr);
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
58 }
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
59 else if (bss_sbrk_ptr + (int) request_size < bss_sbrk_buffer)
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
60 {
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
61 printf
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
62 ("attempt to free too much: avail %d used %d failed request %d\n",
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
63 STATIC_HEAP_SIZE, bss_sbrk_ptr - bss_sbrk_buffer,
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
64 (int) request_size);
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
65 exit (-1);
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
66 return 0;
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
67 }
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
68 else if (bss_sbrk_ptr + (int) request_size >
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
69 bss_sbrk_buffer + STATIC_HEAP_SIZE)
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
70 {
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
71 printf ("static heap exhausted: avail %d used %d failed request %d\n",
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
72 STATIC_HEAP_SIZE,
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
73 bss_sbrk_ptr - bss_sbrk_buffer, (int) request_size);
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
74 exit (-1);
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
75 return 0;
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
76 }
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
77 else if ((int) request_size < 0)
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
78 {
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
79 bss_sbrk_ptr += (int) request_size;
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
80 if (debug_sheap)
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
81 printf ("freed size %d\n", request_size);
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
82 return bss_sbrk_ptr;
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
83 }
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
84 else
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
85 {
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
86 char *ret = bss_sbrk_ptr;
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
87 if (debug_sheap)
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
88 printf ("allocated 0x%08x size %d\n", ret, request_size);
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
89 bss_sbrk_ptr += (int) request_size;
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
90 return ret;
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
91 }
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
92 }
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
93
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
94 void
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
95 report_sheap_usage (int die_if_pure_storage_exceeded)
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
96 {
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
97 char buf[200];
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
98 sprintf (buf, "Static heap usage: %d of %d bytes",
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
99 bss_sbrk_ptr - bss_sbrk_buffer, STATIC_HEAP_SIZE);
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
100 message ("%s", buf);
|
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
diff
changeset
|
101 }
|