On Thu, Nov 1, 2018 at 2:11 AM silvioprog <silviop...@gmail.com> wrote: [...]
> I took a look at some System V ABI manuals to start a draft based on them, > adapting the assembly to the InvokeKernelWin64() signature idea. The > draft works fine for six or more arguments and returns the function value > too, but I need to check (probably next weekend) how to pass floating-point > values to the XMM registers (I'm looking for references/manuals about). > Finally, I found a awesome material to study: the GCC's X86-64 assembly for compiler writers. And after some successful tests, I concluded that: - floating-point types with size 8 or less (double, single etc.) must be set in XMM registers aligned in 8; - floating-point types larger than 8 (extended) must be pushed to the stack aligned in 10. This is a small example to check it (environment used in the tests: Lazarus 2.1.0 r59363 FPC 3.1.1 x86_64-linux-gtk2): === program project1; {$MODE DELPHI} {$MACRO ON} //{$DEFINE USE_EXTENDED} {$IFDEF USE_EXTENDED} {$DEFINE FLOAT_TYPE := extended} // size 10 {$ELSE} {$DEFINE FLOAT_TYPE := double} // size 8 {$ENDIF} uses sysutils; procedure test(const a, b: FLOAT_TYPE); begin writeln(a:0:2, ' - ', b:0:2); end; procedure call_test(arr: pointer; f: codepointer); assembler; nostackframe; asm { save the base pointer } pushq %rbp { set new base pointer } movq %rsp, %rbp { save callee-saved registers } pushq %rbx pushq %r12 pushq %r13 pushq %r14 pushq %r15 {$IFDEF USE_EXTENDED} leaq 0(%rdi), %rdx movq (%rdx), %rax movq %rax, (%rsp) movq 0x8(%rdx), %ax movq %ax, 0x8(%rsp) leaq 10(%rdi), %rax movq (%rax), %rdx movq %rdx, 0x10(%rsp) movq 0x8(%rax), %ax movq %ax, 0x18(%rsp) {$ELSE} movq 0(%rdi), %xmm0 movq 8(%rdi), %xmm1 {$ENDIF} { call the function } callq *%rsi { restore callee-saved registers } popq %r15 popq %r14 popq %r13 popq %r12 popq %rbx { reset stack to base pointer } movq %rbp, %rsp { restore the old base pointer } popq %rbp { return to caller } retq end; var arr: array of FLOAT_TYPE; begin Writeln(IntToStr(SizeOf(FLOAT_TYPE))); {$PUSH}{$WARN 5090 OFF} setlength(arr, 2); {$POP} arr[0] := 12.34; arr[1] := 45.67; call_test(@arr[0], @test); end. === and the results was: //{$DEFINE USE_EXTENDED}: === ... 8 12.34 - 45.67 === {$DEFINE USE_EXTENDED}: === ... 10 12.34 - 45.67 === I'm going to clone the Free Pascal repository from Github to work on this feature (sysv ABI for Linux 64 for now). After some functional code I'll send it as patch to the Mantis issues. I'm very interested in rtti.invoke() support on Linux/ARM and Win32. :-) (P.S.: Could you check the issue #34496? I'm not sure if saving the base pointer is best way to solve the problem, but at least fixed the AV for extended types) -- Silvio Clécio
_______________________________________________ fpc-pascal maillist - fpc-pascal@lists.freepascal.org http://lists.freepascal.org/cgi-bin/mailman/listinfo/fpc-pascal