[m-dev.] Bug 321

Paul Bone paul at bone.id.au
Fri Feb 21 18:16:52 AEDT 2014


Hi.

I'm tracking down bug 321.  https://www.mercurylang.org/bugs/view.php?id=321

As you recall this is the symptom:

    The C linker gives me errors such as:

    paul at oxygen> mmc --rebuild test -O2 --intermodule-optimization
    Making Mercury/int3s/test.int3
    Making Mercury/ints/test.int
    Making Mercury/opts/test.opt
    Making Mercury/cs/test.c
    Making Mercury/os/test.o
    Making test
    ** Error making `test'.
    Mercury/os/test.o: In function `<predicate 'test.try_divide'/3 mode
    0>':
    test.c:(.text+0xe6): undefined reference to `<predicate
    'exception.wrap_success_or_failure'/2 mode 0>'
    collect2: error: ld returned 1 exit status

After looking at the exception.m code and the HLDS dumps during compilation
of my test.m I found that:

    The linker can't find an address for: wrap_success_or_failure/2
    But it can find addresses for: get_determinism/2

Both predicates are called by the same code in the HLDS representation of
test.m but one works and what doesn't.

So I looked at readelf(1) dump of libmer_std.so's symbol table.
get_determinism/2 is exported globally and wrap_success_or_failure/2 is
exported locally.
Both are declared and defined in the implementation section of the exception
module.  So now I want to know why is one being treated differently.

The relevant code seems to be on line 2186 of compiler/intermod.m, where the
compiler performs a fixpoint analysis so that it knows what else needs to be
opt-exported.  Searching further (line 289) I find should_be_processed/8
which appears to determine if a predicate should be opt-exported or not,
I'm not sure as comments are scarce.

Is there anyone familiar with this part of the compiler?  My clues are the
above, plus wrap_success_or_failure takes a higher order argument and is
part of a lambda expression itself, during compilation (in the HLDS) it
appears to be partially applied.

One specific question I have is how does the compiler decide between:
    + opt-exporting a predicate
    + making a predicate available to another module because a caller is
      opt-exported and this predicate is otherwise local.
    + leaving the predicate local.

I haven't found the code for the second case.

Thanks.


-- 
Paul Bone



More information about the developers mailing list