• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 
2 /*--------------------------------------------------------------------*/
3 /*--- Ptrcheck: a pointer-use checker.             pc_intercepts.c ---*/
4 /*--------------------------------------------------------------------*/
5 
6 /*
7    This file is part of Ptrcheck, a Valgrind tool for checking pointer
8    use in programs.
9 
10    Copyright (C) 2003-2010 Nicholas Nethercote
11       njn@valgrind.org
12 
13    This program is free software; you can redistribute it and/or
14    modify it under the terms of the GNU General Public License as
15    published by the Free Software Foundation; either version 2 of the
16    License, or (at your option) any later version.
17 
18    This program is distributed in the hope that it will be useful, but
19    WITHOUT ANY WARRANTY; without even the implied warranty of
20    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
21    General Public License for more details.
22 
23    You should have received a copy of the GNU General Public License
24    along with this program; if not, write to the Free Software
25    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
26    02111-1307, USA.
27 
28    The GNU General Public License is contained in the file COPYING.
29 */
30 
31 /* Nothing actually in here.  However it appears this file is needed
32    to make malloc intercepting work. (jrs, 2 july 08 -- not sure about
33    that).
34 */
35 
36 #include "pub_tool_basics.h"
37 #include "pub_tool_hashtable.h"
38 #include "pub_tool_redir.h"
39 #include "pub_tool_tooliface.h"
40 #include "valgrind.h"
41 
42 
43 /* The following intercepts are copied verbatim from
44    memcheck/mc_replace_strmem.c.  If you copy more in, please keep
45    them in the same order as in mc_replace_strmem.c. */
46 
47 
48 #define STRRCHR(soname, fnname) \
49    char* VG_REPLACE_FUNCTION_ZU(soname,fnname)( const char* s, int c ); \
50    char* VG_REPLACE_FUNCTION_ZU(soname,fnname)( const char* s, int c ) \
51    { \
52       UChar  ch   = (UChar)((UInt)c); \
53       UChar* p    = (UChar*)s; \
54       UChar* last = NULL; \
55       while (True) { \
56          if (*p == ch) last = p; \
57          if (*p == 0) return last; \
58          p++; \
59       } \
60    }
61 
62 // Apparently rindex() is the same thing as strrchr()
63 STRRCHR(VG_Z_LIBC_SONAME,   strrchr)
64 STRRCHR(VG_Z_LIBC_SONAME,   rindex)
65 #if defined(VGO_linux)
66 STRRCHR(VG_Z_LIBC_SONAME,   __GI_strrchr)
67 STRRCHR(VG_Z_LD_LINUX_SO_2, rindex)
68 #elif defined(VGO_darwin)
69 STRRCHR(VG_Z_DYLD,          strrchr)
70 STRRCHR(VG_Z_DYLD,          rindex)
71 #endif
72 
73 
74 #define STRCHR(soname, fnname) \
75    char* VG_REPLACE_FUNCTION_ZU(soname,fnname) ( const char* s, int c ); \
76    char* VG_REPLACE_FUNCTION_ZU(soname,fnname) ( const char* s, int c ) \
77    { \
78       UChar  ch = (UChar)((UInt)c); \
79       UChar* p  = (UChar*)s; \
80       while (True) { \
81          if (*p == ch) return p; \
82          if (*p == 0) return NULL; \
83          p++; \
84       } \
85    }
86 
87 // Apparently index() is the same thing as strchr()
88 STRCHR(VG_Z_LIBC_SONAME,          strchr)
89 STRCHR(VG_Z_LIBC_SONAME,          index)
90 #if defined(VGO_linux)
91 STRCHR(VG_Z_LIBC_SONAME,          __GI_strchr)
92 STRCHR(VG_Z_LD_LINUX_SO_2,        strchr)
93 STRCHR(VG_Z_LD_LINUX_SO_2,        index)
94 STRCHR(VG_Z_LD_LINUX_X86_64_SO_2, strchr)
95 STRCHR(VG_Z_LD_LINUX_X86_64_SO_2, index)
96 #elif defined(VGO_darwin)
97 STRCHR(VG_Z_DYLD,                 strchr)
98 STRCHR(VG_Z_DYLD,                 index)
99 #endif
100 
101 
102 #define STRNLEN(soname, fnname) \
103    SizeT VG_REPLACE_FUNCTION_ZU(soname,fnname) ( const char* str, SizeT n ); \
104    SizeT VG_REPLACE_FUNCTION_ZU(soname,fnname) ( const char* str, SizeT n ) \
105    { \
106       SizeT i = 0; \
107       while (i < n && str[i] != 0) i++; \
108       return i; \
109    }
110 
111 STRNLEN(VG_Z_LIBC_SONAME, strnlen)
112 
113 
114 // Note that this replacement often doesn't get used because gcc inlines
115 // calls to strlen() with its own built-in version.  This can be very
116 // confusing if you aren't expecting it.  Other small functions in this file
117 // may also be inline by gcc.
118 #define STRLEN(soname, fnname) \
119    SizeT VG_REPLACE_FUNCTION_ZU(soname,fnname)( const char* str ); \
120    SizeT VG_REPLACE_FUNCTION_ZU(soname,fnname)( const char* str ) \
121    { \
122       SizeT i = 0; \
123       while (str[i] != 0) i++; \
124       return i; \
125    }
126 
127 STRLEN(VG_Z_LIBC_SONAME,          strlen)
128 #if defined(VGO_linux)
129 STRLEN(VG_Z_LIBC_SONAME,          __GI_strlen)
130 STRLEN(VG_Z_LD_LINUX_SO_2,        strlen)
131 STRLEN(VG_Z_LD_LINUX_X86_64_SO_2, strlen)
132 STRLEN(VG_Z_LD_SO_1,              strlen)
133 #endif
134 
135 
136 #define STRCPY(soname, fnname) \
137    char* VG_REPLACE_FUNCTION_ZU(soname, fnname) ( char* dst, const char* src ); \
138    char* VG_REPLACE_FUNCTION_ZU(soname, fnname) ( char* dst, const char* src ) \
139    { \
140       Char* dst_orig = dst; \
141       \
142       while (*src) *dst++ = *src++; \
143       *dst = 0; \
144       \
145       return dst_orig; \
146    }
147 
148 STRCPY(VG_Z_LIBC_SONAME, strcpy)
149 #if defined(VGO_linux)
150 STRCPY(VG_Z_LIBC_SONAME, __GI_strcpy)
151 #elif defined(VGO_darwin)
152 STRCPY(VG_Z_DYLD,        strcpy)
153 #endif
154 
155 
156 #define STRNCMP(soname, fnname) \
157    int VG_REPLACE_FUNCTION_ZU(soname,fnname) \
158           ( const char* s1, const char* s2, SizeT nmax ); \
159    int VG_REPLACE_FUNCTION_ZU(soname,fnname) \
160           ( const char* s1, const char* s2, SizeT nmax ) \
161    { \
162       SizeT n = 0; \
163       while (True) { \
164          if (n >= nmax) return 0; \
165          if (*s1 == 0 && *s2 == 0) return 0; \
166          if (*s1 == 0) return -1; \
167          if (*s2 == 0) return 1; \
168          \
169          if (*(unsigned char*)s1 < *(unsigned char*)s2) return -1; \
170          if (*(unsigned char*)s1 > *(unsigned char*)s2) return 1; \
171          \
172          s1++; s2++; n++; \
173       } \
174    }
175 
176 STRNCMP(VG_Z_LIBC_SONAME, strncmp)
177 #if defined(VGO_linux)
178 STRNCMP(VG_Z_LIBC_SONAME, __GI_strncmp)
179 #elif defined(VGO_darwin)
180 STRNCMP(VG_Z_DYLD,        strncmp)
181 #endif
182 
183 
184 #define STRCMP(soname, fnname) \
185    int VG_REPLACE_FUNCTION_ZU(soname,fnname) \
186           ( const char* s1, const char* s2 ); \
187    int VG_REPLACE_FUNCTION_ZU(soname,fnname) \
188           ( const char* s1, const char* s2 ) \
189    { \
190       register unsigned char c1; \
191       register unsigned char c2; \
192       while (True) { \
193          c1 = *(unsigned char *)s1; \
194          c2 = *(unsigned char *)s2; \
195          if (c1 != c2) break; \
196          if (c1 == 0) break; \
197          s1++; s2++; \
198       } \
199       if ((unsigned char)c1 < (unsigned char)c2) return -1; \
200       if ((unsigned char)c1 > (unsigned char)c2) return 1; \
201       return 0; \
202    }
203 
204 STRCMP(VG_Z_LIBC_SONAME,          strcmp)
205 #if defined(VGO_linux)
206 STRCMP(VG_Z_LIBC_SONAME,          __GI_strcmp)
207 STRCMP(VG_Z_LD_LINUX_X86_64_SO_2, strcmp)
208 STRCMP(VG_Z_LD64_SO_1,            strcmp)
209 #endif
210 
211 
212 #define MEMCHR(soname, fnname) \
213    void* VG_REPLACE_FUNCTION_ZU(soname,fnname) (const void *s, int c, SizeT n); \
214    void* VG_REPLACE_FUNCTION_ZU(soname,fnname) (const void *s, int c, SizeT n) \
215    { \
216       SizeT i; \
217       UChar c0 = (UChar)c; \
218       UChar* p = (UChar*)s; \
219       for (i = 0; i < n; i++) \
220          if (p[i] == c0) return (void*)(&p[i]); \
221       return NULL; \
222    }
223 
224 MEMCHR(VG_Z_LIBC_SONAME, memchr)
225 #if defined(VGO_darwin)
226 MEMCHR(VG_Z_DYLD,        memchr)
227 #endif
228 
229 
230 #define MEMCPY(soname, fnname) \
231    void* VG_REPLACE_FUNCTION_ZU(soname,fnname) \
232             ( void *dst, const void *src, SizeT sz ); \
233    void* VG_REPLACE_FUNCTION_ZU(soname,fnname) \
234             ( void *dest, const void *src, SizeT sz ) \
235    { \
236    const UChar*  s  = (const UChar*)src; \
237          UChar*  d  =       (UChar*)dest; \
238    const UWord*  sW = (const UWord*)src; \
239          UWord*  dW =       (UWord*)dest; \
240    const UWord   al = sizeof(UWord)-1; \
241    \
242    if (0 == (((UWord)dW) & al) && 0 == (((UWord)sW) & al)) { \
243       while (sz >= 4 * sizeof(UWord)) { \
244          dW[0] = sW[0]; \
245          dW[1] = sW[1]; \
246          dW[2] = sW[2]; \
247          dW[3] = sW[3]; \
248          sz -= 4 * sizeof(UWord); \
249          dW += 4; \
250          sW += 4; \
251       } \
252       if (sz == 0) \
253          return dest; \
254       while (sz >= 1 * sizeof(UWord)) { \
255          dW[0] = sW[0]; \
256          sz -= 1 * sizeof(UWord); \
257          dW += 1; \
258          sW += 1; \
259       } \
260       if (sz == 0) \
261          return dest; \
262       s = (const UChar*)sW; \
263       d = (UChar*)dW; \
264    } \
265    \
266    while (sz--) \
267       *d++ = *s++; \
268    \
269    return dest; \
270    }
271 
272 MEMCPY(VG_Z_LIBC_SONAME, memcpy)
273 #if defined(VGO_linux)
274 MEMCPY(VG_Z_LD_SO_1,     memcpy) /* ld.so.1 */
275 MEMCPY(VG_Z_LD64_SO_1,   memcpy) /* ld64.so.1 */
276 #endif
277 
278 
279 /* Copy SRC to DEST, returning the address of the terminating '\0' in
280    DEST. (minor variant of strcpy) */
281 #define STPCPY(soname, fnname) \
282    char* VG_REPLACE_FUNCTION_ZU(soname,fnname) ( char* dst, const char* src ); \
283    char* VG_REPLACE_FUNCTION_ZU(soname,fnname) ( char* dst, const char* src ) \
284    { \
285       while (*src) *dst++ = *src++; \
286       *dst = 0; \
287       \
288       return dst; \
289    }
290 
291 STPCPY(VG_Z_LIBC_SONAME,          stpcpy)
292 #if defined(VGO_linux)
293 STPCPY(VG_Z_LD_LINUX_SO_2,        stpcpy)
294 STPCPY(VG_Z_LD_LINUX_X86_64_SO_2, stpcpy)
295 #endif
296 
297 
298 /* Find the first occurrence of C in S.  */
299 #define GLIBC232_RAWMEMCHR(soname, fnname) \
300    char* VG_REPLACE_FUNCTION_ZU(soname,fnname) (const char* s, int c_in); \
301    char* VG_REPLACE_FUNCTION_ZU(soname,fnname) (const char* s, int c_in) \
302    { \
303       unsigned char  c        = (unsigned char) c_in; \
304       unsigned char* char_ptr = (unsigned char *)s; \
305       while (1) { \
306          if (*char_ptr == c) return char_ptr; \
307          char_ptr++; \
308       } \
309    }
310 
311 GLIBC232_RAWMEMCHR(VG_Z_LIBC_SONAME, rawmemchr)
312 #if defined (VGO_linux)
313 GLIBC232_RAWMEMCHR(VG_Z_LIBC_SONAME, __GI___rawmemchr)
314 #endif
315 
316 
317 #define STRSTR(soname, fnname) \
318    void* VG_REPLACE_FUNCTION_ZU(soname,fnname) \
319          (void* haystack, void* needle); \
320    void* VG_REPLACE_FUNCTION_ZU(soname,fnname) \
321          (void* haystack, void* needle) \
322    { \
323       UChar* h = (UChar*)haystack; \
324       UChar* n = (UChar*)needle; \
325       \
326       /* find the length of n, not including terminating zero */ \
327       UWord nlen = 0; \
328       while (n[nlen]) nlen++; \
329       \
330       /* if n is the empty string, match immediately. */ \
331       if (nlen == 0) return h; \
332       \
333       /* assert(nlen >= 1); */ \
334       UChar n0 = n[0]; \
335       \
336       while (1) { \
337          UChar hh = *h; \
338          if (hh == 0) return NULL; \
339          if (hh != n0) { h++; continue; } \
340          \
341          UWord i; \
342          for (i = 0; i < nlen; i++) { \
343             if (n[i] != h[i]) \
344                break; \
345          } \
346          /* assert(i >= 0 && i <= nlen); */ \
347          if (i == nlen) \
348             return h; \
349          \
350          h++; \
351       } \
352    }
353 
354 #if defined(VGO_linux)
355 STRSTR(VG_Z_LIBC_SONAME,          strstr)
356 #endif
357 
358 
359 #define STRPBRK(soname, fnname) \
360    void* VG_REPLACE_FUNCTION_ZU(soname,fnname) \
361          (void* sV, void* acceptV); \
362    void* VG_REPLACE_FUNCTION_ZU(soname,fnname) \
363          (void* sV, void* acceptV) \
364    { \
365       UChar* s = (UChar*)sV; \
366       UChar* accept = (UChar*)acceptV; \
367       \
368       /*  find the length of 'accept', not including terminating zero */ \
369       UWord nacc = 0; \
370       while (accept[nacc]) nacc++; \
371       \
372       /* if n is the empty string, fail immediately. */ \
373       if (nacc == 0) return NULL; \
374       \
375       /* assert(nacc >= 1); */ \
376       while (1) { \
377          UWord i; \
378          UChar sc = *s; \
379          if (sc == 0) \
380             break; \
381          for (i = 0; i < nacc; i++) { \
382             if (sc == accept[i]) \
383                return s; \
384          } \
385          s++; \
386       } \
387       \
388       return NULL; \
389    }
390 
391 #if defined(VGO_linux)
392 STRPBRK(VG_Z_LIBC_SONAME,          strpbrk)
393 #endif
394 
395 
396 /*--------------------------------------------------------------------*/
397 /*--- end                                          pc_intercepts.c ---*/
398 /*--------------------------------------------------------------------*/
399