149 строки
7.5 KiB
Plaintext
149 строки
7.5 KiB
Plaintext
|
30 March 2009
|
||
|
|
||
|
This file documents Open MPI's usage of ptmalloc2. This is perhaps
|
||
|
our 7,208,499th iteration of ptmalloc2 support, so let's document it
|
||
|
here so that some future developer might spend *slightly* less time
|
||
|
understanding what the heck is going on.
|
||
|
|
||
|
See glibc documentation about malloc hooks before continuing. This is
|
||
|
pretty much required reading before reading the rest of this file /
|
||
|
having a hope of understanding what's going on here:
|
||
|
|
||
|
http://www.gnu.org/software/libc/manual/html_mono/libc.html#Hooks-for-Malloc
|
||
|
|
||
|
The overall goal is that we're using the Linux glibc hooks to wholly
|
||
|
replace the underlying allocator. We *used* to use horrid linker
|
||
|
tricks to interpose OMPI's ptmalloc2 symbols with the glibc ones --
|
||
|
meaning that user apps would call our symbols and not the glibc ones.
|
||
|
But that scheme is fraught with problems, not the least of which is
|
||
|
that *all* MPI applications will be forced to use our overridden
|
||
|
allocator (not just the ones that need it, such as the ones running on
|
||
|
OpenFabrics-based networks). Instead, what we do here is, frankly,
|
||
|
quite similar to what is done in MX: we use the 4 glibc hooks to
|
||
|
assert our own malloc, realloc, free, and memalign functions. This
|
||
|
allows the decision as to whether to use this internal ptmalloc2
|
||
|
allocate to be a run-time decision. This is quite important; using
|
||
|
this internal allocator has both benefits (allowing using
|
||
|
mpi_leave_pinned=1 behavior) and drawbacks (breaking some debuggers,
|
||
|
being unnecessary for non-OpenFabrics-based networks, etc.).
|
||
|
|
||
|
Here's how it works...
|
||
|
|
||
|
This component *must* be linked statically as part of libopen-pal; it
|
||
|
*cannot* be a DSO. Specifically, this library must be present during
|
||
|
pre-main() initialization phases so that its __malloc_initialize_hook
|
||
|
can be found and executed. Loading it as a DSO during MPI_INIT is far
|
||
|
too late. In configure.m4, we define the M4 macro
|
||
|
MCA_memory_ptmalloc2_COMPILE_MODE to always compile this component in
|
||
|
static mode. Yay flexible build system.
|
||
|
|
||
|
This component provides an munmap() function that will intercept calls
|
||
|
to munmap() and do the Right Thing. That is fairly straightforward to
|
||
|
do. Intercepting the malloc/free/etc. allocator is much more
|
||
|
complicated.
|
||
|
|
||
|
All the ptmalloc2 public symbols in this component have been name
|
||
|
shifted via the rename.h file. Hence, what used to be "malloc" is now
|
||
|
opal_memory_ptmalloc2_malloc. Since all the public symbols are
|
||
|
name-shifted, we can safely link this component in all MPI
|
||
|
applications. Specifically: just because this ptmalloc2 allocator is
|
||
|
present in all OMPI executables and user-level applications, it won't
|
||
|
necessarily be used -- it's a separate/run-time decision as to whether
|
||
|
it will be used.
|
||
|
|
||
|
For hysterical raisins (v1.3.0 and v1.3.1), an empty
|
||
|
libopenmpi_malloc.la is created. This library *used* to be the whole
|
||
|
ptmalloc2 allocator, but since we're now name-shifting all the
|
||
|
ptmalloc2 symbols, it's no longer necessary to make it a separate
|
||
|
library. Having an empty/dummy library of this name ensures that
|
||
|
users who added -lopenmpi_malloc to their compile/link lines won't
|
||
|
fail linking. They may get a linker warning about no symbols being
|
||
|
used in this library -- but that's a Good Thing; it'll encourage users
|
||
|
to stop linking in this library.
|
||
|
|
||
|
We set the __malloc_initialize_hook variable to point to
|
||
|
opal_memory_ptmalloc2_malloc_init_hook (in hooks.c). This function is
|
||
|
called by the underlying glibc allocator before any allocations occur
|
||
|
and before the memory allocation subsystem is setup. As such, this
|
||
|
function is *extremely* restricted in what it can do. It cannot call
|
||
|
any form of malloc, for example (which seems fairly obvious, but it's
|
||
|
worth mentioning :-) ). This function is one of the determining
|
||
|
steps as to whether we'll use the internal ptmalloc2 allocator or
|
||
|
not. It performs checks including (but not limited to; see hooks.c
|
||
|
for the most up-to-date set of tests):
|
||
|
|
||
|
- see if the OMPI_MCA_mpi_leave_pinned environment variable is set.
|
||
|
Yes, I know this is a horrid abstraction violation, but this
|
||
|
function may be invoked pre-main -- it's certainly before MCA
|
||
|
parameters have been setup. So just getenv() and see if it has been
|
||
|
set.
|
||
|
- look for the hard-coded filename /sys/class/infiniband.
|
||
|
- if the env variable was not set, but the file is there, enable our
|
||
|
ptmalloc2.
|
||
|
- if the env variable was set to 0, disable our ptmalloc2
|
||
|
- if the env variable is set to -1, enable our ptmalloc2 if the file
|
||
|
was found
|
||
|
- if the env variable is any other value, enable our ptmalloc2
|
||
|
- if we're enabling our ptmalloc2, initialize ptmalloc (via
|
||
|
ptmalloc_init()) and then set the 4 hooks to point to our
|
||
|
name-shifted ptmalloc2 functions
|
||
|
|
||
|
Hence, sometime during process startup, this function will definitely
|
||
|
be called. It will either set the 4 hook functions to point to our
|
||
|
name-shifted ptmalloc2 functions, or it won't. If the 4 hook
|
||
|
functions are set, then the underlying glibc allocator will always
|
||
|
call our 4 functions in all the relevant places instead of calling its
|
||
|
own functions. Specifically: the process is calling the underlying
|
||
|
glibc allocator, but that underlying glibc allocator will make
|
||
|
function pointer callbacks to our name-shifted ptmalloc2 functions to
|
||
|
actually do the work.
|
||
|
|
||
|
Note that because we know our ptmalloc will not be providing all 5
|
||
|
hook variables (because we want to use the underlying glibc hook
|
||
|
variables), they are #if 0'ed out in our malloc.c. This has the
|
||
|
direct consequence that the *_hook_ini() in hooks.c are never used.
|
||
|
So to avoid compiler/linker warnings, I #if 0'ed those out as well.
|
||
|
|
||
|
All the public functions in malloc.c that call hook functions were
|
||
|
modified to #if 0 the hook function invocations. After all, that's
|
||
|
something that we want the *underlying* glibc allocator to do -- but
|
||
|
we are putting these functions as the hooks, so we don't want to
|
||
|
invoke ourselves in an infinite loop!
|
||
|
|
||
|
The next thing that happens in the startup sequence is that the
|
||
|
ptmalloc2 memory component's "open" function is called during
|
||
|
MPI_INIT. As stated above, we always intercept munmap() -- this is
|
||
|
acceptable in all environments. But we need to test to see if the
|
||
|
glibc memory hooks have been overridden before MPI_INIT was invoked.
|
||
|
If so, we need to signal that our allocator support may not be
|
||
|
complete.
|
||
|
|
||
|
Patrick Geofray/MX suggests a simple test: malloc() 4MB and then free
|
||
|
it. Watch to see if our name-shifted ptmalloc2 free() function was
|
||
|
invoked. If it was, then all of our hooks are probably in place and
|
||
|
we can proceed. If not, then set flags indicating that this memory
|
||
|
allocator only supports MUNMAP (not FREE/CHUNK).
|
||
|
|
||
|
NOTE: we *used* to simply set the FREE/CHUNK support flags during our
|
||
|
ptmalloc2's internal ptmalloc_init() function. This is not a good
|
||
|
idea becaus eeven after our ptmalloc_init() function has been invoked,
|
||
|
someone may come in an override our memory hooks. Doing a malloc/free
|
||
|
test during the ptmalloc2 memory component's open function seems to be
|
||
|
the safest way to test whether we *actually* support FREE/CHUNK (this
|
||
|
is what MX does, too).
|
||
|
|
||
|
Much later in the init sequence during MPI_INIT, components indicate
|
||
|
whether they want to use mpi_leave_pinned[_pipeline] support or not.
|
||
|
For example, the openib BTL queries the opal_mem_hooks_support_level()
|
||
|
function to see if FREE and MUNMAP are supported. If they are, then
|
||
|
the openib BTL sets mpi_leave_pinned = 1.
|
||
|
|
||
|
Finally, the mpool base does a final check. If
|
||
|
mpi_leave_pinned[_pipeline] is set to 1 and/or use_mem_hooks is set,
|
||
|
if FREE/MUNMAP are not set in the supported flags, then a warning is
|
||
|
printed. Otherwise, life continues (assumedly using
|
||
|
mpi_leave_pinned[_pipeline] support).
|
||
|
|
||
|
Simple, right?
|
||
|
|