I'm debugging a problem which occurs only in the PPC64 port of my program.
I have a test case where the C library qsort
is given a libffi-generated closure as a string comparison callback. Strings are passed to the callback properly, and the return value is stored precisely into the return value buffer passed by libffi to the closure function.
However, the array is not correctly sorted by qsort
. Moreover, Valgrind reports that the C library qsort code is accessing uninitialized memory, and --track-orgins=yes
reveals that this memory was stack-allocted by Libffi. I strongly suspect that this is the return value, and so the sorting isn't correct due to garbage comparisons.
I.e. Libffi allocated the buffer for the return value, and is propagating the value of that to the callback caller; but my closure dispatch function is being given the wrong pointer, and so is not putting the return value in the right place.
For some bizarre reason, Valgrind doesn't report the address of the uninitialized memory, only where in the code the use occurred and where it was allocated.
I simply want to compare the address of that location to the pointer that is passed to the closure function: are they even remotely close?
Is there some way to get that information out of Valgrind?
UPDATE: I'm working on a GCC Compile Farm machine where I don't have root; the installed libffi has no debug info. It is version 3.0.13.
However, the issue reproduces with the libffi
git head which I just built.
I have confirmed it is the return value area that is uninitialized.
I added an instruction to the closure dispatching assembly code ffi_closure_LINUX64
to initialize a double-word-sized area at the bottom of the RETVAL
part of the closure dispatch stack frame. This makes the Valgrind error go away; but of course the return value is garbage. It also confirms a basic piece of sanity: that the code before the call to the closure dispatch helper and the code after are referring to the same area for the return value. (The stack pointer hasn't moved unexpectedly and the frame references are correct.) Just whatever address the user code is ultimately getting isn't pointing to that return value.
Next, I moved the initialization of the return area down into the C function called ffi_closure_helper_LINUX64
, near the entry into the function. This also still makes the uninitialized error go away, confirming the helper is getting the correct return value area address through %r6
(argument 4).
There is no feature in valgrind to report the address of the uninit memory, as this would (in most case) not help the user : a stack address or heap address cannot really indicate much.
You might maybe have some more info by setting a breakpoint in the frame reported by Valgrind, and mark various pieces of the stack as initialised, using gdb+vgdb+memcheck monitor commands. When setting the faulty location to initialised, valgrind should not report the error anymore. You might have to do several runs, each time marking other vars/zone of the stack.
See http://www.valgrind.org/docs/manual/mc-manual.html#mc-manual.monitor-commands and GDB user manual to see how to write (sophisticated) commands run when a breakpoint is reached.