blob: fdb8a810232d0990dad972190992ee8649b49f52 [file] [log] [blame]
Evgeniy Stepanovcc603e92012-12-21 10:50:00 +00001================
2MemorySanitizer
3================
4
5.. contents::
6 :local:
7
8Introduction
9============
10
11MemorySanitizer is a detector of uninitialized reads. It consists of a
12compiler instrumentation module and a run-time library.
13
14Typical slowdown introduced by MemorySanitizer is **3x**.
15
16How to build
17============
18
19Follow the `clang build instructions <../get_started.html>`_. CMake
20build is supported.
21
22Usage
23=====
24
25Simply compile and link your program with ``-fsanitize=memory`` flag.
26The MemorySanitizer run-time library should be linked to the final
27executable, so make sure to use ``clang`` (not ``ld``) for the final
28link step. When linking shared libraries, the MemorySanitizer run-time
29is not linked, so ``-Wl,-z,defs`` may cause link errors (don't use it
30with MemorySanitizer). To get a reasonable performance add ``-O1`` or
31higher. To get meaninful stack traces in error messages add
32``-fno-omit-frame-pointer``. To get perfect stack traces you may need
33to disable inlining (just use ``-O1``) and tail call elimination
34(``-fno-optimize-sibling-calls``).
35
36.. code-block:: console
Dmitri Gribenko184e1c42012-12-23 18:36:44 +000037
Evgeniy Stepanovcc603e92012-12-21 10:50:00 +000038 % cat umr.cc
39 #include <stdio.h>
40
41 int main(int argc, char** argv) {
42 int* a = new int[10];
43 a[5] = 0;
44 if (a[argc])
45 printf("xx\n");
46 return 0;
47 }
48
49 % clang -fsanitize=memory -fPIE -pie -fno-omit-frame-pointer -g -O2 umr.cc
50
51If a bug is detected, the program will print an error message to
52stderr and exit with a non-zero exit code. Currently, MemorySanitizer
53does not symbolize its output by default, so you may need to use a
54separate script to symbolize the result offline (this will be fixed in
55future).
56
57.. code-block:: console
58
59 % ./a.out 2>log
60 % projects/compiler-rt/lib/asan/scripts/asan_symbolize.py / < log | c++filt
61 ==30106== WARNING: MemorySanitizer: UMR (uninitialized-memory-read)
62 #0 0x7f45944b418a in main umr.cc:6
63 #1 0x7f45938b676c in __libc_start_main libc-start.c:226
64 Exiting
65
66By default, MemorySanitizer exits on the first detected error.
67
68``__has_feature(memory_sanitizer)``
69------------------------------------
70
71In some cases one may need to execute different code depending on
72whether MemorySanitizer is enabled. :ref:`\_\_has\_feature
73<langext-__has_feature-__has_extension>` can be used for this purpose.
74
75.. code-block:: c
76
77 #if defined(__has_feature)
78 # if __has_feature(memory_sanitizer)
79 // code that builds only under MemorySanitizer
80 # endif
81 #endif
82
Kostya Serebryany85aee962013-02-26 06:58:27 +000083``__attribute__((no_sanitize_memory))``
84-----------------------------------------------
85
86Some code should not be checked by MemorySanitizer.
87One may use the function attribute
88:ref:`no_sanitize_memory <langext-memory_sanitizer>`
89to disable uninitialized checks in a particular function.
90MemorySanitizer may still instrument such functions to avoid false positives.
91This attribute may not be
92supported by other compilers, so we suggest to use it together with
93``__has_feature(memory_sanitizer)``. Note: currently, this attribute will be
94lost if the function is inlined.
95
Evgeniy Stepanovcc603e92012-12-21 10:50:00 +000096Origin Tracking
97===============
98
99MemorySanitizer can track origins of unitialized values, similar to
100Valgrind's --track-origins option. This feature is enabled by
101``-fsanitize-memory-track-origins`` Clang option. With the code from
102the example above,
103
104.. code-block:: console
105
106 % clang -fsanitize=memory -fsanitize-memory-track-origins -fPIE -pie -fno-omit-frame-pointer -g -O2 umr.cc
107 % ./a.out 2>log
108 % projects/compiler-rt/lib/asan/scripts/asan_symbolize.py / < log | c++filt
109 ==14425== WARNING: MemorySanitizer: UMR (uninitialized-memory-read)
110 ==14425== WARNING: Trying to symbolize code, but external symbolizer is not initialized!
111 #0 0x7f8bdda3824b in main umr.cc:6
112 #1 0x7f8bdce3a76c in __libc_start_main libc-start.c:226
113 raw origin id: 2030043137
114 ORIGIN: heap allocation:
115 #0 0x7f8bdda4034b in operator new[](unsigned long) msan_new_delete.cc:39
116 #1 0x7f8bdda3814d in main umr.cc:4
117 #2 0x7f8bdce3a76c in __libc_start_main libc-start.c:226
118 Exiting
119
120Origin tracking has proved to be very useful for debugging UMR
121reports. It slows down program execution by a factor of 1.5x-2x on top
122of the usual MemorySanitizer slowdown.
123
124Handling external code
125============================
126
127MemorySanitizer requires that all program code is instrumented. This
128also includes any libraries that the program depends on, even libc.
129Failing to achieve this may result in false UMR reports.
130
131Full MemorySanitizer instrumentation is very difficult to achieve. To
132make it easier, MemorySanitizer runtime library includes 70+
133interceptors for the most common libc functions. They make it possible
134to run MemorySanitizer-instrumented programs linked with
135uninstrumented libc. For example, the authors were able to bootstrap
136MemorySanitizer-instrumented Clang compiler by linking it with
137self-built instrumented libcxx (as a replacement for libstdc++).
138
139In the case when rebuilding all program dependencies with
140MemorySanitizer is problematic, an experimental MSanDR tool can be
141used. It is a DynamoRio-based tool that uses dynamic instrumentation
142to avoid false positives due to uninstrumented code. The tool simply
143marks memory from instrumented libraries as fully initialized. See
144`http://code.google.com/p/memory-sanitizer/wiki/Running#Running_with_the_dynamic_tool`
145for more information.
146
147Supported Platforms
148===================
149
150MemorySanitizer is supported on
151
152* Linux x86\_64 (tested on Ubuntu 10.04 and 12.04);
153
154Limitations
155===========
156
157* MemorySanitizer uses 2x more real memory than a native run, 3x with
158 origin tracking.
159* MemorySanitizer maps (but not reserves) 64 Terabytes of virtual
160 address space. This means that tools like ``ulimit`` may not work as
161 usually expected.
162* Static linking is not supported.
163* Non-position-independent executables are not supported.
164* Depending on the version of Linux kernel, running without ASLR may
165 be not supported. Note that GDB disables ASLR by default. To debug
166 instrumented programs, use "set disable-randomization off".
167
168Current Status
169==============
170
171MemorySanitizer is an experimental tool. It is known to work on large
172real-world programs, like Clang/LLVM itself.
173
174More Information
175================
176
177`http://code.google.com/p/memory-sanitizer <http://code.google.com/p/memory-sanitizer/>`_
178