| 
									
										
										
										
											2009-07-01 19:13:45 -04:00
										 |  |  | ! Copyright (C) 2005, 2009 Slava Pestov. | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | ! See http://factorcode.org/license.txt for BSD license. | 
					
						
							| 
									
										
										
										
											2009-09-28 05:51:53 -04:00
										 |  |  | USING: locals alien.c-types alien.libraries alien.syntax arrays | 
					
						
							|  |  |  | kernel fry math namespaces sequences system layouts io | 
					
						
							|  |  |  | vocabs.loader accessors init combinators command-line make | 
					
						
							|  |  |  | compiler compiler.units compiler.constants compiler.alien | 
					
						
							|  |  |  | compiler.codegen compiler.codegen.fixup | 
					
						
							|  |  |  | compiler.cfg.instructions compiler.cfg.builder | 
					
						
							|  |  |  | compiler.cfg.intrinsics compiler.cfg.stack-frame | 
					
						
							|  |  |  | cpu.x86.assembler cpu.x86.assembler.operands cpu.x86 | 
					
						
							|  |  |  | cpu.architecture ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | IN: cpu.x86.32 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ! We implement the FFI for Linux, OS X and Windows all at once. | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  | ! OS X requires that the stack be 16-byte aligned. | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-07 21:00:38 -04:00
										 |  |  | M: x86.32 machine-registers | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         { int-regs { EAX ECX EDX EBP EBX } } | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  |         { float-regs { XMM0 XMM1 XMM2 XMM3 XMM4 XMM5 XMM6 XMM7 } } | 
					
						
							| 
									
										
										
										
											2008-10-07 21:00:38 -04:00
										 |  |  |     } ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-02 18:07:38 -04:00
										 |  |  | M: x86.32 ds-reg ESI ;
 | 
					
						
							|  |  |  | M: x86.32 rs-reg EDI ;
 | 
					
						
							|  |  |  | M: x86.32 stack-reg ESP ;
 | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  | M: x86.32 temp-reg ECX ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-15 23:07:03 -04:00
										 |  |  | M: x86.32 %mark-card | 
					
						
							|  |  |  |     drop HEX: ffffffff [+] card-mark <byte> MOV | 
					
						
							|  |  |  |     building get pop
 | 
					
						
							|  |  |  |     rc-absolute-cell rel-cards-offset | 
					
						
							|  |  |  |     building get push ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: x86.32 %mark-deck | 
					
						
							|  |  |  |     drop HEX: ffffffff [+] card-mark <byte> MOV | 
					
						
							|  |  |  |     building get pop
 | 
					
						
							|  |  |  |     rc-absolute-cell rel-decks-offset | 
					
						
							|  |  |  |     building get push ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-29 02:39:14 -04:00
										 |  |  | M:: x86.32 %dispatch ( src temp -- )
 | 
					
						
							| 
									
										
										
										
											2008-11-13 05:16:08 -05:00
										 |  |  |     ! Load jump table base. | 
					
						
							| 
									
										
										
										
											2009-06-30 17:47:22 -04:00
										 |  |  |     temp src HEX: ffffffff [+] LEA | 
					
						
							| 
									
										
										
										
											2009-06-30 19:11:15 -04:00
										 |  |  |     building get length cell - :> start
 | 
					
						
							| 
									
										
										
										
											2009-05-29 02:39:14 -04:00
										 |  |  |     0 rc-absolute-cell rel-here | 
					
						
							| 
									
										
										
										
											2008-11-13 05:16:08 -05:00
										 |  |  |     ! Go | 
					
						
							| 
									
										
										
										
											2009-06-30 17:47:22 -04:00
										 |  |  |     temp HEX: 7f [+] JMP | 
					
						
							| 
									
										
										
										
											2009-06-30 19:11:15 -04:00
										 |  |  |     building get length :> end | 
					
						
							| 
									
										
										
										
											2008-11-13 05:16:08 -05:00
										 |  |  |     ! Fix up the displacement above | 
					
						
							|  |  |  |     cell code-alignment | 
					
						
							| 
									
										
										
										
											2009-06-30 19:11:15 -04:00
										 |  |  |     [ end start - + building get dup pop* push ] | 
					
						
							| 
									
										
										
										
											2008-11-13 05:16:08 -05:00
										 |  |  |     [ align-code ] | 
					
						
							|  |  |  |     bi ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-28 06:33:58 -05:00
										 |  |  | ! Registers for fastcall | 
					
						
							| 
									
										
										
										
											2009-09-25 13:58:55 -04:00
										 |  |  | : param-reg-1 ( -- reg ) EAX ;
 | 
					
						
							|  |  |  | : param-reg-2 ( -- reg ) EDX ;
 | 
					
						
							| 
									
										
										
										
											2008-11-28 06:33:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-06 20:22:22 -04:00
										 |  |  | M: x86.32 pic-tail-reg EBX ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-07 21:33:32 -05:00
										 |  |  | M: x86.32 reserved-area-size 0 ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-06 17:14:53 -04:00
										 |  |  | M: x86.32 %alien-invoke 0 CALL rc-relative rel-dlsym ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-22 05:21:32 -04:00
										 |  |  | : push-vm-ptr ( -- )
 | 
					
						
							| 
									
										
										
										
											2009-10-14 20:24:23 -04:00
										 |  |  |     0 PUSH 0 rc-absolute-cell rel-vm ; ! push the vm ptr as an argument | 
					
						
							| 
									
										
										
										
											2009-08-22 05:21:32 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-12 07:25:07 -05:00
										 |  |  | M: x86.32 return-struct-in-registers? ( c-type -- ? )
 | 
					
						
							|  |  |  |     c-type | 
					
						
							|  |  |  |     [ return-in-registers?>> ] | 
					
						
							|  |  |  |     [ heap-size { 1 2 4 8 } member? ] bi
 | 
					
						
							|  |  |  |     os { linux netbsd solaris } member? not
 | 
					
						
							| 
									
										
										
										
											2009-02-12 09:10:21 -05:00
										 |  |  |     and or ;
 | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-06 01:20:00 -04:00
										 |  |  | : struct-return@ ( n -- operand )
 | 
					
						
							|  |  |  |     [ next-stack@ ] [ stack-frame get params>> stack@ ] if* ;
 | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | ! On x86, parameters are never passed in registers. | 
					
						
							|  |  |  | M: int-regs return-reg drop EAX ;
 | 
					
						
							|  |  |  | M: int-regs param-regs drop { } ;
 | 
					
						
							|  |  |  | M: float-regs param-regs drop { } ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  | GENERIC: push-return-reg ( rep -- )
 | 
					
						
							|  |  |  | GENERIC: load-return-reg ( n rep -- )
 | 
					
						
							|  |  |  | GENERIC: store-return-reg ( n rep -- )
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  | M: int-rep push-return-reg drop EAX PUSH ;
 | 
					
						
							|  |  |  | M: int-rep load-return-reg drop EAX swap next-stack@ MOV ;
 | 
					
						
							|  |  |  | M: int-rep store-return-reg drop stack@ EAX MOV ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-03 21:58:56 -04:00
										 |  |  | M: float-rep push-return-reg drop ESP 4 SUB ESP [] FSTPS ;
 | 
					
						
							|  |  |  | M: float-rep load-return-reg drop next-stack@ FLDS ;
 | 
					
						
							|  |  |  | M: float-rep store-return-reg drop stack@ FSTPS ;
 | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-03 21:58:56 -04:00
										 |  |  | M: double-rep push-return-reg drop ESP 8 SUB ESP [] FSTPL ;
 | 
					
						
							|  |  |  | M: double-rep load-return-reg drop next-stack@ FLDL ;
 | 
					
						
							|  |  |  | M: double-rep store-return-reg drop stack@ FSTPL ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : align-sub ( n -- )
 | 
					
						
							| 
									
										
										
										
											2008-10-06 01:20:00 -04:00
										 |  |  |     [ align-stack ] keep - decr-stack-reg ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : align-add ( n -- )
 | 
					
						
							| 
									
										
										
										
											2008-10-06 01:20:00 -04:00
										 |  |  |     align-stack incr-stack-reg ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : with-aligned-stack ( n quot -- )
 | 
					
						
							| 
									
										
										
										
											2009-07-01 19:13:45 -04:00
										 |  |  |     '[ align-sub @ ] [ align-add ] bi ; inline
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-21 04:21:29 -04:00
										 |  |  | M: x86.32 %prologue ( n -- )
 | 
					
						
							|  |  |  |     dup PUSH | 
					
						
							|  |  |  |     0 PUSH rc-absolute-cell rel-this | 
					
						
							| 
									
										
										
										
											2009-07-01 19:13:45 -04:00
										 |  |  |     3 cells - decr-stack-reg ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  | M: x86.32 %load-param-reg 3drop ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  | M: x86.32 %save-param-reg 3drop ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  | : (%box) ( n rep -- )
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     #! If n is f, push the return register onto the stack; we | 
					
						
							|  |  |  |     #! are boxing a return value of a C function. If n is an | 
					
						
							|  |  |  |     #! integer, push [ESP+n] on the stack; we are boxing a | 
					
						
							|  |  |  |     #! parameter being passed to a callback from C. | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  |     over [ load-return-reg ] [ 2drop ] if ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  | M:: x86.32 %box ( n rep func -- )
 | 
					
						
							|  |  |  |     n rep (%box) | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  |     rep rep-size cell + [ | 
					
						
							| 
									
										
										
										
											2009-08-22 05:21:32 -04:00
										 |  |  |         push-vm-ptr | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  |         rep push-return-reg | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  |         func f %alien-invoke | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     ] with-aligned-stack ;
 | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2008-06-08 16:32:55 -04:00
										 |  |  | : (%box-long-long) ( n -- )
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     [ | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  |         EDX over next-stack@ MOV | 
					
						
							|  |  |  |         EAX swap cell - next-stack@ MOV  | 
					
						
							|  |  |  |     ] when* ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-02 18:07:38 -04:00
										 |  |  | M: x86.32 %box-long-long ( n func -- )
 | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  |     [ (%box-long-long) ] dip
 | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  |     12 [ | 
					
						
							| 
									
										
										
										
											2009-09-02 05:45:03 -04:00
										 |  |  |         push-vm-ptr | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  |         EDX PUSH | 
					
						
							|  |  |  |         EAX PUSH | 
					
						
							|  |  |  |         f %alien-invoke | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     ] with-aligned-stack ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  | M:: x86.32 %box-large-struct ( n c-type -- )
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     ! Compute destination address | 
					
						
							| 
									
										
										
										
											2009-09-02 05:45:03 -04:00
										 |  |  |     EDX n struct-return@ LEA | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  |     12 [ | 
					
						
							| 
									
										
										
										
											2009-09-02 05:45:03 -04:00
										 |  |  |         push-vm-ptr | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |         ! Push struct size | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  |         c-type heap-size PUSH | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |         ! Push destination address | 
					
						
							| 
									
										
										
										
											2009-09-02 05:45:03 -04:00
										 |  |  |         EDX PUSH | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |         ! Copy the struct from the C stack | 
					
						
							|  |  |  |         "box_value_struct" f %alien-invoke | 
					
						
							|  |  |  |     ] with-aligned-stack ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-06 01:20:00 -04:00
										 |  |  | M: x86.32 %prepare-box-struct ( -- )
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     ! Compute target address for value struct return | 
					
						
							| 
									
										
										
										
											2008-10-06 01:20:00 -04:00
										 |  |  |     EAX f struct-return@ LEA | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     ! Store it as the first parameter | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  |     0 stack@ EAX MOV ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  | M: x86.32 %box-small-struct ( c-type -- )
 | 
					
						
							|  |  |  |     #! Box a <= 8-byte struct returned in EAX:EDX. OS X only. | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  |     16 [ | 
					
						
							| 
									
										
										
										
											2009-09-02 05:45:03 -04:00
										 |  |  |         push-vm-ptr | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  |         heap-size PUSH | 
					
						
							|  |  |  |         EDX PUSH | 
					
						
							|  |  |  |         EAX PUSH | 
					
						
							|  |  |  |         "box_small_struct" f %alien-invoke | 
					
						
							|  |  |  |     ] with-aligned-stack ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: x86.32 %prepare-unbox ( -- )
 | 
					
						
							|  |  |  |     #! Move top of data stack to EAX. | 
					
						
							|  |  |  |     EAX ESI [] MOV | 
					
						
							|  |  |  |     ESI 4 SUB ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  | : call-unbox-func ( func -- )
 | 
					
						
							| 
									
										
										
										
											2009-08-21 16:24:53 -04:00
										 |  |  |     8 [ | 
					
						
							| 
									
										
										
										
											2009-08-22 05:21:32 -04:00
										 |  |  |         ! push the vm ptr as an argument | 
					
						
							|  |  |  |         push-vm-ptr | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  |         ! Push parameter | 
					
						
							|  |  |  |         EAX PUSH | 
					
						
							|  |  |  |         ! Call the unboxer | 
					
						
							|  |  |  |         f %alien-invoke | 
					
						
							|  |  |  |     ] with-aligned-stack ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  | M: x86.32 %unbox ( n rep func -- )
 | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  |     #! The value being unboxed must already be in EAX. | 
					
						
							|  |  |  |     #! If n is f, we're unboxing a return value about to be | 
					
						
							|  |  |  |     #! returned by the callback. Otherwise, we're unboxing | 
					
						
							|  |  |  |     #! a parameter to a C function about to be called. | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  |     call-unbox-func | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  |     ! Store the return value on the C stack | 
					
						
							|  |  |  |     over [ store-return-reg ] [ 2drop ] if ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: x86.32 %unbox-long-long ( n func -- )
 | 
					
						
							| 
									
										
										
										
											2009-08-07 18:44:50 -04:00
										 |  |  |     call-unbox-func | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  |     ! Store the return value on the C stack | 
					
						
							|  |  |  |     [ | 
					
						
							|  |  |  |         dup stack@ EAX MOV | 
					
						
							|  |  |  |         cell + stack@ EDX MOV | 
					
						
							|  |  |  |     ] when* ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : %unbox-struct-1 ( -- )
 | 
					
						
							|  |  |  |     #! Alien must be in EAX. | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  |     8 [ | 
					
						
							| 
									
										
										
										
											2009-09-02 05:45:03 -04:00
										 |  |  |         push-vm-ptr | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  |         EAX PUSH | 
					
						
							|  |  |  |         "alien_offset" f %alien-invoke | 
					
						
							|  |  |  |         ! Load first cell | 
					
						
							|  |  |  |         EAX EAX [] MOV | 
					
						
							|  |  |  |     ] with-aligned-stack ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : %unbox-struct-2 ( -- )
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     #! Alien must be in EAX. | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  |     8 [ | 
					
						
							| 
									
										
										
										
											2009-09-02 05:45:03 -04:00
										 |  |  |         push-vm-ptr | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |         EAX PUSH | 
					
						
							|  |  |  |         "alien_offset" f %alien-invoke | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  |         ! Load second cell | 
					
						
							|  |  |  |         EDX EAX 4 [+] MOV | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |         ! Load first cell | 
					
						
							|  |  |  |         EAX EAX [] MOV | 
					
						
							|  |  |  |     ] with-aligned-stack ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  | M: x86 %unbox-small-struct ( size -- )
 | 
					
						
							|  |  |  |     #! Alien must be in EAX. | 
					
						
							|  |  |  |     heap-size cell align cell /i { | 
					
						
							|  |  |  |         { 1 [ %unbox-struct-1 ] } | 
					
						
							|  |  |  |         { 2 [ %unbox-struct-2 ] } | 
					
						
							|  |  |  |     } case ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-25 20:38:48 -04:00
										 |  |  | M:: x86.32 %unbox-large-struct ( n c-type -- )
 | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  |     ! Alien must be in EAX. | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  |     ! Compute destination address | 
					
						
							| 
									
										
										
										
											2009-09-02 05:45:03 -04:00
										 |  |  |     EDX n stack@ LEA | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  |     16 [ | 
					
						
							| 
									
										
										
										
											2009-09-02 05:45:03 -04:00
										 |  |  |         push-vm-ptr | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  |         ! Push struct size | 
					
						
							| 
									
										
										
										
											2009-08-25 20:38:48 -04:00
										 |  |  |         c-type heap-size PUSH | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  |         ! Push destination address | 
					
						
							| 
									
										
										
										
											2009-09-02 05:45:03 -04:00
										 |  |  |         EDX PUSH | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  |         ! Push source address | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |         EAX PUSH | 
					
						
							| 
									
										
										
										
											2008-09-13 21:28:13 -04:00
										 |  |  |         ! Copy the struct to the stack | 
					
						
							|  |  |  |         "to_value_struct" f %alien-invoke | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     ] with-aligned-stack ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-25 14:32:08 -04:00
										 |  |  | M: x86.32 %nest-stacks ( -- )
 | 
					
						
							| 
									
										
										
										
											2009-10-17 22:20:08 -04:00
										 |  |  |     ! Save current frame. See comment in vm/contexts.hpp | 
					
						
							|  |  |  |     EAX stack-reg stack-frame get total-size>> 3 cells - [+] LEA | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  |     8 [ | 
					
						
							| 
									
										
										
										
											2009-09-25 14:32:08 -04:00
										 |  |  |         push-vm-ptr | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  |         EAX PUSH | 
					
						
							| 
									
										
										
										
											2009-09-25 14:32:08 -04:00
										 |  |  |         "nest_stacks" f %alien-invoke | 
					
						
							|  |  |  |     ] with-aligned-stack ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: x86.32 %unnest-stacks ( -- )
 | 
					
						
							|  |  |  |     4 [ | 
					
						
							|  |  |  |         push-vm-ptr | 
					
						
							|  |  |  |         "unnest_stacks" f %alien-invoke | 
					
						
							|  |  |  |     ] with-aligned-stack ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-02 18:07:38 -04:00
										 |  |  | M: x86.32 %prepare-alien-indirect ( -- )
 | 
					
						
							| 
									
										
										
										
											2009-10-16 22:07:09 -04:00
										 |  |  |     4 [ | 
					
						
							|  |  |  |         push-vm-ptr | 
					
						
							|  |  |  |         "unbox_alien" f %alien-invoke | 
					
						
							|  |  |  |     ] with-aligned-stack | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  |     EBP EAX MOV ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-02 18:07:38 -04:00
										 |  |  | M: x86.32 %alien-indirect ( -- )
 | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  |     EBP CALL ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-02 18:07:38 -04:00
										 |  |  | M: x86.32 %alien-callback ( quot -- )
 | 
					
						
							| 
									
										
										
										
											2009-10-16 22:07:09 -04:00
										 |  |  |     param-reg-1 swap %load-reference | 
					
						
							|  |  |  |     param-reg-2 %mov-vm-ptr | 
					
						
							|  |  |  |     "c_to_factor" f %alien-invoke ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-02 18:07:38 -04:00
										 |  |  | M: x86.32 %callback-value ( ctype -- )
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     ! Align C stack | 
					
						
							|  |  |  |     ESP 12 SUB | 
					
						
							| 
									
										
										
										
											2008-10-05 22:30:29 -04:00
										 |  |  |     ! Save top of data stack in non-volatile register | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     %prepare-unbox | 
					
						
							|  |  |  |     EAX PUSH | 
					
						
							| 
									
										
										
										
											2009-08-23 15:04:05 -04:00
										 |  |  |     push-vm-ptr | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     ! Restore data/call/retain stacks | 
					
						
							|  |  |  |     "unnest_stacks" f %alien-invoke | 
					
						
							|  |  |  |     ! Place top of data stack in EAX | 
					
						
							| 
									
										
										
										
											2009-08-23 14:40:59 -04:00
										 |  |  |     temp-reg POP | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     EAX POP | 
					
						
							|  |  |  |     ! Restore C stack | 
					
						
							|  |  |  |     ESP 12 ADD | 
					
						
							|  |  |  |     ! Unbox EAX | 
					
						
							|  |  |  |     unbox-return ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-27 20:28:20 -04:00
										 |  |  | GENERIC: float-function-param ( stack-slot dst src -- )
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M:: spill-slot float-function-param ( stack-slot dst src -- )
 | 
					
						
							| 
									
										
										
										
											2009-09-27 19:06:30 -04:00
										 |  |  |     ! We can clobber dst here since its going to contain the | 
					
						
							|  |  |  |     ! final result | 
					
						
							| 
									
										
										
										
											2009-09-27 20:28:20 -04:00
										 |  |  |     dst src double-rep %copy | 
					
						
							|  |  |  |     stack-slot dst double-rep %copy ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: register float-function-param | 
					
						
							|  |  |  |     nip double-rep %copy ;
 | 
					
						
							| 
									
										
										
										
											2009-09-27 19:06:30 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : float-function-return ( reg -- )
 | 
					
						
							|  |  |  |     ESP [] FSTPL | 
					
						
							|  |  |  |     ESP [] MOVSD | 
					
						
							|  |  |  |     ESP 16 ADD ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M:: x86.32 %unary-float-function ( dst src func -- )
 | 
					
						
							|  |  |  |     ESP -16 [+] dst src float-function-param | 
					
						
							|  |  |  |     ESP 16 SUB | 
					
						
							| 
									
										
										
										
											2009-09-28 05:51:53 -04:00
										 |  |  |     func "libm" load-library %alien-invoke | 
					
						
							| 
									
										
										
										
											2009-09-27 19:06:30 -04:00
										 |  |  |     dst float-function-return ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M:: x86.32 %binary-float-function ( dst src1 src2 func -- )
 | 
					
						
							|  |  |  |     ESP -16 [+] dst src1 float-function-param | 
					
						
							|  |  |  |     ESP  -8 [+] dst src2 float-function-param | 
					
						
							|  |  |  |     ESP 16 SUB | 
					
						
							| 
									
										
										
										
											2009-09-28 05:51:53 -04:00
										 |  |  |     func "libm" load-library %alien-invoke | 
					
						
							| 
									
										
										
										
											2009-09-27 19:06:30 -04:00
										 |  |  |     dst float-function-return ;
 | 
					
						
							| 
									
										
										
										
											2009-09-25 14:32:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-13 00:32:14 -04:00
										 |  |  | M: x86.32 %cleanup ( params -- )
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     #! a) If we just called an stdcall function in Windows, it | 
					
						
							|  |  |  |     #! cleaned up the stack frame for us. But we don't want that | 
					
						
							|  |  |  |     #! so we 'undo' the cleanup since we do that in %epilogue. | 
					
						
							|  |  |  |     #! b) If we just called a function returning a struct, we | 
					
						
							|  |  |  |     #! have to fix ESP. | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2008-03-20 21:13:13 -04:00
										 |  |  |             [ dup abi>> "stdcall" = ] | 
					
						
							| 
									
										
										
										
											2008-10-06 01:20:00 -04:00
										 |  |  |             [ drop ESP stack-frame get params>> SUB ] | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |         } { | 
					
						
							| 
									
										
										
										
											2008-03-20 21:13:13 -04:00
										 |  |  |             [ dup return>> large-struct? ] | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |             [ drop EAX PUSH ] | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-04-11 13:53:22 -04:00
										 |  |  |         [ drop ] | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  |     } cond ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-13 00:32:14 -04:00
										 |  |  | M: x86.32 %callback-return ( n -- )
 | 
					
						
							|  |  |  |     #! a) If the callback is stdcall, we have to clean up the | 
					
						
							|  |  |  |     #! caller's stack frame. | 
					
						
							|  |  |  |     #! b) If the callback is returning a large struct, we have | 
					
						
							|  |  |  |     #! to fix ESP. | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2008-10-19 02:10:45 -04:00
										 |  |  |         { [ dup abi>> "stdcall" = ] [ | 
					
						
							|  |  |  |             <alien-stack-frame> | 
					
						
							|  |  |  |             [ params>> ] [ return>> ] bi +
 | 
					
						
							|  |  |  |         ] } | 
					
						
							| 
									
										
										
										
											2008-10-13 00:32:14 -04:00
										 |  |  |         { [ dup return>> large-struct? ] [ drop 4 ] } | 
					
						
							|  |  |  |         [ drop 0 ] | 
					
						
							|  |  |  |     } cond RET ;
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-25 22:47:05 -04:00
										 |  |  | M:: x86.32 %call-gc ( gc-root-count temp -- )
 | 
					
						
							|  |  |  |     temp gc-root-base param@ LEA | 
					
						
							| 
									
										
										
										
											2009-09-25 13:29:07 -04:00
										 |  |  |     12 [ | 
					
						
							| 
									
										
										
										
											2009-09-25 22:47:05 -04:00
										 |  |  |         ! Pass the VM ptr as the third parameter | 
					
						
							| 
									
										
										
										
											2009-10-14 03:06:01 -04:00
										 |  |  |         push-vm-ptr | 
					
						
							| 
									
										
										
										
											2009-09-25 13:29:07 -04:00
										 |  |  |         ! Pass number of roots as second parameter | 
					
						
							| 
									
										
										
										
											2009-09-25 13:48:13 -04:00
										 |  |  |         gc-root-count PUSH  | 
					
						
							| 
									
										
										
										
											2009-09-25 13:29:07 -04:00
										 |  |  |         ! Pass pointer to start of GC roots as first parameter | 
					
						
							| 
									
										
										
										
											2009-09-25 22:47:05 -04:00
										 |  |  |         temp PUSH  | 
					
						
							| 
									
										
										
										
											2009-09-25 13:29:07 -04:00
										 |  |  |         ! Call GC | 
					
						
							|  |  |  |         "inline_gc" f %alien-invoke | 
					
						
							|  |  |  |     ] with-aligned-stack ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-08 22:40:47 -05:00
										 |  |  | M: x86.32 dummy-stack-params? f ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: x86.32 dummy-int-params? f ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: x86.32 dummy-fp-params? f ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-02 19:25:33 -04:00
										 |  |  | os windows? [ | 
					
						
							| 
									
										
										
										
											2008-08-31 08:45:33 -04:00
										 |  |  |     cell "longlong" c-type (>>align) | 
					
						
							|  |  |  |     cell "ulonglong" c-type (>>align) | 
					
						
							|  |  |  |     4 "double" c-type (>>align) | 
					
						
							| 
									
										
										
										
											2008-02-11 17:56:48 -05:00
										 |  |  | ] unless
 | 
					
						
							| 
									
										
										
										
											2008-02-04 18:33:59 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-20 18:43:16 -04:00
										 |  |  | check-sse |