[GHC] #15333: Weird cachegrind results for binary-trees
GHC
ghc-devs at haskell.org
Fri Aug 10 07:30:54 UTC 2018
#15333: Weird cachegrind results for binary-trees
-------------------------------------+-------------------------------------
Reporter: sgraf | Owner: (none)
Type: bug | Status: new
Priority: normal | Milestone: Research
Component: NoFib benchmark | needed
suite | Version: 8.5
Resolution: | Keywords:
Operating System: Unknown/Multiple | Architecture:
| Unknown/Multiple
Type of failure: None/Unknown | Test Case:
Blocked By: | Blocking:
Related Tickets: | Differential Rev(s):
Wiki Page: |
-------------------------------------+-------------------------------------
Description changed by sgraf:
Old description:
> I'm currently investigating an alleged regression in my branch of the
> late lambda lift and hit a confusing data point. Note that I'm very much
> relying on cachegrinds counted instructions/memory accesses for my
> findings.
>
> Check out the most recent version of `nofib` and run the following
> script:
>
> {{{#!sh
> #! /bin/sh
> sed -i 's/import Debug.Trace//g' Main.hs # Make the following line
> idempotent
> echo "import Debug.Trace" | cat - Main.hs > Main.tmp && mv Main.tmp
> Main.hs # add the import for trace
> sed -i 's/trace "" \$ bit/bit/g' Main.hs # strip `trace $ ` prefixes in
> the call to `bit`
> ghc -O2 -XBangPatterns Main.hs -o bt1
> sed -i 's/bit/trace "" $ bit/g' Main.hs # prepend `trace $ ` to the call
> to `bit`
> ghc -O2 -XBangPatterns Main.hs -o bt2
> valgrind --tool=cachegrind ./bt1 12 2>&1 > /dev/null # without trace
> valgrind --tool=cachegrind ./bt2 12 2>&1 > /dev/null # with trace
> }}}
>
> This will compile two versions of `binary-trees`, one with an extra
> `trace "" $` call before the only call to the `bit` function. One would
> expect the version with the `trace` call (`bt1`) to allocate more than
> the version without (`bt2`). Indeed, the output of `+RTS -s` suggests
> that:
>
> {{{
> $ ./bt1 12 +RTS -s
> ...
> 43,107,560 bytes allocated in the heap
> ...
> $ ./bt2 12 +RTS -s
> ...
> 43,116,888 bytes allocated in the heap
> ...
> }}}
>
> That's fine. A few benchmark runs by hand also suggested the tracing
> version is a little slower (probably due to IO).
>
> Compare that to the output of the above cachegrind calls:
>
> {{{
> $ valgrind --tool=cachegrind ./bt1 12 > /dev/null
> ...
> I refs: 118,697,583
> ...
> D refs: 43,475,212
> ...
> $ valgrind --tool=cachegrind ./bt2 12 > /dev/null
> ...
> I refs: 116,340,710
> ...
> D refs: 42,523,369
> ...
> }}}
>
> It's the other way round here! How's that possible?
>
> Even if this isn't strictly a bug in GHC or NoFib, it's relevant
> nonetheless, as our benchmark infrastructure currently relies on
> instruction counts. I couldn't reproduce this by writing my own no-op
> `trace _ a = a; {-# NOINLINE trace #-}`, btw.
>
> I checked this on GHC 8.2.2 and a semi-recent HEAD commit
> (bb539cfe335eeec7989332b859b1f3162c5e105a).
New description:
I'm currently investigating an alleged regression in my branch of the late
lambda lift and hit a confusing data point. Note that I'm very much
relying on cachegrinds counted instructions/memory accesses for my
findings.
Check out the most recent version of `nofib` and run the following script:
{{{#!sh
#! /bin/sh
sed -i 's/import Debug.Trace//g' Main.hs # Make the following line
idempotent
echo "import Debug.Trace" | cat - Main.hs > Main.tmp && mv Main.tmp
Main.hs # add the import for trace
# bt1: Vanilly
sed -i 's/trace "" \$ bit/bit/g' Main.hs # strip `trace $ ` prefixes in
the call to `bit`
ghc -O2 -XBangPatterns Main.hs -o bt1
# bt2: Additional trace call
sed -i 's/bit/trace "" $ bit/g' Main.hs # prepend `trace $ ` to the call
to `bit`
ghc -O2 -XBangPatterns Main.hs -o bt2
valgrind --tool=cachegrind ./bt1 12 2>&1 > /dev/null # without trace
valgrind --tool=cachegrind ./bt2 12 2>&1 > /dev/null # with trace
}}}
This will compile two versions of `binary-trees`, the original, unchanged
version and one with an extra `trace "" $` call before the only call to
the `bit` function. One would expect the version with the `trace` call
(`bt2`) to allocate more than the version without (`bt1`). Indeed, the
output of `+RTS -s` suggests that:
{{{
$ ./bt1 12 +RTS -s
...
43,107,560 bytes allocated in the heap
...
$ ./bt2 12 +RTS -s
...
43,116,888 bytes allocated in the heap
...
}}}
That's fine. A few benchmark runs by hand also suggested the tracing
version is a little slower (probably due to IO).
Compare that to the output of the above cachegrind calls:
{{{
$ valgrind --tool=cachegrind ./bt1 12 > /dev/null
...
I refs: 118,697,583
...
D refs: 43,475,212
...
$ valgrind --tool=cachegrind ./bt2 12 > /dev/null
...
I refs: 116,340,710
...
D refs: 42,523,369
...
}}}
It's the other way round here! How's that possible?
Even if this isn't strictly a bug in GHC or NoFib, it's relevant
nonetheless, as our benchmark infrastructure currently relies on
instruction counts. I couldn't reproduce this by writing my own no-op
`trace _ a = a; {-# NOINLINE trace #-}`, btw.
I checked this on GHC 8.2.2, 8.4.3 and a semi-recent HEAD commit
(bb539cfe335eeec7989332b859b1f3162c5e105a).
--
--
Ticket URL: <http://ghc.haskell.org/trac/ghc/ticket/15333#comment:3>
GHC <http://www.haskell.org/ghc/>
The Glasgow Haskell Compiler
More information about the ghc-tickets
mailing list