In a fit of stupidity I worked on porting OCaml to PPC64: http://git.infradead.org/?p=users/dwmw2/ocaml-ppc64.git
It seems to work fine, mostly -- but when building large programs such as CIL I found that I overflow the TOC -- I'm using more than 64KiB. An example of this problem is at http://david.woodhou.se/ocaml-test.tar.gz: $ make as -a64 -o cil.o camlasm6944b7.s ld -melf64ppc -r -o foo *.o *.a ld: TOC section size exceeds 64k It shows the assembler output from the OCaml compiler for one file, as well as the resulting link error. I looked at what GCC does for -mminimal-toc, and I'm slightly confused. I don't understand why it still has [EMAIL PROTECTED] in the function descriptor and then uses double-indirect access from the global table. What's the point in having a TOC pointer in function descriptors if you're just going to use the same global table all the time anyway? Why not change the function descriptor to point r2 to its own table? For OCaml, I was going to leave the TOC empty and emit something like a .toc1 section as GCC does -- but I was going to just leave the 'real' .toc section empty, and put the address of the local table into the function descriptors for each function. But I'm slightly concerned about doing that without knowing why the C compiler doesn't do it that way. For example, when I compile 'int a; int main(void) { return a; }' with -mminimal-toc why do I see this: main: .quad .L.main,[EMAIL PROTECTED] .previous .type main, @function .L.main: std 30,-16(1) ld 30,[EMAIL PROTECTED](2) ld 9,.LC0-.LCTOC1(30) ... and not this: main: .quad .L.main,.LCTOC1 .previous .type main, @function .L.main: ld 9,.LC0-.LCTOC1(2) OCaml _seems_ to work with the following patch, albeit with lots of bitching about 'unexpected reloc type 38 in .opd section' from the linker. Is there a better way to fix it? diff --git a/asmcomp/power64/emit.mlp b/asmcomp/power64/emit.mlp index 73f4680..1e7824b 100644 --- a/asmcomp/power64/emit.mlp +++ b/asmcomp/power64/emit.mlp @@ -77,7 +77,7 @@ let emit_label lbl = let toc_space = match Config.system with - | "elf" | "bsd" -> " .section \".toc\",\"aw\"\n" + | "elf" | "bsd" -> " .section \".toc1\",\"aw\"\n" | "rhapsody" -> " .toc\n" | _ -> assert false @@ -183,7 +183,7 @@ let emit_tocentry entry = let emit_tocref entry = let lbl = tocref_label (!tocref_entries,entry) in - emit_label lbl; emit_string "@toc(2) #"; emit_tocentry entry + emit_label lbl; emit_string "-.TOCBASE(2) #"; emit_tocentry entry (* Output a load or store operation *) @@ -846,7 +846,7 @@ let fundecl fundecl = ` .align 3\n`; ` .type {emit_symbol fundecl.fun_name}, @function\n`; `{emit_symbol fundecl.fun_name}:\n`; - ` .quad .L.{emit_symbol fundecl.fun_name},[EMAIL PROTECTED]; + ` .quad .L.{emit_symbol fundecl.fun_name},.TOCBASE\n`; ` .previous\n`; ` .align 2\n`; emit_string code_space; @@ -951,8 +951,9 @@ let end_assembly() = (List.rev !jumptbl_entries); jumptbl_entries := [] end; + emit_string toc_space; + `.TOCBASE = . + 32768\n`; if !tocref_entries <> [] then begin - emit_string toc_space; List.iter (fun (lbl, entry) -> `{emit_label lbl}:\n`; @@ -960,7 +961,7 @@ let end_assembly() = TocFloat f -> ` .double {emit_tocentry entry}\n` | _ -> - ` .tc {emit_label lbl}[TC],{emit_tocentry entry}\n` + ` .quad {emit_tocentry entry}\n` ) !tocref_entries; tocref_entries := [] -- dwmw2 _______________________________________________ Linuxppc-dev mailing list Linuxppc-dev@ozlabs.org https://ozlabs.org/mailman/listinfo/linuxppc-dev