1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
| | = mwrap - LD_PRELOAD malloc wrapper + line stats for Ruby
mwrap is designed to answer the question:
Which lines of Ruby are hitting malloc the most?
mwrap wraps all malloc-family calls to trace the Ruby source
location of such calls and bytes allocated at each callsite.
As of mwrap 2.0.0, it can also function as a leak detector
and show live allocations at every call site. Depending on
your application and workload, the overhead is roughly a 50-100%
increase memory and runtime with default settings.
It works best for allocations under GVL, but tries to track
numeric caller addresses for allocations made without GVL so you
can get an idea of how much memory usage certain extensions and
native libraries use.
As of 3.0, it also gives configurable C backtraces of all
dynamically-linked malloc callsites for any program where backtrace(3)
works, including programs not linked to Ruby.
It requires the concurrent lock-free hash table from the
Userspace RCU project: https://liburcu.org/
It does not require recompiling or rebuilding Ruby, but only
supports Ruby 2.7.0 or later on a few platforms:
* GNU/Linux (only tested --without-jemalloc, mwrap 3.x provides its own malloc)
It may work on FreeBSD, NetBSD, OpenBSD and DragonFly BSD if given
appropriate build options.
== Install
# Debian-based systems: apt-get install liburcu-dev
# Install mwrap via RubyGems.org
gem install mwrap
== Usage
mwrap works as an LD_PRELOAD and supplies a mwrap RubyGem executable to
improve ease-of-use. You can set `dump_csv:' in the MWRAP environment
variable to append the results to a CSV file:
MWRAP=dump_csv:/path/to/log mwrap RUBY_COMMAND
(`dump_csv:' is new in mwrap 3.x, `dump_file:' from earlier versions is
still supported).
For long running processes, you can see the AF_UNIX HTTP interface:
MWRAP=socket_dir:/some/dir mwrap COMMAND
And connect via `curl --unix-socket /some/dir/$PID.sock' or
`mwrap-rproxy(1p)<https://80x24.org/mwrap-perl.git/tree/script/mwrap-rproxy#n44>
distributed with the Perl version of mwrap for more info.
You may also `require "mwrap"' in your Ruby code and use
Mwrap.dump, Mwrap.reset, Mwrap.each, etc.
However, mwrap MUST be loaded via LD_PRELOAD to have any
effect in tracking malloc use. However, it is safe to keep
"require 'mwrap'" in performance-critical deployments,
as overhead is only incurred when used as an LD_PRELOAD.
The output of `dump_csv:' is has self-describing columns and is
subject to change. SQLite 3.32+ can load it with: `.import --csv'.
The output of the `dump_file:' output is a text file with 3 columns:
total_bytes call_count location
Where location is a Ruby source location (if made under GVL)
or an address retrieved by backtrace_symbols(3). It is
recommended to use the sort(1) command on either of the
first two columns to find the hottest malloc locations.
mwrap 3.0.0+ also supports an embedded HTTP server
it is documented at:
https://80x24.org/mwrap.git/tree/Documentation/mwrap.pod
== Known problems
* 32-bit machines are prone to overflow (WONTFIX)
* signalfd(2)-reliant code will need latest URCU with commit
ea3a28a3f71dd02f (Disable signals in URCU background threads, 2022-09-23)
This doesn't affect C Ruby itself, and signalfd(2) use is rare
3rd-party processes.
* Ruby source files over 16.7 million lines long are not supported :P
== Public mail archives (HTTP, Atom feeds, IMAP mailbox, NNTP group, POP3):
* https://80x24.org/mwrap-public/
* nntps://80x24.org/inbox.comp.lang.ruby.mwrap
* imaps://;AUTH=ANONYMOUS@80x24.org/inbox.comp.lang.ruby.mwrap.0
* https://80x24.org/mwrap-public/_/text/help/#pop3
No subscription nor real identities will ever be required to obtain support,
but HTML mail is rejected. Memory usage reductions start with you;
only send plain-text mail to us and do not top-post. HTML mail and
top-posting costs everybody memory and bandwidth.
mwrap-public@80x24.org
== Hacking
git clone https://80x24.org/mwrap.git
Send all patches ("git format-patch" + "git send-email") and
pull requests (use "git request-pull" to format) via email
to mwrap-public@80x24.org. We do not and will not use
proprietary messaging systems.
== License
GPL-3.0+ <https://www.gnu.org/licenses/gpl-3.0.txt>
|