Starting from the default size buffer, we try to read the memory map
[dyninst.git] / dyninstAPI_RT / src / RTheap-linux.c
1 /*
2  * See the dyninst/COPYRIGHT file for copyright information.
3  * 
4  * We provide the Paradyn Tools (below described as "Paradyn")
5  * on an AS IS basis, and do not warrant its validity or performance.
6  * We reserve the right to update, modify, or discontinue this
7  * software at any time.  We shall have no obligation to supply such
8  * updates or modifications or any other form of support to you.
9  * 
10  * By your use of Paradyn, you understand and agree that we (or any
11  * other person or entity with proprietary rights in Paradyn) are
12  * under no obligation to provide either maintenance services,
13  * update services, notices of latent defects, or correction of
14  * defects for Paradyn.
15  * 
16  * This library is free software; you can redistribute it and/or
17  * modify it under the terms of the GNU Lesser General Public
18  * License as published by the Free Software Foundation; either
19  * version 2.1 of the License, or (at your option) any later version.
20  * 
21  * This library is distributed in the hope that it will be useful,
22  * but WITHOUT ANY WARRANTY; without even the implied warranty of
23  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
24  * Lesser General Public License for more details.
25  * 
26  * You should have received a copy of the GNU Lesser General Public
27  * License along with this library; if not, write to the Free Software
28  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
29  */
30
31 /* $Id: RTheap-linux.c,v 1.9 2008/01/31 18:01:54 legendre Exp $ */
32 /* RTheap-linux.c: Linux-specific heap components */
33
34 #include <stdlib.h>
35 #include <stdio.h>
36 #include <string.h>                   /* str* */
37 #include <assert.h>
38 #include <sys/types.h>
39 #include <sys/uio.h>                  /* read() */
40 #include <sys/stat.h>                 /* open() */
41 #include <fcntl.h>                    /* open() */
42 #include <unistd.h>                   /* sbrk(), read(), mmap */
43 #include <sys/mman.h>                 /* mmap() */
44 #include "RTheap.h"
45
46 #define MAX_MAP_SIZE (1<<20)
47 #if defined(MUTATEE64)
48
49 int     DYNINSTheap_align = 4; /* heaps are word-aligned */
50
51 Address DYNINSTheap_loAddr = 0x10000; /* Bump to 64k to make SELinux happier */
52 Address DYNINSTheap_hiAddr = ~0x0;
53 #elif defined(arch_power)
54 int     DYNINSTheap_align  = 4; /* heaps are word-aligned */
55 Address DYNINSTheap_loAddr = ~(Address)0; // should be defined by getpagesize() when used.
56 Address DYNINSTheap_hiAddr = ~(Address)0;
57 #else
58 int     DYNINSTheap_align = 4; /* heaps are word-aligned */
59
60 Address DYNINSTheap_loAddr = 0x50000000;
61 Address DYNINSTheap_hiAddr = 0xb0000000;
62 #endif
63
64 int     DYNINSTheap_mmapFlags = MAP_FIXED | MAP_PRIVATE;
65
66
67 RT_Boolean DYNINSTheap_useMalloc(void *lo, void *hi)
68 {
69   /* We do not save footprint space by allocating in
70      the user's heap on this platform, so we stay out of it. */
71   (void)lo; /* unused parameter */
72   (void)hi; /* unused parameter */
73   return RT_FALSE;
74 }
75
76 int DYNINSTheap_mmapFdOpen(void)
77 {
78   int fd = open("/dev/zero", O_RDWR);
79   return fd;
80 }
81
82 void DYNINSTheap_mmapFdClose(int fd)
83 {
84   close(fd);
85 }
86
87 /* Linux /proc/PID/maps is unreliable when it is read with more than one
88 read call (it can show pages that are not actually allocated).  We
89 read it all in one call into this buffer.
90
91 linux-2.4: reading /proc/PID/maps now returns after each line in the maps,
92 so we must loop to get everything.
93 */
94
95 // Static so we dont have to do the exponential backoff each time.
96 static size_t mapSize = 1 << 15;
97
98 int
99 DYNINSTgetMemoryMap(unsigned *nump, dyninstmm_t **mapp)
100 {
101    int fd, done;
102    ssize_t ret;
103    size_t length;
104    char *p;
105    dyninstmm_t *ms;
106    unsigned i, num;
107    char *procAsciiMap;
108
109
110    /* 
111       Here are two lines from 'cat /proc/self/maps' on Linux 2.2.  Each
112       describes a segment of the address space.  We parse out the first
113       two addresses for the start address and length of the segment.  We
114       throw away the rest.
115
116       |SADDR-| |EADDR-|
117       0804a000-0804c000 rw-p 00001000 08:09 12089      /bin/cat
118       0804c000-0804f000 rwxp 00000000 00:00 0
119    */
120    procAsciiMap = NULL;
121    done = 0;
122    while (1)
123    {
124        if(procAsciiMap == NULL) {
125            procAsciiMap = malloc(mapSize);
126            if (!procAsciiMap) {
127                fprintf(stderr, "DYNINSTgetMemoryMap: Out of memory\n");
128                goto end;
129            }
130        }
131        else
132        {
133            void *newMap = realloc(procAsciiMap, mapSize);
134            if (!newMap) {
135                fprintf(stderr, "DYNINSTgetMemoryMap: Out of memory\n");
136                goto freeBuffer;
137            }
138            procAsciiMap = newMap;
139        }
140        fd = open("/proc/self/maps", O_RDONLY);
141        if (0 > fd) {
142            perror("open /proc");
143            goto freeBuffer;
144        }
145        length = 0;
146        while (1)
147        {
148            ret = read(fd, procAsciiMap + length, mapSize - length);
149            length += ret;
150            if (0 == ret) {
151                done = 1;
152                break;
153            }
154            if (0 > ret) {
155                close(fd);
156                perror("read /proc");
157                goto freeBuffer;
158            }
159            /* Check if the buffer was to small and exponentially
160               increase its size */
161            if (length >= mapSize) {
162                close(fd);
163                mapSize = mapSize << 1;
164
165                   /* Return error if we reached the max size for
166                      the buffer*/
167                   if(mapSize > MAX_MAP_SIZE) {
168                       fprintf(stderr, "DYNINSTgetMemoryMap: memory map buffer \
169                                        is larger than the max size. max size=%d\n", MAX_MAP_SIZE);
170                       goto freeBuffer;
171                   }
172                break;
173            }
174        }
175        if(done) {
176            break;
177        }
178    }
179    procAsciiMap[length] = '\0'; /* Now string processing works */
180
181    /* Count lines, which is the same as the number of segments.
182       Newline characters separating lines are converted to nulls. */
183    for (num = 0, p = strtok(procAsciiMap, "\n");
184         p != NULL;
185         num++, p = strtok(NULL, "\n"))
186       ;
187
188    ms = (dyninstmm_t *) malloc(num * sizeof(dyninstmm_t));
189    if (! ms) {
190       fprintf(stderr, "DYNINSTgetMemoryMap: Out of memory\n");
191       goto freeBuffer;
192    }
193
194    p = procAsciiMap;
195    for (i = 0; i < num; i++) {
196       char *next = p + strlen(p) + 1; /* start of next line */
197       Address saddr, eaddr;
198
199       /* parse start address */
200       p = strtok(p, "-");
201       if (! p) goto parseerr;
202       saddr = strtoul(p, &p, 16);
203       ++p; /* skip '-' */
204
205       /* parse end address */
206       p = strtok(NULL, " ");
207       if (! p) goto parseerr;
208       eaddr = strtoul(p, NULL, 16);
209
210       ms[i].pr_vaddr = saddr;
211       ms[i].pr_size = eaddr - saddr;
212
213       p = next;
214    }
215
216    *nump = num;
217    *mapp = ms;
218    free(procAsciiMap);
219    return 0;
220  parseerr:
221    free(ms);
222    fprintf(stderr, "DYNINSTgetMemoryMap: /proc/self/maps parse error\n");
223  freeBuffer:
224    free(procAsciiMap);
225  end:
226    return -1;
227 }