diff --git a/.gitignore b/.gitignore index 290f075aae..f4334f3727 100644 --- a/.gitignore +++ b/.gitignore @@ -20,3 +20,4 @@ temp logs work build-support/wordsize +*.bak diff --git a/Makefile b/Makefile index 973ba1f3d4..ffcbf6364c 100644 --- a/Makefile +++ b/Makefile @@ -161,7 +161,7 @@ factor: $(DLL_OBJS) $(EXE_OBJS) clean: rm -f vm/*.o - rm -f factor*.dll libfactor*.* + rm -f factor*.dll libfactor.{a,so,dylib} vm/resources.o: $(WINDRES) vm/factor.rs vm/resources.o diff --git a/README.txt b/README.txt index 754791aa1a..98616539d2 100755 --- a/README.txt +++ b/README.txt @@ -43,13 +43,10 @@ Compilation will yield an executable named 'factor' on Unix, For X11 support, you need recent development libraries for libc, Freetype, X11, OpenGL and GLUT. On a Debian-derived Linux distribution -(like Ubuntu), you can use the line +(like Ubuntu), you can use the following line to grab everything: sudo apt-get install libc6-dev libfreetype6-dev libx11-dev glutg3-dev -to grab everything (if you're on a non-debian-derived distro please tell -us what the equivalent command is on there and it can be added). - * Bootstrapping the Factor image Once you have compiled the Factor runtime, you must bootstrap the Factor diff --git a/basis/alarms/alarms-docs.factor b/basis/alarms/alarms-docs.factor index dac8b72dd5..2d494afca3 100644 --- a/basis/alarms/alarms-docs.factor +++ b/basis/alarms/alarms-docs.factor @@ -5,7 +5,7 @@ HELP: alarm { $class-description "An alarm. Can be passed to " { $link cancel-alarm } "." } ; HELP: add-alarm -{ $values { "quot" quotation } { "time" timestamp } { "frequency" "a " { $link duration } " or " { $link f } } { "alarm" alarm } } +{ $values { "quot" quotation } { "time" timestamp } { "frequency" { $maybe duration } } { "alarm" alarm } } { $description "Creates and registers an alarm. If " { $snippet "frequency" } " is " { $link f } ", this will be a one-time alarm, otherwise it will fire with the given frequency. The quotation will be called from the alarm thread." } ; HELP: later diff --git a/basis/alarms/alarms.factor b/basis/alarms/alarms.factor index 7fdeca9ae6..ad1838b3df 100644 --- a/basis/alarms/alarms.factor +++ b/basis/alarms/alarms.factor @@ -35,7 +35,7 @@ ERROR: bad-alarm-frequency frequency ; [ time>> ] dip before=? ; : reschedule-alarm ( alarm -- ) - dup [ swap interval>> time+ ] change-time register-alarm ; + dup [ swap interval>> time+ now max ] change-time register-alarm ; : call-alarm ( alarm -- ) [ entry>> box> drop ] diff --git a/basis/alias/alias-docs.factor b/basis/alias/alias-docs.factor index 4dcf1a7738..3f2eee6460 100644 --- a/basis/alias/alias-docs.factor +++ b/basis/alias/alias-docs.factor @@ -16,7 +16,7 @@ HELP: ALIAS: } } ; -ARTICLE: "alias" "Alias" +ARTICLE: "alias" "Word aliasing" "The " { $vocab-link "alias" } " vocabulary implements a way to make many different names for the same word. Although creating new names for words is generally frowned upon, aliases are useful for the Win32 API and other cases where words need to be renamed for symmetry." $nl "Make a new word that aliases another word:" { $subsection define-alias } diff --git a/basis/alien/arrays/arrays-docs.factor b/basis/alien/arrays/arrays-docs.factor index 09a09cdc6f..c5efe1e030 100644 --- a/basis/alien/arrays/arrays-docs.factor +++ b/basis/alien/arrays/arrays-docs.factor @@ -1,69 +1,7 @@ IN: alien.arrays USING: help.syntax help.markup byte-arrays alien.c-types ; -ARTICLE: "c-arrays-factor" "Converting C arrays to and from Factor arrays" -"Each primitive C type has a pair of words, " { $snippet ">" { $emphasis "type" } "-array" } " and " { $snippet { $emphasis "type" } "-array>" } ", for converting an array of Factor objects to and from a " { $link byte-array } " of C values. This set of words consists of:" -{ $subsection >c-bool-array } -{ $subsection >c-char-array } -{ $subsection >c-double-array } -{ $subsection >c-float-array } -{ $subsection >c-int-array } -{ $subsection >c-long-array } -{ $subsection >c-longlong-array } -{ $subsection >c-short-array } -{ $subsection >c-uchar-array } -{ $subsection >c-uint-array } -{ $subsection >c-ulong-array } -{ $subsection >c-ulonglong-array } -{ $subsection >c-ushort-array } -{ $subsection >c-void*-array } -{ $subsection c-bool-array> } -{ $subsection c-char-array> } -{ $subsection c-double-array> } -{ $subsection c-float-array> } -{ $subsection c-int-array> } -{ $subsection c-long-array> } -{ $subsection c-longlong-array> } -{ $subsection c-short-array> } -{ $subsection c-uchar-array> } -{ $subsection c-uint-array> } -{ $subsection c-ulong-array> } -{ $subsection c-ulonglong-array> } -{ $subsection c-ushort-array> } -{ $subsection c-void*-array> } ; - -ARTICLE: "c-arrays-get/set" "Reading and writing elements in C arrays" -"Each C type has a pair of words, " { $snippet { $emphasis "type" } "-nth" } " and " { $snippet "set-" { $emphasis "type" } "-nth" } ", for reading and writing values of this type stored in an array. This set of words includes but is not limited to:" -{ $subsection char-nth } -{ $subsection set-char-nth } -{ $subsection uchar-nth } -{ $subsection set-uchar-nth } -{ $subsection short-nth } -{ $subsection set-short-nth } -{ $subsection ushort-nth } -{ $subsection set-ushort-nth } -{ $subsection int-nth } -{ $subsection set-int-nth } -{ $subsection uint-nth } -{ $subsection set-uint-nth } -{ $subsection long-nth } -{ $subsection set-long-nth } -{ $subsection ulong-nth } -{ $subsection set-ulong-nth } -{ $subsection longlong-nth } -{ $subsection set-longlong-nth } -{ $subsection ulonglong-nth } -{ $subsection set-ulonglong-nth } -{ $subsection float-nth } -{ $subsection set-float-nth } -{ $subsection double-nth } -{ $subsection set-double-nth } -{ $subsection void*-nth } -{ $subsection set-void*-nth } ; - ARTICLE: "c-arrays" "C arrays" "C arrays are allocated in the same manner as other C data; see " { $link "c-byte-arrays" } " and " { $link "malloc" } "." $nl -"C type specifiers for array types are documented in " { $link "c-types-specs" } "." -{ $subsection "c-arrays-factor" } -{ $subsection "c-arrays-get/set" } ; +"C type specifiers for array types are documented in " { $link "c-types-specs" } "." ; diff --git a/basis/alien/arrays/arrays.factor b/basis/alien/arrays/arrays.factor index 94472e8261..727492edb1 100644 --- a/basis/alien/arrays/arrays.factor +++ b/basis/alien/arrays/arrays.factor @@ -8,6 +8,8 @@ UNION: value-type array struct-type ; M: array c-type ; +M: array c-type-class drop object ; + M: array heap-size unclip heap-size [ * ] reduce ; M: array c-type-align first c-type-align ; diff --git a/basis/alien/c-types/c-types-docs.factor b/basis/alien/c-types/c-types-docs.factor index 03208de63a..a2b555b057 100644 --- a/basis/alien/c-types/c-types-docs.factor +++ b/basis/alien/c-types/c-types-docs.factor @@ -39,12 +39,12 @@ HELP: byte-length { $contract "Outputs the size of the byte array or float array data in bytes as presented to the C library interface." } ; HELP: c-getter -{ $values { "name" string } { "quot" "a quotation with stack effect " { $snippet "( c-ptr n -- obj )" } } } +{ $values { "name" string } { "quot" { $quotation "( c-ptr n -- obj )" } } } { $description "Outputs a quotation which reads values of this C type from a C structure." } { $errors "Throws a " { $link no-c-type } " error if the type does not exist." } ; HELP: c-setter -{ $values { "name" string } { "quot" "a quotation with stack effect " { $snippet "( obj c-ptr n -- )" } } } +{ $values { "name" string } { "quot" { $quotation "( obj c-ptr n -- )" } } } { $description "Outputs a quotation which writes values of this C type to a C structure." } { $errors "Throws an error if the type does not exist." } ; @@ -89,16 +89,6 @@ HELP: malloc-byte-array { $warning "Don't forget to deallocate the memory with a call to " { $link free } "." } { $errors "Throws an error if memory allocation fails." } ; -HELP: define-nth -{ $values { "name" "a word name" } { "vocab" "a vocabulary name" } } -{ $description "Defines a word " { $snippet { $emphasis "name" } "-nth" } " with stack effect " { $snippet "( n c-ptr -- value )" } " for reading the value with C type " { $snippet "name" } " stored at an alien pointer, displaced by a multiple of the C type's size." } -{ $notes "This is an internal word called when defining C types, there is no need to call it on your own." } ; - -HELP: define-set-nth -{ $values { "name" "a word name" } { "vocab" "a vocabulary name" } } -{ $description "Defines a word " { $snippet "set-" { $emphasis "name" } "-nth" } " with stack effect " { $snippet "( value n c-ptr -- )" } " for writing the value with C type " { $snippet "name" } " to an alien pointer, displaced by a multiple of the C type's size." } -{ $notes "This is an internal word called when defining C types, there is no need to call it on your own." } ; - HELP: box-parameter { $values { "n" integer } { "ctype" string } } { $description "Generates code for converting a C value stored at offset " { $snippet "n" } " from the top of the stack into a Factor object to be pushed on the data stack." } @@ -115,12 +105,12 @@ HELP: unbox-return { $notes "This is an internal word used by the compiler when compiling callbacks." } ; HELP: define-deref -{ $values { "name" "a word name" } { "vocab" "a vocabulary name" } } +{ $values { "name" "a word name" } } { $description "Defines a word " { $snippet "*name" } " with stack effect " { $snippet "( c-ptr -- value )" } " for reading a value with C type " { $snippet "name" } " stored at an alien pointer." } { $notes "This is an internal word called when defining C types, there is no need to call it on your own." } ; HELP: define-out -{ $values { "name" "a word name" } { "vocab" "a vocabulary name" } } +{ $values { "name" "a word name" } } { $description "Defines a word " { $snippet "<" { $emphasis "name" } ">" } " with stack effect " { $snippet "( value -- array )" } ". This word allocates a byte array large enough to hold a value with C type " { $snippet "name" } ", and writes the value at the top of the stack to the array." } { $notes "This is an internal word called when defining C types, there is no need to call it on your own." } ; @@ -230,9 +220,7 @@ $nl "You can copy a range of bytes from memory into a byte array:" { $subsection memory>byte-array } "You can copy a byte array to memory unsafely:" -{ $subsection byte-array>memory } -"A wrapper for temporarily allocating a block of memory:" -{ $subsection with-malloc } ; +{ $subsection byte-array>memory } ; ARTICLE: "c-data" "Passing data between Factor and C" "Two defining characteristics of Factor are dynamic typing and automatic memory management, which are somewhat incompatible with the machine-level data model exposed by C. Factor's C library interface defines its own set of C data types, distinct from Factor language types, together with automatic conversion between Factor values and C types. For example, C integer types must be declared and are fixed-width, whereas Factor supports arbitrary-precision integers." diff --git a/basis/alien/c-types/c-types-tests.factor b/basis/alien/c-types/c-types-tests.factor index edda9e7fdb..f57d102452 100644 --- a/basis/alien/c-types/c-types-tests.factor +++ b/basis/alien/c-types/c-types-tests.factor @@ -55,4 +55,6 @@ TYPEDEF: uchar* MyLPBYTE 0 B{ 1 2 3 4 } ] must-fail -[ t ] [ { t f t } >c-bool-array { 1 0 1 } >c-int-array = ] unit-test +os windows? cpu x86.64? and [ + [ -2147467259 ] [ 2147500037 *long ] unit-test +] when diff --git a/basis/alien/c-types/c-types.factor b/basis/alien/c-types/c-types.factor index a93c87611d..c3ae644b47 100644 --- a/basis/alien/c-types/c-types.factor +++ b/basis/alien/c-types/c-types.factor @@ -4,7 +4,7 @@ USING: byte-arrays arrays assocs kernel kernel.private libc math namespaces make parser sequences strings words assocs splitting math.parser cpu.architecture alien alien.accessors quotations layouts system compiler.units io.files io.encodings.binary -accessors combinators effects continuations ; +accessors combinators effects continuations fry ; IN: alien.c-types DEFER: @@ -13,13 +13,15 @@ DEFER: *char : little-endian? ( -- ? ) 1 *char 1 = ; foldable TUPLE: c-type +class boxer boxer-quot unboxer unboxer-quot getter setter reg-class size align stack-align? ; : new-c-type ( class -- type ) new - int-regs >>reg-class ; + int-regs >>reg-class + object >>class ; inline : ( -- type ) \ c-type new-c-type ; @@ -50,7 +52,7 @@ GENERIC: c-type ( name -- type ) foldable : parse-array-type ( name -- array ) "[" split unclip - >r [ "]" ?tail drop string>number ] map r> prefix ; + [ [ "]" ?tail drop string>number ] map ] dip prefix ; M: string c-type ( name -- type ) CHAR: ] over member? [ @@ -63,6 +65,12 @@ M: string c-type ( name -- type ) ] ?if ] if ; +GENERIC: c-type-class ( name -- class ) + +M: c-type c-type-class class>> ; + +M: string c-type-class c-type c-type-class ; + GENERIC: c-type-boxer ( name -- boxer ) M: c-type c-type-boxer boxer>> ; @@ -164,7 +172,7 @@ GENERIC: stack-size ( type -- size ) foldable M: string stack-size c-type stack-size ; -M: c-type stack-size size>> ; +M: c-type stack-size size>> cell align ; GENERIC: byte-length ( seq -- n ) flushable @@ -172,12 +180,12 @@ M: byte-array byte-length length ; : c-getter ( name -- quot ) c-type-getter [ - [ "Cannot read struct fields with type" throw ] + [ "Cannot read struct fields with this type" throw ] ] unless* ; : c-setter ( name -- quot ) c-type-setter [ - [ "Cannot write struct fields with type" throw ] + [ "Cannot write struct fields with this type" throw ] ] unless* ; : ( n type -- array ) @@ -193,36 +201,21 @@ M: byte-array byte-length length ; 1 swap malloc-array ; inline : malloc-byte-array ( byte-array -- alien ) - dup length dup malloc [ -rot memcpy ] keep ; + dup length [ nip malloc dup ] 2keep memcpy ; : memory>byte-array ( alien len -- byte-array ) - dup [ -rot memcpy ] keep ; + [ nip dup ] 2keep memcpy ; : byte-array>memory ( byte-array base -- ) swap dup length memcpy ; -: (define-nth) ( word type quot -- ) +: array-accessor ( type quot -- def ) [ \ swap , [ heap-size , [ * >fixnum ] % ] [ % ] bi* - ] [ ] make define-inline ; - -: nth-word ( name vocab -- word ) - >r "-nth" append r> create ; - -: define-nth ( name vocab -- ) - dupd nth-word swap dup c-getter (define-nth) ; - -: set-nth-word ( name vocab -- word ) - >r "set-" swap "-nth" 3append r> create ; - -: define-set-nth ( name vocab -- ) - dupd set-nth-word swap dup c-setter (define-nth) ; + ] [ ] make ; : typedef ( old new -- ) c-types get set-at ; -: define-c-type ( type name vocab -- ) - >r tuck typedef r> [ define-nth ] 2keep define-set-nth ; - TUPLE: long-long-type < c-type ; : ( -- type ) @@ -240,62 +233,34 @@ M: long-long-type box-parameter ( n type -- ) M: long-long-type box-return ( type -- ) f swap box-parameter ; -: define-deref ( name vocab -- ) - >r dup CHAR: * prefix r> create - swap c-getter 0 prefix define-inline ; +: define-deref ( name -- ) + [ CHAR: * prefix "alien.c-types" create ] + [ c-getter 0 prefix ] bi + define-inline ; -: define-out ( name vocab -- ) - over [ tuck 0 ] over c-setter append swap - >r >r constructor-word r> r> prefix define-inline ; +: define-out ( name -- ) + [ "alien.c-types" constructor-word ] + [ dup c-setter '[ _ [ 0 @ ] keep ] ] + bi define-inline ; : c-bool> ( int -- ? ) zero? not ; -: >c-array ( seq type word -- byte-array ) - [ [ dup length ] dip ] dip - [ [ execute ] 2curry each-index ] 2keep drop ; inline - -: >c-array-quot ( type vocab -- quot ) - dupd set-nth-word [ >c-array ] 2curry ; - -: to-array-word ( name vocab -- word ) - >r ">c-" swap "-array" 3append r> create ; - -: define-to-array ( type vocab -- ) - [ to-array-word ] 2keep >c-array-quot - (( array -- byte-array )) define-declared ; - -: c-array>quot ( type vocab -- quot ) - [ - \ swap , - nth-word 1quotation , - [ curry map ] % - ] [ ] make ; - -: from-array-word ( name vocab -- word ) - >r "c-" swap "-array>" 3append r> create ; - -: define-from-array ( type vocab -- ) - [ from-array-word ] 2keep c-array>quot - (( c-ptr n -- array )) define-declared ; - : define-primitive-type ( type name -- ) - "alien.c-types" - { - [ define-c-type ] - [ define-deref ] - [ define-to-array ] - [ define-from-array ] - [ define-out ] - } 2cleave ; + [ typedef ] + [ define-deref ] + [ define-out ] + tri ; : expand-constants ( c-type -- c-type' ) dup array? [ - unclip >r [ - dup word? [ - def>> { } swap with-datastack first - ] when - ] map r> prefix + unclip [ + [ + dup word? [ + def>> { } swap with-datastack first + ] when + ] map + ] dip prefix ] when ; : malloc-file-contents ( path -- alien len ) @@ -304,8 +269,20 @@ M: long-long-type box-return ( type -- ) : if-void ( type true false -- ) pick "void" = [ drop nip call ] [ nip call ] if ; inline +: primitive-types + { + "char" "uchar" + "short" "ushort" + "int" "uint" + "long" "ulong" + "longlong" "ulonglong" + "float" "double" + "void*" "bool" + } ; + [ + c-ptr >>class [ alien-cell ] >>getter [ set-alien-cell ] >>setter bootstrap-cell >>size @@ -315,6 +292,7 @@ M: long-long-type box-return ( type -- ) "void*" define-primitive-type + integer >>class [ alien-signed-8 ] >>getter [ set-alien-signed-8 ] >>setter 8 >>size @@ -324,6 +302,7 @@ M: long-long-type box-return ( type -- ) "longlong" define-primitive-type + integer >>class [ alien-unsigned-8 ] >>getter [ set-alien-unsigned-8 ] >>setter 8 >>size @@ -333,6 +312,7 @@ M: long-long-type box-return ( type -- ) "ulonglong" define-primitive-type + integer >>class [ alien-signed-cell ] >>getter [ set-alien-signed-cell ] >>setter bootstrap-cell >>size @@ -342,6 +322,7 @@ M: long-long-type box-return ( type -- ) "long" define-primitive-type + integer >>class [ alien-unsigned-cell ] >>getter [ set-alien-unsigned-cell ] >>setter bootstrap-cell >>size @@ -351,6 +332,7 @@ M: long-long-type box-return ( type -- ) "ulong" define-primitive-type + integer >>class [ alien-signed-4 ] >>getter [ set-alien-signed-4 ] >>setter 4 >>size @@ -360,6 +342,7 @@ M: long-long-type box-return ( type -- ) "int" define-primitive-type + integer >>class [ alien-unsigned-4 ] >>getter [ set-alien-unsigned-4 ] >>setter 4 >>size @@ -369,6 +352,7 @@ M: long-long-type box-return ( type -- ) "uint" define-primitive-type + fixnum >>class [ alien-signed-2 ] >>getter [ set-alien-signed-2 ] >>setter 2 >>size @@ -378,6 +362,7 @@ M: long-long-type box-return ( type -- ) "short" define-primitive-type + fixnum >>class [ alien-unsigned-2 ] >>getter [ set-alien-unsigned-2 ] >>setter 2 >>size @@ -387,6 +372,7 @@ M: long-long-type box-return ( type -- ) "ushort" define-primitive-type + fixnum >>class [ alien-signed-1 ] >>getter [ set-alien-signed-1 ] >>setter 1 >>size @@ -396,6 +382,7 @@ M: long-long-type box-return ( type -- ) "char" define-primitive-type + fixnum >>class [ alien-unsigned-1 ] >>getter [ set-alien-unsigned-1 ] >>setter 1 >>size @@ -414,6 +401,7 @@ M: long-long-type box-return ( type -- ) "bool" define-primitive-type + float >>class [ alien-float ] >>getter [ [ >float ] 2dip set-alien-float ] >>setter 4 >>size @@ -425,6 +413,7 @@ M: long-long-type box-return ( type -- ) "float" define-primitive-type + float >>class [ alien-double ] >>getter [ [ >float ] 2dip set-alien-double ] >>setter 8 >>size @@ -436,6 +425,6 @@ M: long-long-type box-return ( type -- ) "double" define-primitive-type "long" "ptrdiff_t" typedef - + "long" "intptr_t" typedef "ulong" "size_t" typedef ] with-compilation-unit diff --git a/basis/alien/parser/parser.factor b/basis/alien/parser/parser.factor new file mode 100644 index 0000000000..193893fabc --- /dev/null +++ b/basis/alien/parser/parser.factor @@ -0,0 +1,19 @@ +! Copyright (C) 2008 Slava Pestov. +! See http://factorcode.org/license.txt for BSD license. +USING: alien alien.c-types arrays assocs effects grouping kernel +parser sequences splitting words fry locals ; +IN: alien.parser + +: parse-arglist ( parameters return -- types effect ) + [ 2 group unzip [ "," ?tail drop ] map ] + [ [ { } ] [ 1array ] if-void ] + bi* ; + +: function-quot ( return library function types -- quot ) + '[ _ _ _ _ alien-invoke ] ; + +:: define-function ( return library function parameters -- ) + function create-in dup reset-generic + return library function + parameters return parse-arglist [ function-quot ] dip + define-declared ; diff --git a/basis/alien/strings/strings.factor b/basis/alien/strings/strings.factor index 70bbe773ee..d482634772 100644 --- a/basis/alien/strings/strings.factor +++ b/basis/alien/strings/strings.factor @@ -3,13 +3,13 @@ USING: arrays sequences kernel accessors math alien.accessors alien.c-types byte-arrays words io io.encodings io.streams.byte-array io.streams.memory io.encodings.utf8 -io.encodings.utf16 system alien strings cpu.architecture ; +io.encodings.utf16 system alien strings cpu.architecture fry ; IN: alien.strings GENERIC# alien>string 1 ( c-ptr encoding -- string/f ) M: c-ptr alien>string - >r r> + [ ] [ ] bi* "\0" swap stream-read-until drop ; M: f alien>string @@ -40,6 +40,9 @@ PREDICATE: string-type < pair M: string-type c-type ; +M: string-type c-type-class + drop object ; + M: string-type heap-size drop "void*" heap-size ; @@ -74,10 +77,10 @@ M: string-type c-type-unboxer drop "void*" c-type-unboxer ; M: string-type c-type-boxer-quot - second [ alien>string ] curry [ ] like ; + second '[ _ alien>string ] ; M: string-type c-type-unboxer-quot - second [ string>alien ] curry [ ] like ; + second '[ _ string>alien ] ; M: string-type c-type-getter drop [ alien-cell ] ; diff --git a/basis/alien/structs/fields/fields.factor b/basis/alien/structs/fields/fields.factor index 19e5b8c326..17294aed87 100644 --- a/basis/alien/structs/fields/fields.factor +++ b/basis/alien/structs/fields/fields.factor @@ -29,10 +29,10 @@ PREDICATE: slot-writer < word "writing" word-prop >boolean ; writer>> swap "writing" set-word-prop ; : reader-word ( class name vocab -- word ) - >r >r "-" r> 3append r> create ; + [ "-" swap 3append ] dip create ; : writer-word ( class name vocab -- word ) - >r [ swap "set-" % % "-" % % ] "" make r> create ; + [ [ swap "set-" % % "-" % % ] "" make ] dip create ; : ( struct-name vocab type field-name -- spec ) field-spec new @@ -52,25 +52,21 @@ PREDICATE: slot-writer < word "writing" word-prop >boolean ; [ (>>offset) ] [ type>> heap-size + ] 2bi ] reduce ; -: define-struct-slot-word ( spec word quot -- ) - rot offset>> prefix define-inline ; +: define-struct-slot-word ( word quot spec -- ) + offset>> prefix define-inline ; : define-getter ( type spec -- ) [ set-reader-props ] keep - [ ] [ reader>> ] [ type>> [ c-getter ] [ c-type-boxer-quot ] bi append - ] tri - define-struct-slot-word ; + ] + [ ] tri define-struct-slot-word ; : define-setter ( type spec -- ) [ set-writer-props ] keep - [ ] - [ writer>> ] - [ type>> c-setter ] tri - define-struct-slot-word ; + [ writer>> ] [ type>> c-setter ] [ ] tri define-struct-slot-word ; : define-field ( type spec -- ) [ define-getter ] [ define-setter ] 2bi ; diff --git a/basis/alien/structs/structs-tests.factor b/basis/alien/structs/structs-tests.factor index 8c7d9f9b29..ec0c01c2e7 100644 --- a/basis/alien/structs/structs-tests.factor +++ b/basis/alien/structs/structs-tests.factor @@ -38,7 +38,7 @@ C-UNION: barx [ 120 ] [ "barx" heap-size ] unit-test "help" vocab [ - "help" "help" lookup "help" set + "print-topic" "help" lookup "help" set [ ] [ \ foox-x "help" get execute ] unit-test [ ] [ \ set-foox-x "help" get execute ] unit-test ] when diff --git a/basis/alien/structs/structs.factor b/basis/alien/structs/structs.factor index ce30a2ee25..a3c616cda2 100644 --- a/basis/alien/structs/structs.factor +++ b/basis/alien/structs/structs.factor @@ -1,58 +1,63 @@ ! Copyright (C) 2004, 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: accessors arrays generic hashtables kernel kernel.private -math namespaces parser sequences strings words libc +math namespaces parser sequences strings words libc fry alien.c-types alien.structs.fields cpu.architecture ; IN: alien.structs -: if-value-structs? ( ctype true false -- ) - value-structs? - [ drop call ] [ >r 2drop "void*" r> call ] if ; inline - TUPLE: struct-type size align fields ; M: struct-type heap-size size>> ; +M: struct-type c-type-class drop object ; + M: struct-type c-type-align align>> ; M: struct-type c-type-stack-align? drop f ; -M: struct-type unbox-parameter - [ %unbox-struct ] [ unbox-parameter ] if-value-structs? ; +: if-value-struct ( ctype true false -- ) + [ dup value-struct? ] 2dip '[ drop "void*" @ ] if ; inline -M: struct-type unbox-return - f swap %unbox-struct ; +M: struct-type unbox-parameter + [ %unbox-large-struct ] [ unbox-parameter ] if-value-struct ; M: struct-type box-parameter - [ %box-struct ] [ box-parameter ] if-value-structs? ; + [ %box-large-struct ] [ box-parameter ] if-value-struct ; + +: if-small-struct ( c-type true false -- ? ) + [ dup struct-small-enough? ] 2dip '[ f swap @ ] if ; inline + +M: struct-type unbox-return + [ %unbox-small-struct ] [ %unbox-large-struct ] if-small-struct ; M: struct-type box-return - f swap %box-struct ; + [ %box-small-struct ] [ %box-large-struct ] if-small-struct ; M: struct-type stack-size - [ heap-size ] [ stack-size ] if-value-structs? ; + [ heap-size ] [ stack-size ] if-value-struct ; : c-struct? ( type -- ? ) (c-type) struct-type? ; -: (define-struct) ( name vocab size align fields -- ) - >r [ align ] keep r> +: (define-struct) ( name size align fields -- ) + [ [ align ] keep ] dip struct-type boa - -rot define-c-type ; + swap typedef ; -: define-struct-early ( name vocab fields -- fields ) - -rot [ rot first2 ] 2curry map ; +: make-fields ( name vocab fields -- fields ) + [ first2 ] with with map ; : compute-struct-align ( types -- n ) [ c-type-align ] map supremum ; : define-struct ( name vocab fields -- ) - pick >r - [ struct-offsets ] keep - [ [ type>> ] map compute-struct-align ] keep - [ (define-struct) ] keep - r> [ swap define-field ] curry each ; + [ + [ 2drop ] [ make-fields ] 3bi + [ struct-offsets ] keep + [ [ type>> ] map compute-struct-align ] keep + [ (define-struct) ] keep + ] [ 2drop '[ _ swap define-field ] ] 3bi each ; -: define-union ( name vocab members -- ) +: define-union ( name members -- ) [ expand-constants ] map [ [ heap-size ] map supremum ] keep compute-struct-align f (define-struct) ; diff --git a/basis/alien/syntax/syntax-docs.factor b/basis/alien/syntax/syntax-docs.factor index 37cbd12801..586bb97402 100644 --- a/basis/alien/syntax/syntax-docs.factor +++ b/basis/alien/syntax/syntax-docs.factor @@ -1,5 +1,5 @@ IN: alien.syntax -USING: alien alien.c-types alien.structs alien.syntax.private +USING: alien alien.c-types alien.parser alien.structs help.markup help.syntax ; HELP: DLL" @@ -54,12 +54,6 @@ HELP: TYPEDEF: { $description "Aliases the C type " { $snippet "old" } " under the name " { $snippet "new" } " if ." } { $notes "This word differs from " { $link typedef } " in that it runs at parse time, to ensure correct ordering of operations when loading source files. Words defined in source files are compiled before top-level forms are run, so if a source file defines C binding words and uses " { $link typedef } ", the type alias won't be available at compile time." } ; -HELP: TYPEDEF-IF: -{ $syntax "TYPEDEF-IF: word old new" } -{ $values { "word" "a word with stack effect " { $snippet "( -- ? )" } } { "old" "a C type" } { "new" "a C type" } } -{ $description "Aliases the C type " { $snippet "old" } " under the name " { $snippet "new" } " if " { $snippet "word" } " evaluates to a true value." } -{ $notes "This word differs from " { $link typedef } " in that it runs at parse time, to ensure correct ordering of operations when loading source files. Words defined in source files are compiled before top-level forms are run, so if a source file defines C binding words and uses " { $link typedef } ", the type alias won't be available at compile time." } ; - HELP: C-STRUCT: { $syntax "C-STRUCT: name pairs... ;" } { $values { "name" "a new C type name" } { "pairs" "C type / field name string pairs" } } @@ -88,7 +82,7 @@ HELP: typedef { $description "Alises the C type " { $snippet "old" } " under the name " { $snippet "new" } "." } { $notes "Using this word in the same source file which defines C bindings can cause problems, because words are compiled before top-level forms are run. Use the " { $link POSTPONE: TYPEDEF: } " word instead." } ; -{ POSTPONE: TYPEDEF-IF: POSTPONE: TYPEDEF: typedef } related-words +{ POSTPONE: TYPEDEF: typedef } related-words HELP: c-struct? { $values { "type" "a string" } { "?" "a boolean" } } diff --git a/basis/alien/syntax/syntax.factor b/basis/alien/syntax/syntax.factor index 7629897fc0..d10c97cd3d 100644 --- a/basis/alien/syntax/syntax.factor +++ b/basis/alien/syntax/syntax.factor @@ -4,35 +4,9 @@ USING: accessors arrays alien alien.c-types alien.structs alien.arrays alien.strings kernel math namespaces parser sequences words quotations math.parser splitting grouping effects prettyprint prettyprint.sections prettyprint.backend -assocs combinators lexer strings.parser ; +assocs combinators lexer strings.parser alien.parser ; IN: alien.syntax - ; - -: function-quot ( type lib func types -- quot ) - [ alien-invoke ] 2curry 2curry ; - -: define-function ( return library function parameters -- ) - >r pick r> parse-arglist - pick create-in dup reset-generic - >r >r function-quot r> r> - -rot define-declared ; - -PRIVATE> - -: indirect-quot ( function-ptr-quot return types abi -- quot ) - [ alien-indirect ] 3curry compose ; - -: define-indirect ( abi return function-ptr-quot function-name parameters -- ) - >r pick r> parse-arglist - rot create-in dup reset-generic - >r >r swapd roll indirect-quot r> r> - -rot define-declared ; - : DLL" lexer get skip-blank parse-string dlopen parsed ; parsing : ALIEN: scan string>number parsed ; parsing @@ -49,22 +23,16 @@ PRIVATE> : TYPEDEF: scan scan typedef ; parsing -: TYPEDEF-IF: - scan-word execute scan scan rot [ typedef ] [ 2drop ] if ; parsing - : C-STRUCT: - scan in get - parse-definition - >r 2dup r> define-struct-early - define-struct ; parsing + scan in get parse-definition define-struct ; parsing : C-UNION: - scan in get parse-definition define-union ; parsing + scan parse-definition define-union ; parsing : C-ENUM: ";" parse-tokens dup length - [ >r create-in r> 1quotation define ] 2each ; + [ [ create-in ] dip 1quotation define ] 2each ; parsing M: alien pprint* diff --git a/basis/binary-search/binary-search-docs.factor b/basis/binary-search/binary-search-docs.factor index caabbd7419..cf7915159a 100644 --- a/basis/binary-search/binary-search-docs.factor +++ b/basis/binary-search/binary-search-docs.factor @@ -2,7 +2,7 @@ IN: binary-search USING: help.markup help.syntax sequences kernel math.order ; HELP: search -{ $values { "seq" "a sorted sequence" } { "quot" "a quotation with stack effect " { $snippet "( elt -- <=> )" } } { "i" "an index, or " { $link f } } { "elt" "an element, or " { $link f } } } +{ $values { "seq" "a sorted sequence" } { "quot" { $quotation "( elt -- <=> )" } } { "i" "an index, or " { $link f } } { "elt" "an element, or " { $link f } } } { $description "Performs a binary search on a sequence, calling the quotation to decide whether to end the search (" { $link +eq+ } "), search lower (" { $link +lt+ } ") or search higher (" { $link +gt+ } ")." $nl "If the sequence is non-empty, outputs the index and value of the closest match, which is either an element for which the quotation output " { $link +eq+ } ", or failing that, least element for which the quotation output " { $link +lt+ } "." diff --git a/basis/bit-arrays/bit-arrays.factor b/basis/bit-arrays/bit-arrays.factor index 11601f7b63..4cb2032f4f 100644 --- a/basis/bit-arrays/bit-arrays.factor +++ b/basis/bit-arrays/bit-arrays.factor @@ -2,7 +2,7 @@ ! See http://factorcode.org/license.txt for BSD license. USING: alien.c-types accessors math alien.accessors kernel kernel.private locals sequences sequences.private byte-arrays -parser prettyprint.backend ; +parser prettyprint.backend fry ; IN: bit-arrays TUPLE: bit-array @@ -24,9 +24,8 @@ TUPLE: bit-array : bits>bytes 7 + n>byte ; inline : (set-bits) ( bit-array n -- ) - [ [ length bits>cells ] keep ] dip - [ -rot underlying>> set-uint-nth ] 2curry - each ; inline + [ [ length bits>cells ] keep ] dip swap underlying>> + '[ [ _ _ ] dip set-alien-unsigned-4 ] each ; inline PRIVATE> @@ -84,9 +83,9 @@ M: bit-array byte-length length 7 + -3 shift ; ] if ; : bit-array>integer ( bit-array -- n ) - 0 swap underlying>> [ length ] keep [ - uchar-nth swap 8 shift bitor - ] curry each ; + 0 swap underlying>> dup length [ + alien-unsigned-1 swap 8 shift bitor + ] with each ; INSTANCE: bit-array sequence diff --git a/basis/bit-vectors/bit-vectors-tests.factor b/basis/bit-vectors/bit-vectors-tests.factor index dff9a8db37..31327999e7 100644 --- a/basis/bit-vectors/bit-vectors-tests.factor +++ b/basis/bit-vectors/bit-vectors-tests.factor @@ -4,7 +4,7 @@ USING: tools.test bit-vectors vectors sequences kernel math ; [ 0 ] [ 123 length ] unit-test : do-it - 1234 swap [ >r even? r> push ] curry each ; + 1234 swap [ [ even? ] dip push ] curry each ; [ t ] [ 3 dup do-it diff --git a/basis/bootstrap/help/help.factor b/basis/bootstrap/help/help.factor index e2a2288988..5b49ce2802 100644 --- a/basis/bootstrap/help/help.factor +++ b/basis/bootstrap/help/help.factor @@ -1,6 +1,6 @@ USING: help help.topics help.syntax help.crossref help.definitions io io.files kernel namespaces vocabs sequences -parser vocabs.loader ; +parser vocabs.loader vocabs.loader.private accessors assocs ; IN: bootstrap.help : load-help ( -- ) @@ -10,8 +10,8 @@ IN: bootstrap.help t load-help? set-global [ drop ] load-vocab-hook [ - vocabs - [ vocab-docs-loaded? not ] filter + dictionary get values + [ docs-loaded?>> not ] filter [ load-docs ] each ] with-variable ; diff --git a/basis/bootstrap/image/download/download.factor b/basis/bootstrap/image/download/download.factor index 71aa2e8adc..f9b7b56779 100644 --- a/basis/bootstrap/image/download/download.factor +++ b/basis/bootstrap/image/download/download.factor @@ -1,6 +1,6 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. -USING: http.client checksums checksums.openssl splitting assocs +USING: http.client checksums checksums.md5 splitting assocs kernel io.files bootstrap.image sequences io urls ; IN: bootstrap.image.download @@ -13,7 +13,7 @@ IN: bootstrap.image.download : need-new-image? ( image -- ? ) dup exists? [ - [ openssl-md5 checksum-file hex-string ] + [ md5 checksum-file hex-string ] [ download-checksums at ] bi = not ] [ drop t ] if ; diff --git a/basis/bootstrap/image/image.factor b/basis/bootstrap/image/image.factor index 8b0051148f..f352a4a254 100644 --- a/basis/bootstrap/image/image.factor +++ b/basis/bootstrap/image/image.factor @@ -72,7 +72,7 @@ SYMBOL: objects : put-object ( n obj -- ) (objects) set-at ; : cache-object ( obj quot -- value ) - >r (objects) r> [ obj>> ] prepose cache ; inline + [ (objects) ] dip [ obj>> ] prepose cache ; inline ! Constants @@ -97,10 +97,10 @@ SYMBOL: sub-primitives { [ { } make ] [ ] [ ] [ ] } spread 4array ; inline : jit-define ( quot rc rt offset name -- ) - >r make-jit r> set ; inline + [ make-jit ] dip set ; inline : define-sub-primitive ( quot rc rt offset word -- ) - >r make-jit r> sub-primitives get set-at ; + [ make-jit ] dip sub-primitives get set-at ; ! The image being constructed; a vector of word-size integers SYMBOL: image @@ -124,22 +124,29 @@ SYMBOL: jit-primitive-word SYMBOL: jit-primitive SYMBOL: jit-word-jump SYMBOL: jit-word-call -SYMBOL: jit-push-literal SYMBOL: jit-push-immediate SYMBOL: jit-if-word -SYMBOL: jit-if-jump +SYMBOL: jit-if-1 +SYMBOL: jit-if-2 SYMBOL: jit-dispatch-word SYMBOL: jit-dispatch +SYMBOL: jit-dip-word +SYMBOL: jit-dip +SYMBOL: jit-2dip-word +SYMBOL: jit-2dip +SYMBOL: jit-3dip-word +SYMBOL: jit-3dip SYMBOL: jit-epilog SYMBOL: jit-return SYMBOL: jit-profiling SYMBOL: jit-declare-word +SYMBOL: jit-save-stack ! Default definition for undefined words SYMBOL: undefined-quot -: userenv-offset ( symbol -- n ) - { +: userenvs ( -- assoc ) + H{ { bootstrap-boot-quot 20 } { bootstrap-global 21 } { jit-code-format 22 } @@ -148,9 +155,9 @@ SYMBOL: undefined-quot { jit-primitive 25 } { jit-word-jump 26 } { jit-word-call 27 } - { jit-push-literal 28 } - { jit-if-word 29 } - { jit-if-jump 30 } + { jit-if-word 28 } + { jit-if-1 29 } + { jit-if-2 30 } { jit-dispatch-word 31 } { jit-dispatch 32 } { jit-epilog 33 } @@ -158,8 +165,18 @@ SYMBOL: undefined-quot { jit-profiling 35 } { jit-push-immediate 36 } { jit-declare-word 42 } + { jit-save-stack 43 } + { jit-dip-word 44 } + { jit-dip 45 } + { jit-2dip-word 46 } + { jit-2dip 47 } + { jit-3dip-word 48 } + { jit-3dip 49 } { undefined-quot 60 } - } at header-size + ; + } ; inline + +: userenv-offset ( symbol -- n ) + userenvs at header-size + ; : emit ( cell -- ) image get push ; @@ -188,7 +205,7 @@ SYMBOL: undefined-quot : emit-fixnum ( n -- ) tag-fixnum emit ; : emit-object ( header tag quot -- addr ) - swap here-as >r swap tag-fixnum emit call align-here r> ; + swap here-as [ swap tag-fixnum emit call align-here ] dip ; inline ! Write an object to the image. @@ -441,6 +458,9 @@ M: quotation ' \ dispatch jit-dispatch-word set \ do-primitive jit-primitive-word set \ declare jit-declare-word set + \ dip jit-dip-word set + \ 2dip jit-2dip-word set + \ 3dip jit-3dip-word set [ undefined ] undefined-quot set { jit-code-format @@ -449,16 +469,23 @@ M: quotation ' jit-primitive jit-word-jump jit-word-call - jit-push-literal jit-push-immediate jit-if-word - jit-if-jump + jit-if-1 + jit-if-2 jit-dispatch-word jit-dispatch + jit-dip-word + jit-dip + jit-2dip-word + jit-2dip + jit-3dip-word + jit-3dip jit-epilog jit-return jit-profiling jit-declare-word + jit-save-stack undefined-quot } [ emit-userenv ] each ; diff --git a/basis/bootstrap/stage2.factor b/basis/bootstrap/stage2.factor index 3b6c04329c..4ab36ec94e 100644 --- a/basis/bootstrap/stage2.factor +++ b/basis/bootstrap/stage2.factor @@ -8,6 +8,8 @@ definitions assocs compiler.errors compiler.units math.parser generic sets debugger command-line ; IN: bootstrap.stage2 +SYMBOL: core-bootstrap-time + SYMBOL: bootstrap-time : default-image-name ( -- string ) @@ -30,11 +32,15 @@ SYMBOL: bootstrap-time : count-words ( pred -- ) all-words swap count number>string write ; -: print-report ( time -- ) +: print-time ( ms -- ) 1000 /i 60 /mod swap - "Bootstrap completed in " write number>string write - " minutes and " write number>string write " seconds." print + number>string write + " minutes and " write number>string write " seconds." print ; + +: print-report ( -- ) + "Core bootstrap completed in " write core-bootstrap-time get print-time + "Bootstrap completed in " write bootstrap-time get print-time [ compiled>> ] count-words " compiled words" print [ symbol? ] count-words " symbol words" print @@ -46,22 +52,22 @@ SYMBOL: bootstrap-time [ ! We time bootstrap - millis >r + millis default-image-name "output-image" set-global "math compiler threads help io tools ui ui.tools unicode handbook" "include" set-global "" "exclude" set-global - parse-command-line + (command-line) parse-command-line - "-no-crossref" cli-args member? [ do-crossref ] unless + do-crossref ! Set dll paths os wince? [ "windows.ce" require ] when os winnt? [ "windows.nt" require ] when - "deploy-vocab" get [ + "staging" get "deploy-vocab" get or [ "stage2: deployment mode" print ] [ "listener" require @@ -71,6 +77,8 @@ SYMBOL: bootstrap-time [ load-components + millis over - core-bootstrap-time set-global + run-bootstrap-init ] with-compiler-errors :errors @@ -84,15 +92,10 @@ SYMBOL: bootstrap-time [ boot do-init-hooks - [ - parse-command-line - run-user-init - "run" get run - output-stream get [ stream-flush ] when* - ] [ print-error 1 exit ] recover + handle-command-line ] set-boot-quot - millis r> - dup bootstrap-time set-global + millis swap - bootstrap-time set-global print-report "output-image" get save-image-and-exit diff --git a/basis/boxes/boxes.factor b/basis/boxes/boxes.factor index 9e2e8a4673..39f8eb44cc 100644 --- a/basis/boxes/boxes.factor +++ b/basis/boxes/boxes.factor @@ -23,4 +23,4 @@ ERROR: box-empty box ; dup occupied>> [ box> t ] [ drop f f ] if ; : if-box? ( box quot -- ) - >r ?box r> [ drop ] if ; inline + [ ?box ] dip [ drop ] if ; inline diff --git a/basis/calendar/calendar-docs.factor b/basis/calendar/calendar-docs.factor index 64c74a494a..748f9d124c 100644 --- a/basis/calendar/calendar-docs.factor +++ b/basis/calendar/calendar-docs.factor @@ -99,6 +99,48 @@ HELP: seconds-per-year { $values { "integer" integer } } { $description "Returns the number of seconds in a year averaged over 400 years. Used internally for adding an arbitrary real number of seconds to a timestamp." } ; +HELP: biweekly +{ $values + { "x" number } + { "y" number } +} +{ $description "Divides a number by the number of two week periods in a year." } ; + +HELP: daily-360 +{ $values + { "x" number } + { "y" number } +} +{ $description "Divides a number by the number of days in a 360-day year." } ; + +HELP: daily-365 +{ $values + { "x" number } + { "y" number } +} +{ $description "Divides a number by the number of days in a 365-day year." } ; + +HELP: monthly +{ $values + { "x" number } + { "y" number } +} +{ $description "Divides a number by the number of months in a year." } ; + +HELP: semimonthly +{ $values + { "x" number } + { "y" number } +} +{ $description "Divides a number by the number of half-months in a year. Note that biweekly has two more periods than semimonthly." } ; + +HELP: weekly +{ $values + { "x" number } + { "y" number } +} +{ $description "Divides a number by the number of weeks in a year." } ; + HELP: julian-day-number { $values { "year" integer } { "month" integer } { "day" integer } { "n" integer } } { $description "Calculates the Julian day number from a year, month, and day. The difference between two Julian day numbers is the number of days that have elapsed between the two corresponding dates." } @@ -365,12 +407,12 @@ HELP: unix-1970 { $values { "timestamp" timestamp } } { $description "Outputs the beginning of UNIX time, or midnight, January 1, 1970." } ; -HELP: millis>timestamp +HELP: micros>timestamp { $values { "x" number } { "timestamp" timestamp } } -{ $description "Converts a number of milliseconds into a timestamp value in GMT time." } +{ $description "Converts a number of microseconds into a timestamp value in GMT time." } { $examples { $example "USING: accessors calendar prettyprint ;" - "1000 millis>timestamp year>> ." + "1000 micros>timestamp year>> ." "1970" } } ; @@ -540,6 +582,8 @@ ARTICLE: "calendar" "Calendar" { $subsection "years" } { $subsection "months" } { $subsection "days" } +"Calculating amounts per period of time:" +{ $subsection "time-period-calculations" } "Meta-data about the calendar:" { $subsection "calendar-facts" } ; @@ -626,6 +670,18 @@ ARTICLE: "calendar-facts" "Calendar facts" { $subsection day-of-week } ; +ARTICLE: "time-period-calculations" "Calculations over periods of time" +{ $subsection monthly } +{ $subsection semimonthly } +{ $subsection biweekly } +{ $subsection weekly } +{ $subsection daily-360 } +{ $subsection daily-365 } +{ $subsection biweekly } +{ $subsection biweekly } +{ $subsection biweekly } +; + ARTICLE: "years" "Year operations" "Leap year predicate:" { $subsection leap-year? } diff --git a/basis/calendar/calendar-tests.factor b/basis/calendar/calendar-tests.factor index 995bd23c09..943ba8c3d5 100644 --- a/basis/calendar/calendar-tests.factor +++ b/basis/calendar/calendar-tests.factor @@ -143,10 +143,10 @@ IN: calendar.tests [ +gt+ ] [ 2005 1 1 12 30 0 instant 2004 1 1 13 30 0 instant <=> ] unit-test -[ t ] [ now timestamp>millis millis - 1000 < ] unit-test -[ t ] [ 0 millis>timestamp unix-1970 = ] unit-test -[ t ] [ 123456789000 [ millis>timestamp timestamp>millis ] keep = ] unit-test -[ t ] [ 123456789123456 [ millis>timestamp timestamp>millis ] keep = ] unit-test +[ t ] [ now timestamp>micros micros - 1000000 < ] unit-test +[ t ] [ 0 micros>timestamp unix-1970 = ] unit-test +[ t ] [ 123456789000000 [ micros>timestamp timestamp>micros ] keep = ] unit-test +[ t ] [ 123456789123456000 [ micros>timestamp timestamp>micros ] keep = ] unit-test : checktime+ now dup clone [ rot time+ drop ] keep = ; @@ -167,3 +167,5 @@ IN: calendar.tests [ t ] [ now 50 milliseconds sleep now before? ] unit-test [ t ] [ now 50 milliseconds sleep now swap after? ] unit-test [ t ] [ now 50 milliseconds sleep now 50 milliseconds sleep now swapd between? ] unit-test + +[ 4+1/6 ] [ 100 semimonthly ] unit-test diff --git a/basis/calendar/calendar.factor b/basis/calendar/calendar.factor index c002760748..e2564b5a28 100644 --- a/basis/calendar/calendar.factor +++ b/basis/calendar/calendar.factor @@ -89,6 +89,13 @@ PRIVATE> : minutes-per-year ( -- ratio ) 5259492/10 ; inline : seconds-per-year ( -- integer ) 31556952 ; inline +: monthly ( x -- y ) 12 / ; inline +: semimonthly ( x -- y ) 24 / ; inline +: biweekly ( x -- y ) 26 / ; inline +: weekly ( x -- y ) 52 / ; inline +: daily-360 ( x -- y ) 360 / ; inline +: daily-365 ( x -- y ) 365 / ; inline + :: julian-day-number ( year month day -- n ) #! Returns a composite date number #! Not valid before year -4800 @@ -173,7 +180,7 @@ M: real +year ( timestamp n -- timestamp ) 12 /rem dup zero? [ drop 1- 12 ] when swap ; inline M: integer +month ( timestamp n -- timestamp ) - [ over month>> + months/years >r >>month r> +year ] unless-zero ; + [ over month>> + months/years [ >>month ] dip +year ] unless-zero ; M: real +month ( timestamp n -- timestamp ) [ float>whole-part swapd average-month * +day swap +month ] unless-zero ; @@ -181,7 +188,7 @@ M: real +month ( timestamp n -- timestamp ) M: integer +day ( timestamp n -- timestamp ) [ over >date< julian-day-number + julian-day-number>date - >r >r >>year r> >>month r> >>day + [ >>year ] [ >>month ] [ >>day ] tri* ] unless-zero ; M: real +day ( timestamp n -- timestamp ) @@ -191,7 +198,7 @@ M: real +day ( timestamp n -- timestamp ) 24 /rem swap ; M: integer +hour ( timestamp n -- timestamp ) - [ over hour>> + hours/days >r >>hour r> +day ] unless-zero ; + [ over hour>> + hours/days [ >>hour ] dip +day ] unless-zero ; M: real +hour ( timestamp n -- timestamp ) float>whole-part swapd 60 * +minute swap +hour ; @@ -200,7 +207,7 @@ M: real +hour ( timestamp n -- timestamp ) 60 /rem swap ; M: integer +minute ( timestamp n -- timestamp ) - [ over minute>> + minutes/hours >r >>minute r> +hour ] unless-zero ; + [ over minute>> + minutes/hours [ >>minute ] dip +hour ] unless-zero ; M: real +minute ( timestamp n -- timestamp ) [ float>whole-part swapd 60 * +second swap +minute ] unless-zero ; @@ -209,7 +216,7 @@ M: real +minute ( timestamp n -- timestamp ) 60 /rem swap >integer ; M: number +second ( timestamp n -- timestamp ) - [ over second>> + seconds/minutes >r >>second r> +minute ] unless-zero ; + [ over second>> + seconds/minutes [ >>second ] dip +minute ] unless-zero ; : (time+) [ second>> +second ] keep @@ -226,7 +233,7 @@ PRIVATE> GENERIC# time+ 1 ( time1 time2 -- time3 ) M: timestamp time+ - >r clone r> (time+) drop ; + [ clone ] dip (time+) drop ; M: duration time+ dup timestamp? [ @@ -284,7 +291,7 @@ M: timestamp <=> ( ts1 ts2 -- n ) : (time-) ( timestamp timestamp -- n ) [ >gmt ] bi@ [ [ >date< julian-day-number ] bi@ - 86400 * ] 2keep - [ >time< >r >r 3600 * r> 60 * r> + + ] bi@ - + ; + [ >time< [ [ 3600 * ] [ 60 * ] bi* ] dip + + ] bi@ - + ; M: timestamp time- #! Exact calendar-time difference @@ -320,14 +327,20 @@ M: duration time- 1970 1 1 0 0 0 instant ; : millis>timestamp ( x -- timestamp ) - >r unix-1970 r> milliseconds time+ ; + [ unix-1970 ] dip milliseconds time+ ; : timestamp>millis ( timestamp -- n ) unix-1970 (time-) 1000 * >integer ; +: micros>timestamp ( x -- timestamp ) + [ unix-1970 ] dip microseconds time+ ; + +: timestamp>micros ( timestamp -- n ) + unix-1970 (time-) 1000000 * >integer ; + : gmt ( -- timestamp ) #! GMT time, right now - unix-1970 millis milliseconds time+ ; + unix-1970 micros microseconds time+ ; : now ( -- timestamp ) gmt >local-time ; : hence ( duration -- timestamp ) now swap time+ ; @@ -337,10 +350,11 @@ M: duration time- #! Zeller Congruence #! http://web.textfiles.com/computers/formulas.txt #! good for any date since October 15, 1582 - >r dup 2 <= [ 12 + >r 1- r> ] when - >r dup [ 4 /i + ] keep [ 100 /i - ] keep 400 /i + r> - [ 1+ 3 * 5 /i + ] keep 2 * + r> - 1+ + 7 mod ; + [ + dup 2 <= [ [ 1- ] [ 12 + ] bi* ] when + [ dup [ 4 /i + ] keep [ 100 /i - ] keep 400 /i + ] dip + [ 1+ 3 * 5 /i + ] keep 2 * + + ] dip 1+ + 7 mod ; GENERIC: days-in-year ( obj -- n ) @@ -404,7 +418,7 @@ PRIVATE> : since-1970 ( duration -- timestamp ) unix-1970 time+ >local-time ; -M: timestamp sleep-until timestamp>millis sleep-until ; +M: timestamp sleep-until timestamp>micros sleep-until ; M: duration sleep hence sleep-until ; diff --git a/basis/calendar/format/format.factor b/basis/calendar/format/format.factor index b15da42409..8d34e8a3a4 100644 --- a/basis/calendar/format/format.factor +++ b/basis/calendar/format/format.factor @@ -138,11 +138,11 @@ M: timestamp year. ( timestamp -- ) : read-rfc3339-gmt-offset ( ch -- dt ) dup CHAR: Z = [ drop instant ] [ - >r - read-00 hours - read1 { { CHAR: : [ read-00 ] } { f [ 0 ] } } case minutes - time+ - r> signed-gmt-offset + [ + read-00 hours + read1 { { CHAR: : [ read-00 ] } { f [ 0 ] } } case minutes + time+ + ] dip signed-gmt-offset ] if ; : read-ymd ( -- y m d ) @@ -152,8 +152,9 @@ M: timestamp year. ( timestamp -- ) read-00 ":" expect read-00 ":" expect read-00 ; : read-rfc3339-seconds ( s -- s' ch ) - "+-Z" read-until >r - [ string>number ] [ length 10 swap ^ ] bi / + r> ; + "+-Z" read-until [ + [ string>number ] [ length 10 swap ^ ] bi / + + ] dip ; : (rfc3339>timestamp) ( -- timestamp ) read-ymd @@ -181,9 +182,9 @@ ERROR: invalid-timestamp-format ; : parse-rfc822-gmt-offset ( string -- dt ) dup "GMT" = [ drop instant ] [ - unclip >r - 2 cut [ string>number ] bi@ [ hours ] [ minutes ] bi* time+ - r> signed-gmt-offset + unclip [ + 2 cut [ string>number ] bi@ [ hours ] [ minutes ] bi* time+ + ] dip signed-gmt-offset ] if ; : (rfc822>timestamp) ( -- timestamp ) diff --git a/basis/calendar/model/model.factor b/basis/calendar/model/model.factor index 60a61c2026..8665cc22ce 100644 --- a/basis/calendar/model/model.factor +++ b/basis/calendar/model/model.factor @@ -7,7 +7,7 @@ SYMBOL: time : (time-thread) ( -- ) now time get set-model - 1000 sleep (time-thread) ; + 1 seconds sleep (time-thread) ; : time-thread ( -- ) [ diff --git a/basis/calendar/windows/tags.txt b/basis/calendar/windows/tags.txt old mode 100644 new mode 100755 index 02ec70f741..6bf68304bb --- a/basis/calendar/windows/tags.txt +++ b/basis/calendar/windows/tags.txt @@ -1,2 +1 @@ unportable -windows diff --git a/basis/channels/remote/remote.factor b/basis/channels/remote/remote.factor index 1a7addac12..6e10b23407 100644 --- a/basis/channels/remote/remote.factor +++ b/basis/channels/remote/remote.factor @@ -14,7 +14,7 @@ IN: channels.remote PRIVATE> : publish ( channel -- id ) - 256 random-bits dup >r remote-channels set-at r> ; + 256 random-bits dup [ remote-channels set-at ] dip ; : get-channel ( id -- channel ) remote-channels at ; diff --git a/basis/checksums/common/common.factor b/basis/checksums/common/common.factor index 1f25efef24..7d5f34777d 100644 --- a/basis/checksums/common/common.factor +++ b/basis/checksums/common/common.factor @@ -18,4 +18,4 @@ SYMBOL: bytes-read ] "" make 64 group ; : update-old-new ( old new -- ) - [ get >r get r> ] 2keep >r >r w+ dup r> set r> set ; inline + [ [ get ] bi@ w+ dup ] 2keep [ set ] bi@ ; inline diff --git a/basis/checksums/md5/md5.factor b/basis/checksums/md5/md5.factor index 6158254f84..257fd930c4 100644 --- a/basis/checksums/md5/md5.factor +++ b/basis/checksums/md5/md5.factor @@ -14,7 +14,7 @@ IN: checksums.md5 SYMBOLS: a b c d old-a old-b old-c old-d ; : T ( N -- Y ) - sin abs 4294967296 * >bignum ; foldable + sin abs 4294967296 * >integer ; foldable : initialize-md5 ( -- ) 0 bytes-read set diff --git a/basis/checksums/openssl/openssl.factor b/basis/checksums/openssl/openssl.factor index d42febb541..821cbe2f3a 100644 --- a/basis/checksums/openssl/openssl.factor +++ b/basis/checksums/openssl/openssl.factor @@ -28,7 +28,7 @@ M: evp-md-context dispose handle>> EVP_MD_CTX_cleanup drop ; : with-evp-md-context ( quot -- ) - maybe-init-ssl >r r> with-disposal ; inline + maybe-init-ssl [ ] dip with-disposal ; inline : digest-named ( name -- md ) dup EVP_get_digestbyname diff --git a/basis/checksums/sha1/sha1.factor b/basis/checksums/sha1/sha1.factor index bbae421b16..3767af7c55 100644 --- a/basis/checksums/sha1/sha1.factor +++ b/basis/checksums/sha1/sha1.factor @@ -41,9 +41,9 @@ SYMBOLS: h0 h1 h2 h3 h4 A B C D E w K ; : sha1-f ( B C D t -- f_tbcd ) 20 /i { - { 0 [ >r over bitnot r> bitand >r bitand r> bitor ] } + { 0 [ [ over bitnot ] dip bitand [ bitand ] dip bitor ] } { 1 [ bitxor bitxor ] } - { 2 [ 2dup bitand >r pick bitand >r bitand r> r> bitor bitor ] } + { 2 [ 2dup bitand [ pick bitand [ bitand ] dip ] dip bitor bitor ] } { 3 [ bitxor bitxor ] } } case ; diff --git a/basis/checksums/sha2/sha2.factor b/basis/checksums/sha2/sha2.factor index 0a6d8c26ab..beb657bd3e 100644 --- a/basis/checksums/sha2/sha2.factor +++ b/basis/checksums/sha2/sha2.factor @@ -59,7 +59,7 @@ SYMBOLS: vars M K H S0 S1 process-M word-size block-size ; [ 15 - swap nth s0-256 ] 2keep [ 7 - swap nth ] 2keep [ 2 - swap nth s1-256 ] 2keep - >r >r + + w+ r> r> swap set-nth ; inline + [ + + w+ ] 2dip swap set-nth ; inline : prepare-message-schedule ( seq -- w-seq ) word-size get group [ be> ] map block-size get 0 pad-right @@ -71,7 +71,7 @@ SYMBOLS: vars M K H S0 S1 process-M word-size block-size ; [ bitxor bitand ] keep bitxor ; : maj ( x y z -- x' ) - >r [ bitand ] 2keep bitor r> bitand bitor ; + [ [ bitand ] 2keep bitor ] dip bitand bitor ; : S0-256 ( x -- x' ) [ -2 bitroll-32 ] keep @@ -83,7 +83,7 @@ SYMBOLS: vars M K H S0 S1 process-M word-size block-size ; [ -11 bitroll-32 ] keep -25 bitroll-32 bitxor bitxor ; inline -: slice3 ( n seq -- a b c ) >r dup 3 + r> first3 ; inline +: slice3 ( n seq -- a b c ) [ dup 3 + ] dip first3 ; inline : T1 ( W n -- T1 ) [ swap nth ] keep @@ -105,7 +105,7 @@ SYMBOLS: vars M K H S0 S1 process-M word-size block-size ; d c pick exchange c b pick exchange b a pick exchange - >r w+ a r> set-nth ; + [ w+ a ] dip set-nth ; : process-chunk ( M -- ) H get clone vars set @@ -118,7 +118,7 @@ SYMBOLS: vars M K H S0 S1 process-M word-size block-size ; : preprocess-plaintext ( string big-endian? -- padded-string ) #! pad 0x80 then 00 til 8 bytes left, then 64bit length in bits - >r >sbuf r> over [ + [ >sbuf ] dip over [ HEX: 80 , dup length HEX: 3f bitand calculate-pad-length 0 % diff --git a/basis/cocoa/application/application.factor b/basis/cocoa/application/application.factor index 8f32782d76..c62fab0f15 100644 --- a/basis/cocoa/application/application.factor +++ b/basis/cocoa/application/application.factor @@ -40,12 +40,13 @@ FUNCTION: void NSBeep ( ) ; dup next-event [ -> sendEvent: t ] [ drop f ] if* ; : add-observer ( observer selector name object -- ) - >r >r >r >r NSNotificationCenter -> defaultCenter - r> r> sel_registerName - r> r> -> addObserver:selector:name:object: ; + [ + [ NSNotificationCenter -> defaultCenter ] 2dip + sel_registerName + ] 2dip -> addObserver:selector:name:object: ; : remove-observer ( observer -- ) - >r NSNotificationCenter -> defaultCenter r> + [ NSNotificationCenter -> defaultCenter ] dip -> removeObserver: ; : finish-launching ( -- ) NSApp -> finishLaunching ; diff --git a/basis/cocoa/dialogs/dialogs.factor b/basis/cocoa/dialogs/dialogs.factor index 606526a240..2b01c5d751 100644 --- a/basis/cocoa/dialogs/dialogs.factor +++ b/basis/cocoa/dialogs/dialogs.factor @@ -1,4 +1,4 @@ -! Copyright (C) 2006, 2007 Slava Pestov +! Copyright (C) 2006, 2008 Slava Pestov ! See http://factorcode.org/license.txt for BSD license. USING: kernel cocoa cocoa.messages cocoa.classes cocoa.application sequences splitting core-foundation ; @@ -26,9 +26,9 @@ IN: cocoa.dialogs [ -> filenames CF>string-array ] [ drop f ] if ; : split-path ( path -- dir file ) - "/" last-split1 [ ] bi@ ; + "/" split1-last [ ] bi@ ; : save-panel ( path -- paths ) - dup - rot split-path -> runModalForDirectory:file: NSOKButton = + [ dup ] dip + split-path -> runModalForDirectory:file: NSOKButton = [ -> filename CF>string ] [ drop f ] if ; diff --git a/basis/cocoa/enumeration/enumeration.factor b/basis/cocoa/enumeration/enumeration.factor index 7de1f24a3c..7f5b777283 100644 --- a/basis/cocoa/enumeration/enumeration.factor +++ b/basis/cocoa/enumeration/enumeration.factor @@ -1,26 +1,31 @@ -USING: kernel cocoa cocoa.types alien.c-types locals math sequences -vectors fry libc ; +! Copyright (C) 2008 Joe Groff. +! See http://factorcode.org/license.txt for BSD license. +USING: kernel cocoa cocoa.types alien.c-types locals math +sequences vectors fry libc destructors +specialized-arrays.direct.alien ; IN: cocoa.enumeration : NS-EACH-BUFFER-SIZE 16 ; inline -: (with-enumeration-buffers) ( quot -- ) - "NSFastEnumerationState" heap-size swap '[ - NS-EACH-BUFFER-SIZE "id" heap-size * [ - NS-EACH-BUFFER-SIZE @ - ] with-malloc - ] with-malloc ; inline +: with-enumeration-buffers ( quot -- ) + [ + [ + "NSFastEnumerationState" malloc-object &free + NS-EACH-BUFFER-SIZE "id" heap-size * malloc-object &free + NS-EACH-BUFFER-SIZE + ] dip call + ] with-destructors ; inline :: (NSFastEnumeration-each) ( object quot: ( elt -- ) state stackbuf count -- ) object state stackbuf count -> countByEnumeratingWithState:objects:count: - dup zero? [ drop ] [ + dup 0 = [ drop ] [ state NSFastEnumerationState-itemsPtr [ stackbuf ] unless* - '[ _ void*-nth quot call ] each + swap quot each object quot state stackbuf count (NSFastEnumeration-each) ] if ; inline recursive : NSFastEnumeration-each ( object quot -- ) - [ (NSFastEnumeration-each) ] (with-enumeration-buffers) ; inline + [ (NSFastEnumeration-each) ] with-enumeration-buffers ; inline : NSFastEnumeration-map ( object quot -- vector ) NS-EACH-BUFFER-SIZE diff --git a/basis/cocoa/messages/messages-docs.factor b/basis/cocoa/messages/messages-docs.factor index 9b5e3fdfd9..400599383f 100644 --- a/basis/cocoa/messages/messages-docs.factor +++ b/basis/cocoa/messages/messages-docs.factor @@ -31,7 +31,7 @@ HELP: alien>objc-types { objc>alien-types alien>objc-types } related-words HELP: import-objc-class -{ $values { "name" string } { "quot" "a quotation with stack effect " { $snippet "( -- )" } } } +{ $values { "name" string } { "quot" { $quotation "( -- )" } } } { $description "If a class named " { $snippet "name" } " is already known to the Objective C interface, does nothing. Otherwise, first calls the quotation. The quotation should make the class available to the Objective C runtime if necessary, either by loading a framework or defining it directly. After the quotation returns, this word makes the class available to Factor programs by importing methods and creating a class word the class object in the " { $vocab-link "cocoa.classes" } " vocabulary." } ; HELP: root-class diff --git a/basis/cocoa/messages/messages.factor b/basis/cocoa/messages/messages.factor index 09b2255913..791674428b 100644 --- a/basis/cocoa/messages/messages.factor +++ b/basis/cocoa/messages/messages.factor @@ -5,7 +5,8 @@ combinators compiler compiler.alien kernel math namespaces make parser prettyprint prettyprint.sections quotations sequences strings words cocoa.runtime io macros memoize debugger io.encodings.ascii effects libc libc.private parser lexer init -core-foundation fry ; +core-foundation fry generalizations +specialized-arrays.direct.alien ; IN: cocoa.messages : make-sender ( method function -- quot ) @@ -27,7 +28,7 @@ super-message-senders global [ H{ } assoc-like ] change-at : cache-stub ( method function hash -- ) [ - over get [ 2drop ] [ over >r sender-stub r> set ] if + over get [ 2drop ] [ over [ sender-stub ] dip set ] if ] bind ; : cache-stubs ( method -- ) @@ -37,7 +38,7 @@ super-message-senders global [ H{ } assoc-like ] change-at : ( receiver -- super ) "objc-super" [ - >r dup object_getClass class_getSuperclass r> + [ dup object_getClass class_getSuperclass ] dip set-objc-super-class ] keep [ set-objc-super-receiver ] keep ; @@ -62,23 +63,18 @@ objc-methods global [ H{ } assoc-like ] change-at dup objc-methods get at [ ] [ "No such method: " prepend throw ] ?if ; -: make-dip ( quot n -- quot' ) - dup - \ >r >quotation -rot - \ r> >quotation 3append ; - MEMO: make-prepare-send ( selector method super? -- quot ) [ [ \ , ] when swap , \ selector , ] [ ] make - swap second length 2 - make-dip ; + swap second length 2 - '[ _ _ ndip ] ; MACRO: (send) ( selector super? -- quot ) - >r dup lookup-method r> + [ dup lookup-method ] dip [ make-prepare-send ] 2keep super-message-senders message-senders ? get at - [ slip execute ] 2curry ; + '[ _ call _ execute ] ; : send ( receiver args... selector -- return... ) f (send) ; inline @@ -165,14 +161,14 @@ objc>alien-types get [ swap ] assoc-map assoc-union alien>objc-types set-global : objc-struct-type ( i string -- ctype ) - 2dup CHAR: = -rot index-from swap subseq + [ CHAR: = ] 2keep index-from swap subseq dup c-types get key? [ "Warning: no such C type: " write dup print drop "void*" ] unless ; : (parse-objc-type) ( i string -- ctype ) - 2dup nth >r >r 1+ r> r> { + [ [ 1+ ] dip ] [ nth ] 2bi { { [ dup "rnNoORV" member? ] [ drop (parse-objc-type) ] } { [ dup CHAR: ^ = ] [ 3drop "void*" ] } { [ dup CHAR: { = ] [ drop objc-struct-type ] } @@ -203,8 +199,11 @@ assoc-union alien>objc-types set-global objc-methods get set-at ; : each-method-in-class ( class quot -- ) - [ 0 [ class_copyMethodList ] keep *uint over ] dip - '[ _ void*-nth @ ] each (free) ; inline + [ 0 [ class_copyMethodList ] keep *uint ] dip + over 0 = [ 3drop ] [ + [ ] dip + [ each ] [ drop underlying>> (free) ] 2bi + ] if ; inline : register-objc-methods ( class -- ) [ register-objc-method ] each-method-in-class ; @@ -223,22 +222,23 @@ assoc-union alien>objc-types set-global : class-exists? ( string -- class ) objc_getClass >boolean ; : unless-defined ( class quot -- ) - >r class-exists? r> unless ; inline + [ class-exists? ] dip unless ; inline : define-objc-class-word ( name quot -- ) [ over , , \ unless-defined , dup , \ objc-class , - ] [ ] make >r "cocoa.classes" create r> + ] [ ] make [ "cocoa.classes" create ] dip (( -- class )) define-declared ; : import-objc-class ( name quot -- ) 2dup unless-defined dupd define-objc-class-word - [ + '[ + _ dup objc-class register-objc-methods objc-meta-class register-objc-methods - ] curry try ; + ] try ; : root-class ( class -- root ) dup class_getSuperclass [ root-class ] [ ] ?if ; diff --git a/basis/cocoa/pasteboard/pasteboard.factor b/basis/cocoa/pasteboard/pasteboard.factor index d266c2452f..b530ccbc37 100644 --- a/basis/cocoa/pasteboard/pasteboard.factor +++ b/basis/cocoa/pasteboard/pasteboard.factor @@ -1,8 +1,8 @@ -! Copyright (C) 2006, 2007 Slava Pestov. +! Copyright (C) 2006, 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. -USING: alien.c-types arrays kernel cocoa.messages -cocoa.classes cocoa.application cocoa core-foundation -sequences ; +USING: alien.accessors arrays kernel cocoa.messages +cocoa.classes cocoa.application cocoa core-foundation sequences +; IN: cocoa.pasteboard : NSStringPboardType "NSStringPboardType" ; @@ -20,11 +20,11 @@ IN: cocoa.pasteboard : set-pasteboard-string ( str pasteboard -- ) NSStringPboardType dup 1array pick set-pasteboard-types - >r swap r> -> setString:forType: drop ; + [ swap ] dip -> setString:forType: drop ; : pasteboard-error ( error -- f ) "Pasteboard does not hold a string" - 0 spin set-void*-nth f ; + 0 set-alien-cell f ; : ?pasteboard-string ( pboard error -- str/f ) over pasteboard-string? [ diff --git a/basis/cocoa/subclassing/subclassing.factor b/basis/cocoa/subclassing/subclassing.factor index fd18c7fa89..40f21d25b8 100644 --- a/basis/cocoa/subclassing/subclassing.factor +++ b/basis/cocoa/subclassing/subclassing.factor @@ -36,7 +36,7 @@ IN: cocoa.subclassing ] map concat ; : prepare-method ( ret types quot -- type imp ) - >r [ encode-types ] 2keep r> [ + [ [ encode-types ] 2keep ] dip [ "cdecl" swap 4array % \ alien-callback , ] [ ] make define-temp ; diff --git a/basis/cocoa/views/views.factor b/basis/cocoa/views/views.factor index d03688b2be..be67f03184 100644 --- a/basis/cocoa/views/views.factor +++ b/basis/cocoa/views/views.factor @@ -1,8 +1,8 @@ -! Copyright (C) 2006, 2007 Slava Pestov +! Copyright (C) 2006, 2008 Slava Pestov ! See http://factorcode.org/license.txt for BSD license. -USING: alien.c-types arrays kernel math namespaces make cocoa -cocoa.messages cocoa.classes cocoa.types sequences -continuations ; +USING: specialized-arrays.int arrays kernel math namespaces make +cocoa cocoa.messages cocoa.classes cocoa.types sequences +continuations accessors ; IN: cocoa.views : NSOpenGLPFAAllRenderers 1 ; @@ -69,12 +69,12 @@ PRIVATE> NSOpenGLPFASamples , 8 , ] when 0 , - ] { } make >c-int-array + ] int-array{ } make underlying>> -> initWithAttributes: -> autorelease ; : ( class dim -- view ) - >r -> alloc 0 0 r> first2 + [ -> alloc 0 0 ] dip first2 -> initWithFrame:pixelFormat: dup 1 -> setPostsBoundsChangedNotifications: dup 1 -> setPostsFrameChangedNotifications: ; @@ -85,10 +85,11 @@ PRIVATE> swap NSRect-h >fixnum 2array ; : mouse-location ( view event -- loc ) - over >r - -> locationInWindow f -> convertPoint:fromView: - dup NSPoint-x swap NSPoint-y - r> -> frame NSRect-h swap - 2array ; + [ + -> locationInWindow f -> convertPoint:fromView: + [ NSPoint-x ] [ NSPoint-y ] bi + ] [ drop -> frame NSRect-h ] 2bi + swap - 2array ; USE: opengl.gl USE: alien.syntax diff --git a/basis/cocoa/windows/windows.factor b/basis/cocoa/windows/windows.factor index dd2d1bfd41..3a53a1cc3c 100644 --- a/basis/cocoa/windows/windows.factor +++ b/basis/cocoa/windows/windows.factor @@ -34,5 +34,6 @@ IN: cocoa.windows dup 0 -> setReleasedWhenClosed: ; : window-content-rect ( window -- rect ) - NSWindow over -> frame rot -> styleMask + [ NSWindow ] dip + [ -> frame ] [ -> styleMask ] bi -> contentRectForFrameRect:styleMask: ; diff --git a/basis/combinators/short-circuit/short-circuit-docs.factor b/basis/combinators/short-circuit/short-circuit-docs.factor index 54fc3aac43..6cd18201fe 100644 --- a/basis/combinators/short-circuit/short-circuit-docs.factor +++ b/basis/combinators/short-circuit/short-circuit-docs.factor @@ -52,17 +52,17 @@ HELP: 3|| { "quot" quotation } } { $description "Returns true if any quotation in the sequence returns true. Each quotation takes the same three elements from the datastack and must return a boolean." } ; -HELP: n&&-rewrite +HELP: n&& { $values { "quots" "a sequence of quotations" } { "N" integer } { "quot" quotation } } -{ $description "A macro that reqrites the code to pass " { $snippet "N" } " parameters from the stack to each AND quotation." } ; +{ $description "A macro that rewrites the code to pass " { $snippet "n" } " parameters from the stack to each AND quotation." } ; -HELP: n||-rewrite +HELP: n|| { $values - { "quots" "a sequence of quotations" } { "N" integer } + { "quots" "a sequence of quotations" } { "n" integer } { "quot" quotation } } -{ $description "A macro that reqrites the code to pass " { $snippet "N" } " parameters from the stack to each OR quotation." } ; +{ $description "A macro that rewrites the code to pass " { $snippet "n" } " parameters from the stack to each OR quotation." } ; ARTICLE: "combinators.short-circuit" "Short-circuit combinators" "The " { $vocab-link "combinators.short-circuit" } " vocabulary stops a computation early once a condition is met." $nl @@ -77,8 +77,8 @@ ARTICLE: "combinators.short-circuit" "Short-circuit combinators" { $subsection 2|| } { $subsection 3|| } "Generalized combinators:" -{ $subsection n&&-rewrite } -{ $subsection n||-rewrite } +{ $subsection n&& } +{ $subsection n|| } ; ABOUT: "combinators.short-circuit" diff --git a/basis/combinators/short-circuit/short-circuit.factor b/basis/combinators/short-circuit/short-circuit.factor index 7b6c1d126d..d8bab4dd34 100644 --- a/basis/combinators/short-circuit/short-circuit.factor +++ b/basis/combinators/short-circuit/short-circuit.factor @@ -1,35 +1,33 @@ - USING: kernel combinators quotations arrays sequences assocs - locals generalizations macros fry ; - +locals generalizations macros fry ; IN: combinators.short-circuit -! !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! +MACRO:: n&& ( quots n -- quot ) + [ f ] quots [| q | + n + [ q '[ drop _ ndup @ dup not ] ] + [ '[ drop _ ndrop f ] ] + bi 2array + ] map + n '[ _ nnip ] suffix 1array + [ cond ] 3append ; -:: n&&-rewrite ( quots N -- quot ) - quots - [ '[ drop N ndup @ dup not ] [ drop N ndrop f ] 2array ] - map - [ t ] [ N nnip ] 2array suffix - '[ f _ cond ] ; +MACRO: 0&& ( quots -- quot ) '[ _ 0 n&& ] ; +MACRO: 1&& ( quots -- quot ) '[ _ 1 n&& ] ; +MACRO: 2&& ( quots -- quot ) '[ _ 2 n&& ] ; +MACRO: 3&& ( quots -- quot ) '[ _ 3 n&& ] ; -MACRO: 0&& ( quots -- quot ) 0 n&&-rewrite ; -MACRO: 1&& ( quots -- quot ) 1 n&&-rewrite ; -MACRO: 2&& ( quots -- quot ) 2 n&&-rewrite ; -MACRO: 3&& ( quots -- quot ) 3 n&&-rewrite ; +MACRO:: n|| ( quots n -- quot ) + [ f ] quots [| q | + n + [ q '[ drop _ ndup @ dup ] ] + [ '[ _ nnip ] ] + bi 2array + ] map + n '[ drop _ ndrop t ] [ f ] 2array suffix 1array + [ cond ] 3append ; -! !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! - -:: n||-rewrite ( quots N -- quot ) - quots - [ '[ drop N ndup @ dup ] [ N nnip ] 2array ] - map - [ drop N ndrop t ] [ f ] 2array suffix - '[ f _ cond ] ; - -MACRO: 0|| ( quots -- quot ) 0 n||-rewrite ; -MACRO: 1|| ( quots -- quot ) 1 n||-rewrite ; -MACRO: 2|| ( quots -- quot ) 2 n||-rewrite ; -MACRO: 3|| ( quots -- quot ) 3 n||-rewrite ; - -! !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! +MACRO: 0|| ( quots -- quot ) '[ _ 0 n|| ] ; +MACRO: 1|| ( quots -- quot ) '[ _ 1 n|| ] ; +MACRO: 2|| ( quots -- quot ) '[ _ 2 n|| ] ; +MACRO: 3|| ( quots -- quot ) '[ _ 3 n|| ] ; diff --git a/basis/combinators/short-circuit/smart/smart.factor b/basis/combinators/short-circuit/smart/smart.factor index ca659cacbe..b80e7294d1 100644 --- a/basis/combinators/short-circuit/smart/smart.factor +++ b/basis/combinators/short-circuit/smart/smart.factor @@ -1,7 +1,5 @@ - USING: kernel sequences math stack-checker effects accessors macros - combinators.short-circuit ; - +fry combinators.short-circuit ; IN: combinators.short-circuit.smart -MACRO: && ( quots -- quot ) dup arity n&&-rewrite ; +MACRO: && ( quots -- quot ) dup arity '[ _ _ n&& ] ; -MACRO: || ( quots -- quot ) dup arity n||-rewrite ; +MACRO: || ( quots -- quot ) dup arity '[ _ _ n|| ] ; diff --git a/basis/command-line/command-line-docs.factor b/basis/command-line/command-line-docs.factor index d1b18ab5da..3d06bd97b7 100644 --- a/basis/command-line/command-line-docs.factor +++ b/basis/command-line/command-line-docs.factor @@ -1,13 +1,17 @@ -USING: help.markup help.syntax parser vocabs.loader strings ; +USING: help.markup help.syntax parser vocabs.loader strings +command-line.private ; IN: command-line HELP: run-bootstrap-init -{ $description "Runs the " { $snippet ".factor-boot-rc" } " file in the user's home directory unless the " { $snippet "-no-user-init" } " command line switch was given." } ; +{ $description "Runs the bootstrap initialization file in the user's home directory, unless the " { $snippet "-no-user-init" } " command line switch was given. This file is named " { $snippet ".factor-boot-rc" } " on Unix and " { $snippet "factor-boot-rc" } " on Windows." } ; HELP: run-user-init -{ $description "Runs the " { $snippet ".factor-rc" } " file in the user's home directory unless the " { $snippet "-no-user-init" } " command line switch was given." } ; +{ $description "Runs the startup initialization file in the user's home directory, unless the " { $snippet "-no-user-init" } " command line switch was given. This file is named " { $snippet ".factor-rc" } " on Unix and " { $snippet "factor-rc" } " on Windows." } ; -HELP: cli-param +HELP: load-vocab-roots +{ $description "Loads the newline-separated list of additional vocabulary roots from the file named " { $snippet ".factor-roots" } " on Unix and " { $snippet "factor-roots" } " on Windows." } ; + +HELP: param { $values { "param" string } } { $description "Process a command-line switch." $nl @@ -17,10 +21,13 @@ $nl $nl "Otherwise, sets the global variable named by the parameter to " { $link t } "." } ; -HELP: cli-args +HELP: (command-line) { $values { "args" "a sequence of strings" } } { $description "Outputs the command line parameters which were passed to the Factor VM on startup." } ; +HELP: command-line +{ $var-description "The command line parameters which follow the name of the script on the command line." } ; + HELP: main-vocab-hook { $var-description "Global variable holding a quotation which outputs a vocabulary name. UI backends set this so that the UI can automatically start if the prerequisites are met (for example, " { $snippet "$DISPLAY" } " being set on X11)." } ; @@ -35,9 +42,6 @@ HELP: ignore-cli-args? { $values { "?" "a boolean" } } { $description "On Mac OS X, source files to run are supplied by the Cocoa API, so to avoid running them twice the startup code has to call this word." } ; -HELP: parse-command-line -{ $description "Called on startup to process command line arguments. This sets global variables with " { $link cli-param } ", runs source files, and evaluates the string given by the " { $snippet "-e" } " switch, if there is one." } ; - ARTICLE: "runtime-cli-args" "Command line switches for the VM" "A handful of command line switches are processed by the VM and not the library. They control low-level features." { $table @@ -57,16 +61,19 @@ ARTICLE: "bootstrap-cli-args" "Command line switches for bootstrap" "A number of command line switches can be passed to a bootstrap image to modify the behavior of the resulting image:" { $table { { $snippet "-output-image=" { $emphasis "image" } } { "Save the result to " { $snippet "image" } ". The default is " { $snippet "factor.image" } "." } } - { { $snippet "-no-user-init" } { "Inhibits the running of the " { $snippet ".factor-boot-rc" } " file in the user's home directory." } } + { { $snippet "-no-user-init" } { "Inhibits the running of user initialization files on startup. See " { $link "rc-files" } "." } } { { $snippet "-include=" { $emphasis "components..." } } "A list of components to include (see below)." } { { $snippet "-exclude=" { $emphasis "components..." } } "A list of components to exclude." } { { $snippet "-ui-backend=" { $emphasis "backend" } } { "One of " { $snippet "x11" } ", " { $snippet "windows" } ", or " { $snippet "cocoa" } ". The default is platform-specific." } } } "Bootstrap can load various optional components:" { $table + { { $snippet "math" } "Rational and complex number support." } + { { $snippet "threads" } "Thread support." } { { $snippet "compiler" } "The compiler." } { { $snippet "tools" } "Terminal-based developer tools." } { { $snippet "help" } "The help system." } + { { $snippet "help.handbook" } "The help handbook." } { { $snippet "ui" } "The graphical user interface." } { { $snippet "ui.tools" } "Graphical developer tools." } { { $snippet "io" } "Non-blocking I/O and networking." } @@ -74,9 +81,9 @@ ARTICLE: "bootstrap-cli-args" "Command line switches for bootstrap" "By default, all optional components are loaded. To load all optional components except for a given list, use the " { $snippet "-exclude=" } " switch; to only load specified optional components, use the " { $snippet "-include=" } "." $nl "For example, to build an image with the compiler but no other components, you could do:" -{ $code "./factor -i=boot.ppc.image -include=compiler" } +{ $code "./factor -i=boot.macosx-ppc.image -include=compiler" } "To build an image with everything except for the user interface and graphical tools," -{ $code "./factor -i=boot.ppc.image -exclude=\"ui ui.tools\"" } +{ $code "./factor -i=boot.macosx-ppc.image -exclude=\"ui ui.tools\"" } "To generate a bootstrap image in the first place, see " { $link "bootstrap.image" } "." ; ARTICLE: "standard-cli-args" "Command line switches for general usage" @@ -84,20 +91,60 @@ ARTICLE: "standard-cli-args" "Command line switches for general usage" { $table { { $snippet "-e=" { $emphasis "code" } } { "This specifies a code snippet to evaluate. If you want Factor to exit immediately after, also specify " { $snippet "-run=none" } "." } } { { $snippet "-run=" { $emphasis "vocab" } } { { $snippet { $emphasis "vocab" } } " is the name of a vocabulary with a " { $link POSTPONE: MAIN: } " hook to run on startup, for example " { $vocab-link "listener" } ", " { $vocab-link "ui" } " or " { $vocab-link "none" } "." } } - { { $snippet "-no-user-init" } { "Inhibits the running of the " { $snippet ".factor-rc" } " file in the user's home directory on startup." } } + { { $snippet "-no-user-init" } { "Inhibits the running of user initialization files on startup. See " { $link "rc-files" } "." } } { { $snippet "-quiet" } { "If set, " { $link run-file } " and " { $link require } " will not print load messages." } } - { { $snippet "-script" } { "Equivalent to " { $snippet "-quiet -run=none" } "." $nl "On Unix systems, Factor can be used for scripting - just create an executable text file whose first line is:" { $code "#! /usr/local/bin/factor -script" } "The space after " { $snippet "#!" } " is necessary because of Factor syntax." } } } ; -ARTICLE: "rc-files" "Running code on startup" -"Unless the " { $snippet "-no-user-init" } " command line switch is specified, The startup routine runs the " { $snippet ".factor-rc" } " file in the user's home directory, if it exists. This file can contain initialization and customization for your development environment." +ARTICLE: "factor-boot-rc" "Bootstrap initialization file" +"The botstrap initialization file is named " { $snippet "factor-boot-rc" } " on Windows and " { $snippet ".factor-boot-rc" } " on Unix. This file can contain " { $link require } " calls for vocabularies you use frequently, and other such long-running tasks that you do not want to perform every time Factor starts." $nl -"The " { $snippet ".factor-rc" } " and " { $snippet ".factor-boot-rc" } " files can be run explicitly:" -{ $subsection run-user-init } -{ $subsection run-bootstrap-init } ; +"A word to run this file from an existing Factor session:" +{ $subsection run-bootstrap-init } +"For example, if you changed " { $snippet ".factor-boot-rc" } " and do not want to bootstrap again, you can just invoke " { $link run-bootstrap-init } " in the listener." ; -ARTICLE: "cli" "Command line usage" -"Zero or more command line arguments may be passed to the Factor runtime. Command line arguments starting with a dash (" { $snippet "-" } ") is interpreted as switches. All other arguments are taken to be file names to be run by " { $link run-file } "." +ARTICLE: "factor-rc" "Startup initialization file" +"The startup initialization file is named " { $snippet "factor-rc" } " on Windows and " { $snippet ".factor-rc" } " on Unix. If it exists, it is run every time Factor starts." +$nl +"A word to run this file from an existing Factor session:" +{ $subsection run-user-init } ; + +ARTICLE: "factor-roots" "Additional vocabulary roots file" +"The vocabulary roots file is named " { $snippet "factor-roots" } " on Windows and " { $snippet ".factor-roots" } " on Unix. If it exists, it is loaded every time Factor starts. It contains a newline-separated list of " { $link "vocabs.roots" } "." +$nl +"A word to run this file from an existing Factor session:" +{ $subsection load-vocab-roots } ; + +ARTICLE: "rc-files" "Running code on startup" +"Factor looks for three optional files in your home directory." +{ $subsection "factor-boot-rc" } +{ $subsection "factor-rc" } +{ $subsection "factor-roots" } +"The " { $snippet "-no-user-init" } " command line switch will inhibit loading running of these files." +$nl +"If you are unsure where the files should be located, evaluate the following code:" +{ $code + "USE: command-line" + "\"factor-rc\" rc-path print" + "\"factor-boot-rc\" rc-path print" +} +"Here is an example " { $snippet ".factor-boot-rc" } " which sets up GVIM editor integration, adds an additional vocabulary root (see " { $link "vocabs.roots" } "), and increases the font size in the UI by setting the DPI (dots-per-inch) variable:" +{ $code + "USING: editors.gvim vocabs.loader ui.freetype namespaces sequences ;" + "\"/opt/local/bin\" \\ gvim-path set-global" + "\"/home/jane/src/\" vocab-roots get push" + "100 dpi set-global" +} ; + +ARTICLE: "cli" "Command line arguments" +"Factor command line usage:" +{ $code "factor [system switches...] [script args...]" } +"Zero or more system switches can be passed in, followed by an optional script file name. If the script file is specified, it will be run on startup, any arguments after the script file are stored in a variable, with no further processing by Factor itself:" +{ $subsection command-line } +"Instead of running a script, it is also possible to run a vocabulary; this invokes the vocabulary's " { $link POSTPONE: MAIN: } " word:" +{ $code "factor [system switches...] -run=" } +"If no script file or " { $snippet "-run=" } " switch is specified, Factor will start " { $link "listener" } " or " { $link "ui-tools" } ", depending on the operating system." +$nl +"As stated above, arguments in the first part of the command line, before the optional script name, are interpreted by to the Factor system. These arguments all start with a dash (" { $snippet "-" } ")." $nl "Switches can take one of the following three forms:" { $list @@ -108,9 +155,9 @@ $nl { $subsection "runtime-cli-args" } { $subsection "bootstrap-cli-args" } { $subsection "standard-cli-args" } -"The list of command line arguments can be obtained and inspected directly:" -{ $subsection cli-args } -"There is a way to override the default vocabulary to run on startup:" +"The raw list of command line arguments can also be obtained and inspected directly:" +{ $subsection (command-line) } +"There is a way to override the default vocabulary to run on startup, if no script name or " { $snippet "-run" } " switch is specified:" { $subsection main-vocab-hook } ; ABOUT: "cli" diff --git a/basis/command-line/command-line-tests.factor b/basis/command-line/command-line-tests.factor deleted file mode 100644 index 226765bafe..0000000000 --- a/basis/command-line/command-line-tests.factor +++ /dev/null @@ -1,12 +0,0 @@ -USING: namespaces tools.test kernel command-line ; -IN: command-line.tests - -[ - [ f ] [ "-no-user-init" cli-arg ] unit-test - [ f ] [ "user-init" get ] unit-test - - [ f ] [ "-user-init" cli-arg ] unit-test - [ t ] [ "user-init" get ] unit-test - - [ "sdl.factor" ] [ "sdl.factor" cli-arg ] unit-test -] with-scope diff --git a/basis/command-line/command-line.factor b/basis/command-line/command-line.factor index 37dbf9b7a6..1b58053b64 100644 --- a/basis/command-line/command-line.factor +++ b/basis/command-line/command-line.factor @@ -1,31 +1,56 @@ ! Copyright (C) 2003, 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. -USING: init continuations debugger hashtables io kernel -kernel.private namespaces parser sequences strings system -splitting io.files eval ; +USING: init continuations debugger hashtables io +io.encodings.utf8 io.files kernel kernel.private namespaces +parser sequences strings system splitting eval vocabs.loader ; IN: command-line +SYMBOL: script +SYMBOL: command-line + +: (command-line) ( -- args ) 10 getenv sift ; + +: rc-path ( name -- path ) + os windows? [ "." prepend ] unless + home prepend-path ; + : run-bootstrap-init ( -- ) "user-init" get [ - home ".factor-boot-rc" append-path ?run-file + "factor-boot-rc" rc-path ?run-file ] when ; : run-user-init ( -- ) "user-init" get [ - home ".factor-rc" append-path ?run-file + "factor-rc" rc-path ?run-file ] when ; -: cli-var-param ( name value -- ) swap set-global ; +: load-vocab-roots ( -- ) + "user-init" get [ + "factor-roots" rc-path dup exists? [ + utf8 file-lines [ add-vocab-root ] each + ] [ drop ] if + ] when ; -: cli-bool-param ( name -- ) "no-" ?head not cli-var-param ; + + +: parse-command-line ( args -- ) + [ command-line off script off ] [ + unclip "-" ?head + [ param parse-command-line ] + [ script set command-line set ] if + ] if-empty ; SYMBOL: main-vocab-hook @@ -49,14 +74,17 @@ SYMBOL: main-vocab-hook : ignore-cli-args? ( -- ? ) os macosx? "run" get "ui" = and ; -: script-mode ( -- ) - t "quiet" set-global - "none" "run" set-global ; +: script-mode ( -- ) ; -: parse-command-line ( -- ) - cli-args [ cli-arg ] filter - "script" get [ script-mode ] when - ignore-cli-args? [ drop ] [ [ run-file ] each ] if - "e" get [ eval ] when* ; +: handle-command-line ( -- ) + [ + (command-line) parse-command-line + load-vocab-roots + run-user-init + "e" get [ eval ] when* + ignore-cli-args? not script get and + [ run-script ] [ "run" get run ] if* + output-stream get [ stream-flush ] when* + ] [ print-error 1 exit ] recover ; [ default-cli-args ] "command-line" add-init-hook diff --git a/basis/compiler/alien/alien.factor b/basis/compiler/alien/alien.factor index e414d6e29b..4a41014ab2 100644 --- a/basis/compiler/alien/alien.factor +++ b/basis/compiler/alien/alien.factor @@ -18,7 +18,7 @@ IN: compiler.alien dup c-type-stack-align? [ c-type-align ] [ drop cell ] if ; : parameter-align ( n type -- n delta ) - over >r c-type-stack-align align dup r> - ; + [ c-type-stack-align align dup ] [ drop ] 2bi - ; : parameter-sizes ( types -- total offsets ) #! Compute stack frame locations. diff --git a/basis/compiler/cfg/builder/builder.factor b/basis/compiler/cfg/builder/builder.factor index 17a5942af2..9ffe4a6aa0 100755 --- a/basis/compiler/cfg/builder/builder.factor +++ b/basis/compiler/cfg/builder/builder.factor @@ -21,8 +21,6 @@ IN: compiler.cfg.builder ! Convert tree SSA IR to CFG SSA IR. -: stop-iterating ( -- next ) end-basic-block f ; - SYMBOL: procedures SYMBOL: current-word SYMBOL: current-label @@ -190,7 +188,7 @@ M: #if emit-node : emit-dispatch ( node -- ) ##epilogue - ds-pop ^^offset>slot i ##dispatch + ds-pop ^^offset>slot i 0 ##dispatch dispatch-branches ; : ( -- word ) @@ -211,7 +209,7 @@ M: #dispatch emit-node ! #call M: #call emit-node dup word>> dup "intrinsic" word-prop - [ emit-intrinsic iterate-next ] [ nip emit-call ] if ; + [ emit-intrinsic ] [ nip emit-call ] if ; ! #call-recursive M: #call-recursive emit-node label>> id>> emit-call ; @@ -221,21 +219,14 @@ M: #push emit-node literal>> ^^load-literal ds-push iterate-next ; ! #shuffle -: emit-shuffle ( effect -- ) - [ out>> ] [ in>> dup length ds-load zip ] bi - '[ _ at ] map ds-store ; - M: #shuffle emit-node - shuffle-effect emit-shuffle iterate-next ; - -M: #>r emit-node - [ in-d>> length ] [ out-r>> empty? ] bi - [ neg ##inc-d ] [ ds-load rs-store ] if - iterate-next ; - -M: #r> emit-node - [ in-r>> length ] [ out-d>> empty? ] bi - [ neg ##inc-r ] [ rs-load ds-store ] if + dup + H{ } clone + [ [ in-d>> [ length ds-load ] keep ] dip '[ _ set-at ] 2each ] + [ [ in-r>> [ length rs-load ] keep ] dip '[ _ set-at ] 2each ] + [ nip ] 2tri + [ [ [ out-d>> ] [ mapping>> ] bi ] dip '[ _ at _ at ] map ds-store ] + [ [ [ out-r>> ] [ mapping>> ] bi ] dip '[ _ at _ at ] map rs-store ] 2bi iterate-next ; ! #return @@ -269,7 +260,7 @@ M: #terminate emit-node drop stop-iterating ; : emit-alien-node ( node quot -- next ) [ params>> ] dip [ drop alien-stack-frame ] [ call ] 2bi - begin-basic-block iterate-next ; inline + ##branch begin-basic-block iterate-next ; inline M: #alien-invoke emit-node [ ##alien-invoke ] emit-alien-node ; diff --git a/basis/compiler/cfg/def-use/def-use.factor b/basis/compiler/cfg/def-use/def-use.factor index 7553407e00..3825ae480e 100644 --- a/basis/compiler/cfg/def-use/def-use.factor +++ b/basis/compiler/cfg/def-use/def-use.factor @@ -12,9 +12,14 @@ M: ##write-barrier defs-vregs [ card#>> ] [ table>> ] bi 2array ; M: ##unary/temp defs-vregs dst/tmp-vregs ; M: ##allot defs-vregs dst/tmp-vregs ; M: ##dispatch defs-vregs temp>> 1array ; -M: ##slot defs-vregs [ dst>> ] [ temp>> ] bi 2array ; +M: ##slot defs-vregs dst/tmp-vregs ; M: ##set-slot defs-vregs temp>> 1array ; -M: ##string-nth defs-vregs [ dst>> ] [ temp>> ] bi 2array ; +M: ##string-nth defs-vregs dst/tmp-vregs ; +M: ##compare defs-vregs dst/tmp-vregs ; +M: ##compare-imm defs-vregs dst/tmp-vregs ; +M: ##compare-float defs-vregs dst/tmp-vregs ; +M: ##fixnum-mul defs-vregs [ temp1>> ] [ temp2>> ] bi 2array ; +M: ##fixnum-mul-tail defs-vregs [ temp1>> ] [ temp2>> ] bi 2array ; M: insn defs-vregs drop f ; M: ##unary uses-vregs src>> 1array ; @@ -31,6 +36,7 @@ M: ##compare-imm-branch uses-vregs src1>> 1array ; M: ##dispatch uses-vregs src>> 1array ; M: ##alien-getter uses-vregs src>> 1array ; M: ##alien-setter uses-vregs [ src>> ] [ value>> ] bi 2array ; +M: ##fixnum-overflow uses-vregs [ src1>> ] [ src2>> ] bi 2array ; M: _conditional-branch uses-vregs [ src1>> ] [ src2>> ] bi 2array ; M: _compare-imm-branch uses-vregs src1>> 1array ; M: insn uses-vregs drop f ; @@ -40,6 +46,7 @@ UNION: vreg-insn ##write-barrier ##dispatch ##effect +##fixnum-overflow ##conditional-branch ##compare-imm-branch _conditional-branch diff --git a/basis/compiler/cfg/hats/hats.factor b/basis/compiler/cfg/hats/hats.factor index e6e05abbd5..4b98ccb0ae 100644 --- a/basis/compiler/cfg/hats/hats.factor +++ b/basis/compiler/cfg/hats/hats.factor @@ -65,9 +65,9 @@ IN: compiler.cfg.hats : ^^alien-cell ( src -- dst ) ^^i1 ##alien-cell ; inline : ^^alien-float ( src -- dst ) ^^d1 ##alien-float ; inline : ^^alien-double ( src -- dst ) ^^d1 ##alien-double ; inline -: ^^compare ( src1 src2 cc -- dst ) ^^i3 ##compare ; inline -: ^^compare-imm ( src1 src2 cc -- dst ) ^^i3 ##compare-imm ; inline -: ^^compare-float ( src1 src2 cc -- dst ) ^^i3 ##compare-float ; inline +: ^^compare ( src1 src2 cc -- dst ) ^^i3 i ##compare ; inline +: ^^compare-imm ( src1 src2 cc -- dst ) ^^i3 i ##compare-imm ; inline +: ^^compare-float ( src1 src2 cc -- dst ) ^^i3 i ##compare-float ; inline : ^^offset>slot ( vreg -- vreg' ) cell 4 = [ 1 ^^shr-imm ] when ; inline : ^^tag-fixnum ( src -- dst ) ^^i1 ##tag-fixnum ; inline : ^^untag-fixnum ( src -- dst ) ^^i1 ##untag-fixnum ; inline diff --git a/basis/compiler/cfg/instructions/instructions.factor b/basis/compiler/cfg/instructions/instructions.factor index c39f517671..62d4990c92 100644 --- a/basis/compiler/cfg/instructions/instructions.factor +++ b/basis/compiler/cfg/instructions/instructions.factor @@ -62,7 +62,7 @@ INSN: ##jump word ; INSN: ##return ; ! Jump tables -INSN: ##dispatch src temp ; +INSN: ##dispatch src temp offset ; INSN: ##dispatch-label label ; ! Slot access @@ -92,6 +92,15 @@ INSN: ##shr-imm < ##binary-imm ; INSN: ##sar-imm < ##binary-imm ; INSN: ##not < ##unary ; +! Overflowing arithmetic +TUPLE: ##fixnum-overflow < insn src1 src2 ; +INSN: ##fixnum-add < ##fixnum-overflow ; +INSN: ##fixnum-add-tail < ##fixnum-overflow ; +INSN: ##fixnum-sub < ##fixnum-overflow ; +INSN: ##fixnum-sub-tail < ##fixnum-overflow ; +INSN: ##fixnum-mul < ##fixnum-overflow temp1 temp2 ; +INSN: ##fixnum-mul-tail < ##fixnum-overflow temp1 temp2 ; + : ##tag-fixnum ( dst src -- ) tag-bits get ##shl-imm ; inline : ##untag-fixnum ( dst src -- ) tag-bits get ##sar-imm ; inline @@ -198,11 +207,11 @@ TUPLE: ##conditional-branch < insn { src1 vreg } { src2 vreg } cc ; INSN: ##compare-branch < ##conditional-branch ; INSN: ##compare-imm-branch { src1 vreg } { src2 integer } cc ; -INSN: ##compare < ##binary cc ; -INSN: ##compare-imm < ##binary-imm cc ; +INSN: ##compare < ##binary cc temp ; +INSN: ##compare-imm < ##binary-imm cc temp ; INSN: ##compare-float-branch < ##conditional-branch ; -INSN: ##compare-float < ##binary cc ; +INSN: ##compare-float < ##binary cc temp ; ! Instructions used by machine IR only. INSN: _prologue stack-frame ; diff --git a/basis/compiler/cfg/intrinsics/fixnum/fixnum.factor b/basis/compiler/cfg/intrinsics/fixnum/fixnum.factor index 04c9097725..68ee7489f8 100644 --- a/basis/compiler/cfg/intrinsics/fixnum/fixnum.factor +++ b/basis/compiler/cfg/intrinsics/fixnum/fixnum.factor @@ -3,10 +3,22 @@ USING: sequences accessors layouts kernel math namespaces combinators fry locals compiler.tree.propagation.info -compiler.cfg.stacks compiler.cfg.hats compiler.cfg.instructions -compiler.cfg.utilities ; +compiler.cfg.hats +compiler.cfg.stacks +compiler.cfg.iterator +compiler.cfg.instructions +compiler.cfg.utilities +compiler.cfg.registers ; IN: compiler.cfg.intrinsics.fixnum +: emit-both-fixnums? ( -- ) + D 0 ^^peek + D 1 ^^peek + ^^or + tag-mask get ^^and-imm + 0 cc= ^^compare-imm + ds-push ; + : (emit-fixnum-imm-op) ( infos insn -- dst ) ds-drop [ ds-pop ] @@ -64,3 +76,16 @@ IN: compiler.cfg.intrinsics.fixnum : emit-fixnum>bignum ( -- ) ds-pop ^^untag-fixnum ^^integer>bignum ds-push ; + +: emit-fixnum-overflow-op ( quot quot-tail -- next ) + [ 2inputs 1 ##inc-d ] 2dip + tail-call? [ + ##epilogue + nip call + stop-iterating + ] [ + drop call + ##branch + begin-basic-block + iterate-next + ] if ; inline diff --git a/basis/compiler/cfg/intrinsics/intrinsics.factor b/basis/compiler/cfg/intrinsics/intrinsics.factor index ef1cde337a..aaa45c3937 100644 --- a/basis/compiler/cfg/intrinsics/intrinsics.factor +++ b/basis/compiler/cfg/intrinsics/intrinsics.factor @@ -8,7 +8,8 @@ compiler.cfg.intrinsics.alien compiler.cfg.intrinsics.allot compiler.cfg.intrinsics.fixnum compiler.cfg.intrinsics.float -compiler.cfg.intrinsics.slots ; +compiler.cfg.intrinsics.slots +compiler.cfg.iterator ; QUALIFIED: kernel QUALIFIED: arrays QUALIFIED: byte-arrays @@ -22,6 +23,10 @@ IN: compiler.cfg.intrinsics { kernel.private:tag + math.private:both-fixnums? + math.private:fixnum+ + math.private:fixnum- + math.private:fixnum* math.private:fixnum+fast math.private:fixnum-fast math.private:fixnum-bitand @@ -85,60 +90,64 @@ IN: compiler.cfg.intrinsics alien.accessors:set-alien-double } [ t "intrinsic" set-word-prop ] each ; -: emit-intrinsic ( node word -- ) +: emit-intrinsic ( node word -- node/f ) { - { \ kernel.private:tag [ drop emit-tag ] } - { \ math.private:fixnum+fast [ [ ^^add ] [ ^^add-imm ] emit-fixnum-op ] } - { \ math.private:fixnum-fast [ [ ^^sub ] [ ^^sub-imm ] emit-fixnum-op ] } - { \ math.private:fixnum-bitand [ [ ^^and ] [ ^^and-imm ] emit-fixnum-op ] } - { \ math.private:fixnum-bitor [ [ ^^or ] [ ^^or-imm ] emit-fixnum-op ] } - { \ math.private:fixnum-bitxor [ [ ^^xor ] [ ^^xor-imm ] emit-fixnum-op ] } - { \ math.private:fixnum-shift-fast [ emit-fixnum-shift-fast ] } - { \ math.private:fixnum-bitnot [ drop emit-fixnum-bitnot ] } - { \ math.private:fixnum*fast [ emit-fixnum*fast ] } - { \ math.private:fixnum< [ cc< emit-fixnum-comparison ] } - { \ math.private:fixnum<= [ cc<= emit-fixnum-comparison ] } - { \ math.private:fixnum>= [ cc>= emit-fixnum-comparison ] } - { \ math.private:fixnum> [ cc> emit-fixnum-comparison ] } - { \ kernel:eq? [ cc= emit-fixnum-comparison ] } - { \ math.private:bignum>fixnum [ drop emit-bignum>fixnum ] } - { \ math.private:fixnum>bignum [ drop emit-fixnum>bignum ] } - { \ math.private:float+ [ drop [ ^^add-float ] emit-float-op ] } - { \ math.private:float- [ drop [ ^^sub-float ] emit-float-op ] } - { \ math.private:float* [ drop [ ^^mul-float ] emit-float-op ] } - { \ math.private:float/f [ drop [ ^^div-float ] emit-float-op ] } - { \ math.private:float< [ drop cc< emit-float-comparison ] } - { \ math.private:float<= [ drop cc<= emit-float-comparison ] } - { \ math.private:float>= [ drop cc>= emit-float-comparison ] } - { \ math.private:float> [ drop cc> emit-float-comparison ] } - { \ math.private:float= [ drop cc= emit-float-comparison ] } - { \ math.private:float>fixnum [ drop emit-float>fixnum ] } - { \ math.private:fixnum>float [ drop emit-fixnum>float ] } - { \ slots.private:slot [ emit-slot ] } - { \ slots.private:set-slot [ emit-set-slot ] } - { \ strings.private:string-nth [ drop emit-string-nth ] } - { \ classes.tuple.private: [ emit- ] } - { \ arrays: [ emit- ] } - { \ byte-arrays: [ emit- ] } - { \ math.private: [ emit-simple-allot ] } - { \ math.private: [ emit-simple-allot ] } - { \ kernel: [ emit-simple-allot ] } - { \ alien.accessors:alien-unsigned-1 [ 1 emit-alien-unsigned-getter ] } - { \ alien.accessors:set-alien-unsigned-1 [ 1 emit-alien-integer-setter ] } - { \ alien.accessors:alien-signed-1 [ 1 emit-alien-signed-getter ] } - { \ alien.accessors:set-alien-signed-1 [ 1 emit-alien-integer-setter ] } - { \ alien.accessors:alien-unsigned-2 [ 2 emit-alien-unsigned-getter ] } - { \ alien.accessors:set-alien-unsigned-2 [ 2 emit-alien-integer-setter ] } - { \ alien.accessors:alien-signed-2 [ 2 emit-alien-signed-getter ] } - { \ alien.accessors:set-alien-signed-2 [ 2 emit-alien-integer-setter ] } - { \ alien.accessors:alien-unsigned-4 [ 4 emit-alien-unsigned-getter ] } - { \ alien.accessors:set-alien-unsigned-4 [ 4 emit-alien-integer-setter ] } - { \ alien.accessors:alien-signed-4 [ 4 emit-alien-signed-getter ] } - { \ alien.accessors:set-alien-signed-4 [ 4 emit-alien-integer-setter ] } - { \ alien.accessors:alien-cell [ emit-alien-cell-getter ] } - { \ alien.accessors:set-alien-cell [ emit-alien-cell-setter ] } - { \ alien.accessors:alien-float [ single-float-regs emit-alien-float-getter ] } - { \ alien.accessors:set-alien-float [ single-float-regs emit-alien-float-setter ] } - { \ alien.accessors:alien-double [ double-float-regs emit-alien-float-getter ] } - { \ alien.accessors:set-alien-double [ double-float-regs emit-alien-float-setter ] } + { \ kernel.private:tag [ drop emit-tag iterate-next ] } + { \ math.private:both-fixnums? [ drop emit-both-fixnums? iterate-next ] } + { \ math.private:fixnum+ [ drop [ ##fixnum-add ] [ ##fixnum-add-tail ] emit-fixnum-overflow-op ] } + { \ math.private:fixnum- [ drop [ ##fixnum-sub ] [ ##fixnum-sub-tail ] emit-fixnum-overflow-op ] } + { \ math.private:fixnum* [ drop [ i i ##fixnum-mul ] [ i i ##fixnum-mul-tail ] emit-fixnum-overflow-op ] } + { \ math.private:fixnum+fast [ [ ^^add ] [ ^^add-imm ] emit-fixnum-op iterate-next ] } + { \ math.private:fixnum-fast [ [ ^^sub ] [ ^^sub-imm ] emit-fixnum-op iterate-next ] } + { \ math.private:fixnum-bitand [ [ ^^and ] [ ^^and-imm ] emit-fixnum-op iterate-next ] } + { \ math.private:fixnum-bitor [ [ ^^or ] [ ^^or-imm ] emit-fixnum-op iterate-next ] } + { \ math.private:fixnum-bitxor [ [ ^^xor ] [ ^^xor-imm ] emit-fixnum-op iterate-next ] } + { \ math.private:fixnum-shift-fast [ emit-fixnum-shift-fast iterate-next ] } + { \ math.private:fixnum-bitnot [ drop emit-fixnum-bitnot iterate-next ] } + { \ math.private:fixnum*fast [ emit-fixnum*fast iterate-next ] } + { \ math.private:fixnum< [ cc< emit-fixnum-comparison iterate-next ] } + { \ math.private:fixnum<= [ cc<= emit-fixnum-comparison iterate-next ] } + { \ math.private:fixnum>= [ cc>= emit-fixnum-comparison iterate-next ] } + { \ math.private:fixnum> [ cc> emit-fixnum-comparison iterate-next ] } + { \ kernel:eq? [ cc= emit-fixnum-comparison iterate-next ] } + { \ math.private:bignum>fixnum [ drop emit-bignum>fixnum iterate-next ] } + { \ math.private:fixnum>bignum [ drop emit-fixnum>bignum iterate-next ] } + { \ math.private:float+ [ drop [ ^^add-float ] emit-float-op iterate-next ] } + { \ math.private:float- [ drop [ ^^sub-float ] emit-float-op iterate-next ] } + { \ math.private:float* [ drop [ ^^mul-float ] emit-float-op iterate-next ] } + { \ math.private:float/f [ drop [ ^^div-float ] emit-float-op iterate-next ] } + { \ math.private:float< [ drop cc< emit-float-comparison iterate-next ] } + { \ math.private:float<= [ drop cc<= emit-float-comparison iterate-next ] } + { \ math.private:float>= [ drop cc>= emit-float-comparison iterate-next ] } + { \ math.private:float> [ drop cc> emit-float-comparison iterate-next ] } + { \ math.private:float= [ drop cc= emit-float-comparison iterate-next ] } + { \ math.private:float>fixnum [ drop emit-float>fixnum iterate-next ] } + { \ math.private:fixnum>float [ drop emit-fixnum>float iterate-next ] } + { \ slots.private:slot [ emit-slot iterate-next ] } + { \ slots.private:set-slot [ emit-set-slot iterate-next ] } + { \ strings.private:string-nth [ drop emit-string-nth iterate-next ] } + { \ classes.tuple.private: [ emit- iterate-next ] } + { \ arrays: [ emit- iterate-next ] } + { \ byte-arrays: [ emit- iterate-next ] } + { \ math.private: [ emit-simple-allot iterate-next ] } + { \ math.private: [ emit-simple-allot iterate-next ] } + { \ kernel: [ emit-simple-allot iterate-next ] } + { \ alien.accessors:alien-unsigned-1 [ 1 emit-alien-unsigned-getter iterate-next ] } + { \ alien.accessors:set-alien-unsigned-1 [ 1 emit-alien-integer-setter iterate-next ] } + { \ alien.accessors:alien-signed-1 [ 1 emit-alien-signed-getter iterate-next ] } + { \ alien.accessors:set-alien-signed-1 [ 1 emit-alien-integer-setter iterate-next ] } + { \ alien.accessors:alien-unsigned-2 [ 2 emit-alien-unsigned-getter iterate-next ] } + { \ alien.accessors:set-alien-unsigned-2 [ 2 emit-alien-integer-setter iterate-next ] } + { \ alien.accessors:alien-signed-2 [ 2 emit-alien-signed-getter iterate-next ] } + { \ alien.accessors:set-alien-signed-2 [ 2 emit-alien-integer-setter iterate-next ] } + { \ alien.accessors:alien-unsigned-4 [ 4 emit-alien-unsigned-getter iterate-next ] } + { \ alien.accessors:set-alien-unsigned-4 [ 4 emit-alien-integer-setter iterate-next ] } + { \ alien.accessors:alien-signed-4 [ 4 emit-alien-signed-getter iterate-next ] } + { \ alien.accessors:set-alien-signed-4 [ 4 emit-alien-integer-setter iterate-next ] } + { \ alien.accessors:alien-cell [ emit-alien-cell-getter iterate-next ] } + { \ alien.accessors:set-alien-cell [ emit-alien-cell-setter iterate-next ] } + { \ alien.accessors:alien-float [ single-float-regs emit-alien-float-getter iterate-next ] } + { \ alien.accessors:set-alien-float [ single-float-regs emit-alien-float-setter iterate-next ] } + { \ alien.accessors:alien-double [ double-float-regs emit-alien-float-getter iterate-next ] } + { \ alien.accessors:set-alien-double [ double-float-regs emit-alien-float-setter iterate-next ] } } case ; diff --git a/basis/compiler/cfg/linearization/linearization.factor b/basis/compiler/cfg/linearization/linearization.factor index d397c9d448..7433df9617 100644 --- a/basis/compiler/cfg/linearization/linearization.factor +++ b/basis/compiler/cfg/linearization/linearization.factor @@ -43,8 +43,8 @@ M: ##branch linearize-insn : binary-conditional ( basic-block insn -- basic-block successor label2 src1 src2 cc ) [ (binary-conditional) ] - [ drop dup successors>> first useless-branch? ] 2bi - [ [ number>> ] 3dip negate-cc ] [ [ swap number>> ] 3dip ] if ; + [ drop dup successors>> second useless-branch? ] 2bi + [ [ swap number>> ] 3dip ] [ [ number>> ] 3dip negate-cc ] if ; M: ##compare-branch linearize-insn binary-conditional _compare-branch emit-branch ; diff --git a/basis/compiler/cfg/rpo/rpo.factor b/basis/compiler/cfg/rpo/rpo.factor index 7f4b09e68f..158903b4bf 100644 --- a/basis/compiler/cfg/rpo/rpo.factor +++ b/basis/compiler/cfg/rpo/rpo.factor @@ -9,7 +9,10 @@ SYMBOL: visited : post-order-traversal ( bb -- ) dup id>> visited get key? [ drop ] [ dup id>> visited get conjoin - [ successors>> [ post-order-traversal ] each ] [ , ] bi + [ + successors>> + [ post-order-traversal ] each + ] [ , ] bi ] if ; : post-order ( bb -- blocks ) diff --git a/basis/compiler/cfg/stack-frame/stack-frame.factor b/basis/compiler/cfg/stack-frame/stack-frame.factor index ec9ffaba49..d545b6d15c 100644 --- a/basis/compiler/cfg/stack-frame/stack-frame.factor +++ b/basis/compiler/cfg/stack-frame/stack-frame.factor @@ -34,6 +34,12 @@ M: insn compute-stack-frame* \ _gc t frame-required? set-word-prop \ _spill t frame-required? set-word-prop +\ ##fixnum-add t frame-required? set-word-prop +\ ##fixnum-sub t frame-required? set-word-prop +\ ##fixnum-mul t frame-required? set-word-prop +\ ##fixnum-add-tail f frame-required? set-word-prop +\ ##fixnum-sub-tail f frame-required? set-word-prop +\ ##fixnum-mul-tail f frame-required? set-word-prop : compute-stack-frame ( insns -- ) frame-required? off diff --git a/basis/compiler/cfg/stacks/stacks.factor b/basis/compiler/cfg/stacks/stacks.factor index f138f673e0..c8fcae87c0 100755 --- a/basis/compiler/cfg/stacks/stacks.factor +++ b/basis/compiler/cfg/stacks/stacks.factor @@ -15,16 +15,28 @@ IN: compiler.cfg.stacks 1 ##inc-d D 0 ##replace ; : ds-load ( n -- vregs ) - [ [ ^^peek ] map ] [ neg ##inc-d ] bi ; + dup 0 = + [ drop f ] + [ [ [ ^^peek ] map ] [ neg ##inc-d ] bi ] if ; : ds-store ( vregs -- ) - [ length ##inc-d ] [ [ ##replace ] each-index ] bi ; + [ + + [ length ##inc-d ] + [ [ ##replace ] each-index ] bi + ] unless-empty ; : rs-load ( n -- vregs ) - [ [ ^^peek ] map ] [ neg ##inc-r ] bi ; + dup 0 = + [ drop f ] + [ [ [ ^^peek ] map ] [ neg ##inc-r ] bi ] if ; : rs-store ( vregs -- ) - [ length ##inc-r ] [ [ ##replace ] each-index ] bi ; + [ + + [ length ##inc-r ] + [ [ ##replace ] each-index ] bi + ] unless-empty ; : 2inputs ( -- vreg1 vreg2 ) D 1 ^^peek D 0 ^^peek -2 ##inc-d ; diff --git a/basis/compiler/cfg/utilities/utilities.factor b/basis/compiler/cfg/utilities/utilities.factor index cef14d06e4..99a138a763 100644 --- a/basis/compiler/cfg/utilities/utilities.factor +++ b/basis/compiler/cfg/utilities/utilities.factor @@ -33,5 +33,7 @@ IN: compiler.cfg.utilities building off basic-block off ; +: stop-iterating ( -- next ) end-basic-block f ; + : emit-primitive ( node -- ) word>> ##call ##branch begin-basic-block ; diff --git a/basis/compiler/cfg/value-numbering/propagate/propagate.factor b/basis/compiler/cfg/value-numbering/propagate/propagate.factor index a3c9725838..d5c9830c0b 100644 --- a/basis/compiler/cfg/value-numbering/propagate/propagate.factor +++ b/basis/compiler/cfg/value-numbering/propagate/propagate.factor @@ -62,4 +62,8 @@ M: ##compare-imm-branch propagate M: ##dispatch propagate [ resolve ] change-src ; +M: ##fixnum-overflow propagate + [ resolve ] change-src1 + [ resolve ] change-src2 ; + M: insn propagate ; diff --git a/basis/compiler/cfg/value-numbering/rewrite/rewrite.factor b/basis/compiler/cfg/value-numbering/rewrite/rewrite.factor index 94c3f0d6f9..990543ed7a 100644 --- a/basis/compiler/cfg/value-numbering/rewrite/rewrite.factor +++ b/basis/compiler/cfg/value-numbering/rewrite/rewrite.factor @@ -1,7 +1,8 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: kernel sequences layouts accessors combinators namespaces -math +math fry +compiler.cfg.hats compiler.cfg.instructions compiler.cfg.value-numbering.graph compiler.cfg.value-numbering.simplify @@ -63,7 +64,7 @@ M: ##compare-imm-branch rewrite-tagged-comparison M: ##compare-imm rewrite-tagged-comparison [ dst>> ] [ (rewrite-tagged-comparison) ] bi - f \ ##compare-imm boa ; + i f \ ##compare-imm boa ; M: ##compare-imm-branch rewrite dup rewrite-boolean-comparison? [ rewrite-boolean-comparison ] when @@ -78,7 +79,7 @@ M: ##compare-imm-branch rewrite [ dst>> ] [ src2>> ] [ src1>> vreg>vn vn>constant ] tri - cc= f \ ##compare-imm boa ; + cc= f i \ ##compare-imm boa ; M: ##compare rewrite dup flip-comparison? [ @@ -95,9 +96,9 @@ M: ##compare rewrite : rewrite-redundant-comparison ( insn -- insn' ) [ cc>> ] [ dst>> ] [ src1>> vreg>expr dup op>> ] tri { - { \ ##compare [ >compare-expr< f \ ##compare boa ] } - { \ ##compare-imm [ >compare-imm-expr< f \ ##compare-imm boa ] } - { \ ##compare-float [ >compare-expr< f \ ##compare-float boa ] } + { \ ##compare [ >compare-expr< i f \ ##compare boa ] } + { \ ##compare-imm [ >compare-imm-expr< i f \ ##compare-imm boa ] } + { \ ##compare-float [ >compare-expr< i f \ ##compare-float boa ] } } case swap cc= eq? [ [ negate-cc ] change-cc ] when ; @@ -113,4 +114,18 @@ M: ##compare-imm rewrite ] when ] when ; +: dispatch-offset ( expr -- n ) + [ in2>> vn>constant tag-bits get neg shift ] [ op>> ] bi + \ ##sub-imm eq? [ neg ] when ; + +: add-dispatch-offset? ( insn -- expr ? ) + src>> vreg>expr dup op>> { ##add-imm ##sub-imm } memq? ; inline + +M: ##dispatch rewrite + dup add-dispatch-offset? [ + [ clone ] dip + [ in1>> vn>vreg >>src ] + [ dispatch-offset '[ _ + ] change-offset ] bi + ] [ drop ] if ; + M: insn rewrite ; diff --git a/basis/compiler/cfg/value-numbering/value-numbering-tests.factor b/basis/compiler/cfg/value-numbering/value-numbering-tests.factor index d3be68c3c9..8adeaa21f4 100644 --- a/basis/compiler/cfg/value-numbering/value-numbering-tests.factor +++ b/basis/compiler/cfg/value-numbering/value-numbering-tests.factor @@ -1,6 +1,17 @@ IN: compiler.cfg.value-numbering.tests USING: compiler.cfg.value-numbering compiler.cfg.instructions -compiler.cfg.registers cpu.architecture tools.test kernel math ; +compiler.cfg.registers cpu.architecture tools.test kernel math +combinators.short-circuit accessors sequences ; + +: trim-temps ( insns -- insns ) + [ + dup { + [ ##compare? ] + [ ##compare-imm? ] + [ ##compare-float? ] + } 1|| [ f >>temp ] when + ] map ; + [ { T{ ##peek f V int-regs 45 D 1 } @@ -34,7 +45,7 @@ compiler.cfg.registers cpu.architecture tools.test kernel math ; [ t ] [ { T{ ##peek f V int-regs 1 D 0 } - T{ ##dispatch f V int-regs 1 V int-regs 2 } + T{ ##dispatch f V int-regs 1 V int-regs 2 0 } } dup value-numbering = ] unit-test @@ -82,7 +93,7 @@ compiler.cfg.registers cpu.architecture tools.test kernel math ; T{ ##compare f V int-regs 4 V int-regs 2 V int-regs 1 cc> } T{ ##compare-imm f V int-regs 6 V int-regs 4 7 cc/= } T{ ##replace f V int-regs 6 D 0 } - } value-numbering + } value-numbering trim-temps ] unit-test [ @@ -100,7 +111,7 @@ compiler.cfg.registers cpu.architecture tools.test kernel math ; T{ ##compare f V int-regs 4 V int-regs 2 V int-regs 1 cc<= } T{ ##compare-imm f V int-regs 6 V int-regs 4 7 cc= } T{ ##replace f V int-regs 6 D 0 } - } value-numbering + } value-numbering trim-temps ] unit-test [ @@ -122,7 +133,7 @@ compiler.cfg.registers cpu.architecture tools.test kernel math ; T{ ##compare-float f V int-regs 12 V double-float-regs 10 V double-float-regs 11 cc< } T{ ##compare-imm f V int-regs 14 V int-regs 12 7 cc= } T{ ##replace f V int-regs 14 D 0 } - } value-numbering + } value-numbering trim-temps ] unit-test [ @@ -138,5 +149,5 @@ compiler.cfg.registers cpu.architecture tools.test kernel math ; T{ ##peek f V int-regs 30 D -2 } T{ ##compare f V int-regs 33 V int-regs 29 V int-regs 30 cc<= } T{ ##compare-imm-branch f V int-regs 33 7 cc/= } - } value-numbering + } value-numbering trim-temps ] unit-test diff --git a/basis/compiler/codegen/codegen.factor b/basis/compiler/codegen/codegen.factor index 35d4d59253..2161c8b091 100644 --- a/basis/compiler/codegen/codegen.factor +++ b/basis/compiler/codegen/codegen.factor @@ -93,7 +93,7 @@ M: ##return generate-insn drop %return ; M: ##dispatch-label generate-insn label>> %dispatch-label ; M: ##dispatch generate-insn - [ src>> register ] [ temp>> register ] bi %dispatch ; + [ src>> register ] [ temp>> register ] [ offset>> ] tri %dispatch ; : >slot< { @@ -156,6 +156,19 @@ M: ##shr-imm generate-insn dst/src1/src2 %shr-imm ; M: ##sar-imm generate-insn dst/src1/src2 %sar-imm ; M: ##not generate-insn dst/src %not ; +: src1/src2 ( insn -- src1 src2 ) + [ src1>> register ] [ src2>> register ] bi ; inline + +: src1/src2/temp1/temp2 ( insn -- src1 src2 temp1 temp2 ) + [ src1/src2 ] [ temp1>> register ] [ temp2>> register ] tri ; inline + +M: ##fixnum-add generate-insn src1/src2 %fixnum-add ; +M: ##fixnum-add-tail generate-insn src1/src2 %fixnum-add-tail ; +M: ##fixnum-sub generate-insn src1/src2 %fixnum-sub ; +M: ##fixnum-sub-tail generate-insn src1/src2 %fixnum-sub-tail ; +M: ##fixnum-mul generate-insn src1/src2/temp1/temp2 %fixnum-mul ; +M: ##fixnum-mul-tail generate-insn src1/src2/temp1/temp2 %fixnum-mul-tail ; + : dst/src/temp ( insn -- dst src temp ) [ dst/src ] [ temp>> register ] bi ; inline @@ -235,7 +248,7 @@ M: float-regs reg-class-variable drop float-regs ; GENERIC: inc-reg-class ( register-class -- ) : ?dummy-stack-params ( reg-class -- ) - dummy-stack-params? [ reg-size stack-params +@ ] [ drop ] if ; + dummy-stack-params? [ reg-size cell align stack-params +@ ] [ drop ] if ; : ?dummy-int-params ( reg-class -- ) dummy-int-params? [ reg-size cell /i 1 max int-regs +@ ] [ drop ] if ; @@ -264,7 +277,7 @@ M: object reg-class-full? : spill-param ( reg-class -- n reg-class ) stack-params get - >r reg-size stack-params +@ r> + [ reg-size cell align stack-params +@ ] dip stack-params ; : fastcall-param ( reg-class -- n reg-class ) @@ -300,10 +313,10 @@ M: long-long-type flatten-value-type ( type -- types ) ] { } make ; : each-parameter ( parameters quot -- ) - >r [ parameter-sizes nip ] keep r> 2each ; inline + [ [ parameter-sizes nip ] keep ] dip 2each ; inline : reverse-each-parameter ( parameters quot -- ) - >r [ parameter-sizes nip ] keep r> 2reverse-each ; inline + [ [ parameter-sizes nip ] keep ] dip 2reverse-each ; inline : reset-freg-counts ( -- ) { int-regs float-regs stack-params } [ 0 swap set ] each ; @@ -316,15 +329,13 @@ M: long-long-type flatten-value-type ( type -- types ) #! Moves values from C stack to registers (if word is #! %load-param-reg) and registers to C stack (if word is #! %save-param-reg). - >r - alien-parameters - flatten-value-types - r> '[ alloc-parameter _ execute ] each-parameter ; - inline + [ alien-parameters flatten-value-types ] + [ '[ alloc-parameter _ execute ] ] + bi* each-parameter ; inline : unbox-parameters ( offset node -- ) parameters>> [ - %prepare-unbox >r over + r> unbox-parameter + %prepare-unbox [ over + ] dip unbox-parameter ] reverse-each-parameter drop ; : prepare-box-struct ( node -- offset ) @@ -491,9 +502,10 @@ M: _label generate-insn M: _branch generate-insn label>> lookup-label %jump-label ; -: >compare< ( insn -- label cc src1 src2 ) +: >compare< ( insn -- dst temp cc src1 src2 ) { [ dst>> register ] + [ temp>> register ] [ cc>> ] [ src1>> register ] [ src2>> ?register ] diff --git a/basis/compiler/codegen/fixup/fixup.factor b/basis/compiler/codegen/fixup/fixup.factor index fe270f4410..0302218652 100755 --- a/basis/compiler/codegen/fixup/fixup.factor +++ b/basis/compiler/codegen/fixup/fixup.factor @@ -46,34 +46,33 @@ M: integer fixup* , ; : indq ( elt seq -- n ) [ eq? ] with find drop ; : adjoin* ( obj table -- n ) - 2dup indq [ 2nip ] [ dup length >r push r> ] if* ; + 2dup indq [ 2nip ] [ dup length [ push ] dip ] if* ; SYMBOL: literal-table : add-literal ( obj -- n ) literal-table get adjoin* ; : add-dlsym-literals ( symbol dll -- ) - >r string>symbol r> 2array literal-table get push-all ; + [ string>symbol ] dip 2array literal-table get push-all ; : rel-dlsym ( name dll class -- ) - >r literal-table get length >r - add-dlsym-literals - r> r> rt-dlsym rel-fixup ; + [ literal-table get length [ add-dlsym-literals ] dip ] dip + rt-dlsym rel-fixup ; : rel-word ( word class -- ) - >r add-literal r> rt-xt rel-fixup ; + [ add-literal ] dip rt-xt rel-fixup ; : rel-primitive ( word class -- ) - >r def>> first r> rt-primitive rel-fixup ; + [ def>> first ] dip rt-primitive rel-fixup ; -: rel-literal ( literal class -- ) - >r add-literal r> rt-literal rel-fixup ; +: rel-immediate ( literal class -- ) + [ add-literal ] dip rt-immediate rel-fixup ; : rel-this ( class -- ) 0 swap rt-label rel-fixup ; -: rel-here ( class -- ) - 0 swap rt-here rel-fixup ; +: rel-here ( offset class -- ) + rt-here rel-fixup ; : init-fixup ( -- ) BV{ } clone relocation-table set diff --git a/basis/compiler/compiler-docs.factor b/basis/compiler/compiler-docs.factor index 6cb860d33f..512d26f4bf 100644 --- a/basis/compiler/compiler-docs.factor +++ b/basis/compiler/compiler-docs.factor @@ -6,7 +6,7 @@ HELP: enable-compiler { $description "Enables the optimizing compiler." } ; HELP: disable-compiler -{ $description "Enables the optimizing compiler." } ; +{ $description "Disable the optimizing compiler." } ; ARTICLE: "compiler-usage" "Calling the optimizing compiler" "Normally, new word definitions are recompiled automatically. This can be changed:" diff --git a/basis/compiler/compiler.factor b/basis/compiler/compiler.factor index b01a835b4a..e5cbd888d9 100644 --- a/basis/compiler/compiler.factor +++ b/basis/compiler/compiler.factor @@ -1,8 +1,8 @@ ! Copyright (C) 2004, 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: accessors kernel namespaces arrays sequences io debugger -words fry continuations vocabs assocs dlists definitions math -threads graphs generic combinators deques search-deques +words fry continuations vocabs assocs dlists definitions +math threads graphs generic combinators deques search-deques prettyprint io stack-checker stack-checker.state stack-checker.inlining compiler.errors compiler.units compiler.tree.builder compiler.tree.optimizer @@ -91,8 +91,8 @@ t compile-dependencies? set-global [ dup crossref? [ - dependencies get >alist - generic-dependencies get >alist + dependencies get + generic-dependencies get compiled-xref ] [ drop ] if ] tri ; diff --git a/basis/compiler/constants/constants.factor b/basis/compiler/constants/constants.factor index cd68602768..48ea958818 100644 --- a/basis/compiler/constants/constants.factor +++ b/basis/compiler/constants/constants.factor @@ -37,14 +37,14 @@ IN: compiler.constants : rc-indirect-arm-pc 8 ; inline ! Relocation types -: rt-primitive 0 ; inline -: rt-dlsym 1 ; inline -: rt-literal 2 ; inline -: rt-dispatch 3 ; inline -: rt-xt 4 ; inline -: rt-here 5 ; inline -: rt-label 6 ; inline -: rt-immediate 7 ; inline +: rt-primitive 0 ; inline +: rt-dlsym 1 ; inline +: rt-dispatch 2 ; inline +: rt-xt 3 ; inline +: rt-here 4 ; inline +: rt-label 5 ; inline +: rt-immediate 6 ; inline +: rt-stack-chain 7 ; inline : rc-absolute? ( n -- ? ) [ rc-absolute-ppc-2/2 = ] diff --git a/basis/compiler/tests/alien.factor b/basis/compiler/tests/alien.factor index d7e82402d5..230a7bf542 100644 --- a/basis/compiler/tests/alien.factor +++ b/basis/compiler/tests/alien.factor @@ -3,7 +3,8 @@ USING: alien alien.c-types alien.syntax compiler kernel namespaces namespaces tools.test sequences stack-checker stack-checker.errors words arrays parser quotations continuations effects namespaces.private io io.streams.string -memory system threads tools.test math accessors combinators ; +memory system threads tools.test math accessors combinators +specialized-arrays.float ; FUNCTION: void ffi_test_0 ; [ ] [ ffi_test_0 ] unit-test @@ -146,13 +147,21 @@ FUNCTION: void ffi_test_20 double x1, double x2, double x3, ! Make sure XT doesn't get clobbered in stack frame -: ffi_test_31 ( a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a ptr -- result y ) - "void" +: ffi_test_31 ( a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a -- result y ) + "int" f "ffi_test_31" { "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" "int" } alien-invoke gc 3 ; -[ 3 ] [ 42 [ ] each ffi_test_31 ] unit-test +[ 861 3 ] [ 42 [ ] each ffi_test_31 ] unit-test + +: ffi_test_31_point_5 ( a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a -- result ) + "float" + f "ffi_test_31_point_5" + { "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" "float" } + alien-invoke ; + +[ 861.0 ] [ 42 [ >float ] each ffi_test_31_point_5 ] unit-test FUNCTION: longlong ffi_test_21 long x long y ; @@ -188,7 +197,11 @@ FUNCTION: int ffi_test_12 int a int b rect c int d int e int f ; FUNCTION: float ffi_test_23 ( float[3] x, float[3] y ) ; -[ 32.0 ] [ { 1.0 2.0 3.0 } >c-float-array { 4.0 5.0 6.0 } >c-float-array ffi_test_23 ] unit-test +[ 32.0 ] [ + { 1.0 2.0 3.0 } >float-array underlying>> + { 4.0 5.0 6.0 } >float-array underlying>> + ffi_test_23 +] unit-test ! Test odd-size structs C-STRUCT: test-struct-1 { { "char" 1 } "x" } ; @@ -353,7 +366,7 @@ FUNCTION: ulonglong ffi_test_38 ( ulonglong x, ulonglong y ) ; [ 1 2 3 ] [ callback-6 callback_test_1 1 2 3 ] unit-test : callback-7 ( -- callback ) - "void" { } "cdecl" [ 1000 sleep ] alien-callback ; + "void" { } "cdecl" [ 1000000 sleep ] alien-callback ; [ 1 2 3 ] [ callback-7 callback_test_1 1 2 3 ] unit-test diff --git a/basis/compiler/tests/templates.factor b/basis/compiler/tests/codegen.factor similarity index 81% rename from basis/compiler/tests/templates.factor rename to basis/compiler/tests/codegen.factor index 0a109a15eb..e743c8484b 100644 --- a/basis/compiler/tests/templates.factor +++ b/basis/compiler/tests/codegen.factor @@ -3,7 +3,7 @@ kernel.private math hashtables.private math.private namespaces sequences sequences.private tools.test namespaces.private slots.private sequences.private byte-arrays alien alien.accessors layouts words definitions compiler.units io -combinators vectors float-arrays ; +combinators vectors grouping make ; IN: compiler.tests ! Originally, this file did black box testing of templating @@ -230,3 +230,49 @@ TUPLE: id obj ; 10000000 [ "hi" 0 (gc-check-bug) drop ] times ; [ ] [ gc-check-bug ] unit-test + +! New optimization +: test-1 ( a -- b ) 8 fixnum-fast { [ "a" ] [ "b" ] } dispatch ; + +[ "a" ] [ 8 test-1 ] unit-test +[ "b" ] [ 9 test-1 ] unit-test + +: test-2 ( a -- b ) 1 fixnum-fast { [ "a" ] [ "b" ] } dispatch ; + +[ "a" ] [ 1 test-2 ] unit-test +[ "b" ] [ 2 test-2 ] unit-test + +! I accidentally fixnum/i-fast on PowerPC +[ { { 1 2 } { 3 4 } } ] [ + { 1 2 3 4 } + [ + [ { array } declare 2 [ , ] each ] compile-call + ] { } make +] unit-test + +[ 2 ] [ + { 1 2 3 4 } + [ { array } declare 2 length ] compile-call +] unit-test + +! Oops with new intrinsics +: fixnum-overflow-control-flow-test ( a b -- c ) + [ 1 fixnum- ] [ 2 fixnum- ] if 3 fixnum+fast ; + +[ 3 ] [ 1 t fixnum-overflow-control-flow-test ] unit-test +[ 2 ] [ 1 f fixnum-overflow-control-flow-test ] unit-test + +! LOL +: blah ( a -- b ) + { float } declare dup 0 = + [ drop 1 ] [ + dup 0 >= + [ 2 "double" "libm" "pow" { "double" "double" } alien-invoke ] + [ -0.5 "double" "libm" "pow" { "double" "double" } alien-invoke ] + if + ] if ; + +[ 4.0 ] [ 2.0 blah ] unit-test + +[ 4 ] [ 2 [ dup fixnum* ] compile-call ] unit-test +[ 7 ] [ 2 [ dup fixnum* 3 fixnum+fast ] compile-call ] unit-test diff --git a/basis/compiler/tests/intrinsics.factor b/basis/compiler/tests/intrinsics.factor index c90a31fc61..df5f484952 100644 --- a/basis/compiler/tests/intrinsics.factor +++ b/basis/compiler/tests/intrinsics.factor @@ -160,6 +160,11 @@ IN: compiler.tests [ -2 ] [ 4 [ -2 fixnum/i ] compile-call ] unit-test [ 3 1 ] [ 10 3 [ fixnum/mod ] compile-call ] unit-test +[ 2 ] [ 4 2 [ fixnum/i-fast ] compile-call ] unit-test +[ 2 ] [ 4 [ 2 fixnum/i-fast ] compile-call ] unit-test +[ -2 ] [ 4 [ -2 fixnum/i-fast ] compile-call ] unit-test +[ 3 1 ] [ 10 3 [ fixnum/mod-fast ] compile-call ] unit-test + [ 4 ] [ 1 3 [ fixnum+ ] compile-call ] unit-test [ 4 ] [ 1 [ 3 fixnum+ ] compile-call ] unit-test [ 4 ] [ [ 1 3 fixnum+ ] compile-call ] unit-test @@ -208,6 +213,7 @@ IN: compiler.tests [ -1 ] [ [ -123 -64 fixnum-shift ] compile-call ] unit-test [ -1 ] [ -123 -64 [ fixnum-shift ] compile-call ] unit-test +[ HEX: 10000000 ] [ HEX: 1000000 HEX: 10 [ fixnum* ] compile-call ] unit-test [ HEX: 10000000 ] [ HEX: -10000000 >fixnum [ 0 swap fixnum- ] compile-call ] unit-test [ HEX: 10000000 ] [ HEX: -fffffff >fixnum [ 1 swap fixnum- ] compile-call ] unit-test diff --git a/basis/compiler/tests/optimizer.factor b/basis/compiler/tests/optimizer.factor index f1b3e32eed..41df6e7ae5 100644 --- a/basis/compiler/tests/optimizer.factor +++ b/basis/compiler/tests/optimizer.factor @@ -286,9 +286,7 @@ HINTS: recursive-inline-hang-2 array ; HINTS: recursive-inline-hang-3 array ; ! Regression -USE: sequences.private - -[ ] [ { (3append) } compile ] unit-test +[ ] [ { 3append-as } compile ] unit-test ! Wow : counter-example ( a b c d -- a' b' c' d' ) diff --git a/basis/compiler/tests/redefine13.factor b/basis/compiler/tests/redefine13.factor new file mode 100644 index 0000000000..d092cd4ee1 --- /dev/null +++ b/basis/compiler/tests/redefine13.factor @@ -0,0 +1,14 @@ +USING: math fry macros eval tools.test ; +IN: compiler.tests.redefine13 + +: breakage-word ( a b -- c ) + ; + +MACRO: breakage-macro ( a -- ) '[ _ breakage-word ] ; + +GENERIC: breakage-caller ( a -- c ) + +M: fixnum breakage-caller 2 breakage-macro ; + +: breakage ( -- obj ) 2 breakage-caller ; + +! [ ] [ "IN: compiler.tests.redefine13 : breakage-word ( a b -- c ) ;" eval ] unit-test diff --git a/basis/compiler/tests/redefine14.factor b/basis/compiler/tests/redefine14.factor new file mode 100644 index 0000000000..807f3ed2c7 --- /dev/null +++ b/basis/compiler/tests/redefine14.factor @@ -0,0 +1,8 @@ +USING: compiler.units definitions tools.test sequences ; +IN: compiler.tests.redefine14 + +! TUPLE: bad ; +! +! M: bad length 1 2 3 ; +! +! [ ] [ [ { bad length } forget ] with-compilation-unit ] unit-test diff --git a/basis/compiler/tests/spilling.factor b/basis/compiler/tests/spilling.factor index 156fdfff02..ee8c2f056a 100644 --- a/basis/compiler/tests/spilling.factor +++ b/basis/compiler/tests/spilling.factor @@ -1,5 +1,5 @@ USING: math.private kernel combinators accessors arrays -generalizations float-arrays tools.test ; +generalizations tools.test ; IN: compiler.tests : float-spill-bug ( a -- b b b b b b b b b b b b b b b b b b b b b b b b b b b b b b b b b b b b b b ) diff --git a/basis/compiler/tree/builder/builder.factor b/basis/compiler/tree/builder/builder.factor index 19d80ec14f..4e79c4cd2d 100644 --- a/basis/compiler/tree/builder/builder.factor +++ b/basis/compiler/tree/builder/builder.factor @@ -1,9 +1,13 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: fry accessors quotations kernel sequences namespaces -assocs words arrays vectors hints combinators stack-checker -stack-checker.state stack-checker.visitor stack-checker.errors -stack-checker.backend compiler.tree ; +assocs words arrays vectors hints combinators compiler.tree +stack-checker +stack-checker.state +stack-checker.errors +stack-checker.visitor +stack-checker.backend +stack-checker.recursive-state ; IN: compiler.tree.builder : with-tree-builder ( quot -- nodes ) @@ -12,12 +16,13 @@ IN: compiler.tree.builder : build-tree ( quot -- nodes ) #! Not safe to call from inference transforms. - [ f infer-quot ] with-tree-builder nip ; + [ f initial-recursive-state infer-quot ] with-tree-builder nip ; : build-tree-with ( in-stack quot -- nodes out-stack ) #! Not safe to call from inference transforms. [ - [ >vector meta-d set ] [ f infer-quot ] bi* + [ >vector meta-d set ] + [ f initial-recursive-state infer-quot ] bi* ] with-tree-builder nip unclip-last in-d>> ; @@ -29,14 +34,10 @@ IN: compiler.tree.builder if ; : (build-tree-from-word) ( word -- ) - dup - [ "inline" word-prop ] - [ "recursive" word-prop ] bi and [ - 1quotation f infer-quot - ] [ - [ specialized-def ] - [ dup 2array 1array ] bi infer-quot - ] if ; + dup initial-recursive-state recursive-state set + dup [ "inline" word-prop ] [ "recursive" word-prop ] bi and + [ 1quotation ] [ specialized-def ] if + infer-quot-here ; : check-cannot-infer ( word -- ) dup "cannot-infer" word-prop [ cannot-infer-effect ] [ drop ] if ; diff --git a/basis/compiler/tree/checker/checker.factor b/basis/compiler/tree/checker/checker.factor index b712a6e354..4f99fa015d 100644 --- a/basis/compiler/tree/checker/checker.factor +++ b/basis/compiler/tree/checker/checker.factor @@ -22,8 +22,8 @@ ERROR: check-use-error value message ; GENERIC: check-node* ( node -- ) M: #shuffle check-node* - [ [ mapping>> values ] [ in-d>> ] bi subset? [ "Bad mapping inputs" throw ] unless ] - [ [ mapping>> keys ] [ out-d>> ] bi set= [ "Bad mapping outputs" throw ] unless ] + [ [ mapping>> values ] [ [ in-d>> ] [ in-r>> ] bi append ] bi subset? [ "Bad mapping inputs" throw ] unless ] + [ [ mapping>> keys ] [ [ out-d>> ] [ out-r>> ] bi append ] bi set= [ "Bad mapping outputs" throw ] unless ] bi ; : check-lengths ( seq -- ) @@ -31,13 +31,6 @@ M: #shuffle check-node* M: #copy check-node* inputs/outputs 2array check-lengths ; -: check->r/r> ( node -- ) - inputs/outputs [ drop ] [ 2array check-lengths ] if-empty ; - -M: #>r check-node* check->r/r> ; - -M: #r> check-node* check->r/r> ; - M: #return-recursive check-node* inputs/outputs 2array check-lengths ; M: #phi check-node* @@ -113,11 +106,8 @@ M: #push check-stack-flow* check-out-d ; M: #call check-stack-flow* [ check-in-d ] [ check-out-d ] bi ; -M: #shuffle check-stack-flow* [ check-in-d ] [ check-out-d ] bi ; - -M: #>r check-stack-flow* [ check-in-d ] [ check-out-r ] bi ; - -M: #r> check-stack-flow* [ check-in-r ] [ check-out-d ] bi ; +M: #shuffle check-stack-flow* + { [ check-in-d ] [ check-in-r ] [ check-out-d ] [ check-out-r ] } cleave ; : assert-datastack-empty ( -- ) datastack get empty? [ "Data stack not empty" throw ] unless ; diff --git a/basis/compiler/tree/cleanup/cleanup-tests.factor b/basis/compiler/tree/cleanup/cleanup-tests.factor index b77a27800f..71c6fb5675 100644 --- a/basis/compiler/tree/cleanup/cleanup-tests.factor +++ b/basis/compiler/tree/cleanup/cleanup-tests.factor @@ -5,7 +5,7 @@ strings sbufs sequences.private slots.private combinators definitions system layouts vectors math.partial-dispatch math.order math.functions accessors hashtables classes assocs io.encodings.utf8 io.encodings.ascii io.encodings fry slots -sorting.private combinators.short-circuit +sorting.private combinators.short-circuit grouping prettyprint compiler.tree compiler.tree.combinators compiler.tree.cleanup @@ -71,7 +71,7 @@ M: object xyz ; 2over fixnum>= [ 3drop ] [ - [ swap >r call 1 fixnum+fast r> ] keep (fx-repeat) + [ swap [ call 1 fixnum+fast ] dip ] keep (fx-repeat) ] if ; inline recursive : fx-repeat ( n quot -- ) @@ -87,10 +87,10 @@ M: object xyz ; 2over dup xyz drop >= [ 3drop ] [ - [ swap >r call 1+ r> ] keep (i-repeat) + [ swap [ call 1+ ] dip ] keep (i-repeat) ] if ; inline recursive -: i-repeat >r { integer } declare r> 0 -rot (i-repeat) ; inline +: i-repeat [ { integer } declare ] dip 0 -rot (i-repeat) ; inline [ t ] [ [ [ dup xyz drop ] i-repeat ] \ xyz inlined? @@ -194,7 +194,7 @@ M: fixnum annotate-entry-test-1 drop ; 2dup >= [ 2drop ] [ - >r dup annotate-entry-test-1 1+ r> (annotate-entry-test-2) + [ dup annotate-entry-test-1 1+ ] dip (annotate-entry-test-2) ] if ; inline recursive : annotate-entry-test-2 0 -rot (annotate-entry-test-2) ; inline @@ -448,7 +448,7 @@ cell-bits 32 = [ ] unit-test [ ] [ - [ [ >r "A" throw r> ] [ "B" throw ] if ] + [ [ [ "A" throw ] dip ] [ "B" throw ] if ] cleaned-up-tree drop ] unit-test @@ -463,7 +463,7 @@ cell-bits 32 = [ : buffalo-wings ( i seq -- ) 2dup < [ 2dup chicken-fingers - >r 1+ r> buffalo-wings + [ 1+ ] dip buffalo-wings ] [ 2drop ] if ; inline recursive @@ -482,7 +482,7 @@ cell-bits 32 = [ : ribs ( i seq -- ) 2dup < [ steak - >r 1+ r> ribs + [ 1+ ] dip ribs ] [ 2drop ] if ; inline recursive @@ -500,3 +500,13 @@ cell-bits 32 = [ [ { array-capacity } declare 1 fixnum+ ] cleaned-up-tree [ { [ #call? ] [ node-input-infos second literal>> 1 = ] } 1&& ] contains? ] unit-test + +[ ] [ + [ { null } declare [ 1 ] [ 2 ] if ] + build-tree normalize propagate cleanup check-nodes +] unit-test + +[ t ] [ + [ { array } declare 2 [ . . ] assoc-each ] + \ nth-unsafe inlined? +] unit-test diff --git a/basis/compiler/tree/cleanup/cleanup.factor b/basis/compiler/tree/cleanup/cleanup.factor index 4ca058b2e3..becac01cd5 100644 --- a/basis/compiler/tree/cleanup/cleanup.factor +++ b/basis/compiler/tree/cleanup/cleanup.factor @@ -102,7 +102,7 @@ M: #declare cleanup* drop f ; #! If only one branch is live we don't need to branch at #! all; just drop the condition value. dup live-children sift dup length { - { 0 [ 2drop f ] } + { 0 [ drop in-d>> #drop ] } { 1 [ first swap in-d>> #drop prefix ] } [ 2drop ] } case ; diff --git a/basis/compiler/tree/dead-code/branches/branches.factor b/basis/compiler/tree/dead-code/branches/branches.factor index 719c80f911..eba82384ab 100644 --- a/basis/compiler/tree/dead-code/branches/branches.factor +++ b/basis/compiler/tree/dead-code/branches/branches.factor @@ -39,7 +39,7 @@ M: #branch remove-dead-code* [ drop filter-live ] [ swap nths ] 2bi [ make-values ] keep [ drop ] [ zip ] 2bi - #shuffle ; + #data-shuffle ; : insert-drops ( nodes values indices -- nodes' ) '[ diff --git a/basis/compiler/tree/dead-code/dead-code-tests.factor b/basis/compiler/tree/dead-code/dead-code-tests.factor index 7b15fdf856..b64e30d8f9 100644 --- a/basis/compiler/tree/dead-code/dead-code-tests.factor +++ b/basis/compiler/tree/dead-code/dead-code-tests.factor @@ -75,9 +75,9 @@ IN: compiler.tree.dead-code.tests remove-dead-code "no-check" get [ dup check-nodes ] unless nodes>quot ; -[ [ drop 1 ] ] [ [ >r 1 r> drop ] optimize-quot ] unit-test +[ [ drop 1 ] ] [ [ [ 1 ] dip drop ] optimize-quot ] unit-test -[ [ read drop 1 2 ] ] [ [ read >r 1 2 r> drop ] optimize-quot ] unit-test +[ [ read drop 1 2 ] ] [ [ read [ 1 2 ] dip drop ] optimize-quot ] unit-test [ [ over >r + r> ] ] [ [ [ + ] [ drop ] 2bi ] optimize-quot ] unit-test diff --git a/basis/compiler/tree/dead-code/liveness/liveness.factor b/basis/compiler/tree/dead-code/liveness/liveness.factor index 08bfde55b2..44b71935c8 100644 --- a/basis/compiler/tree/dead-code/liveness/liveness.factor +++ b/basis/compiler/tree/dead-code/liveness/liveness.factor @@ -1,8 +1,9 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: fry accessors namespaces assocs deques search-deques -kernel sequences sequences.deep words sets stack-checker.branches -compiler.tree compiler.tree.def-use compiler.tree.combinators ; +dlists kernel sequences sequences.deep words sets +stack-checker.branches compiler.tree compiler.tree.def-use +compiler.tree.combinators ; IN: compiler.tree.dead-code.liveness SYMBOL: work-list diff --git a/basis/compiler/tree/dead-code/simple/simple.factor b/basis/compiler/tree/dead-code/simple/simple.factor index addb13ced3..185c776c4e 100644 --- a/basis/compiler/tree/dead-code/simple/simple.factor +++ b/basis/compiler/tree/dead-code/simple/simple.factor @@ -39,12 +39,6 @@ M: #copy compute-live-values* M: #call compute-live-values* nip look-at-inputs ; -M: #>r compute-live-values* - [ out-r>> ] [ in-d>> ] bi look-at-mapping ; - -M: #r> compute-live-values* - [ out-d>> ] [ in-r>> ] bi look-at-mapping ; - M: #shuffle compute-live-values* mapping>> at look-at-value ; @@ -61,7 +55,7 @@ M: #alien-indirect compute-live-values* nip look-at-inputs ; zip filter-mapping values ; : filter-live ( values -- values' ) - [ live-value? ] filter ; + dup empty? [ [ live-value? ] filter ] unless ; :: drop-values ( inputs outputs mapping-keys mapping-values -- #shuffle ) inputs @@ -69,7 +63,7 @@ M: #alien-indirect compute-live-values* nip look-at-inputs ; outputs mapping-keys mapping-values - filter-corresponding zip #shuffle ; inline + filter-corresponding zip #data-shuffle ; inline :: drop-dead-values ( outputs -- #shuffle ) [let* | new-outputs [ outputs make-values ] @@ -95,16 +89,6 @@ M: #alien-indirect compute-live-values* nip look-at-inputs ; M: #introduce remove-dead-code* ( #introduce -- nodes ) maybe-drop-dead-outputs ; -M: #>r remove-dead-code* - [ filter-live ] change-out-r - [ filter-live ] change-in-d - dup in-d>> empty? [ drop f ] when ; - -M: #r> remove-dead-code* - [ filter-live ] change-out-d - [ filter-live ] change-in-r - dup in-r>> empty? [ drop f ] when ; - M: #push remove-dead-code* dup out-d>> first live-value? [ drop f ] unless ; @@ -125,12 +109,14 @@ M: #call remove-dead-code* M: #shuffle remove-dead-code* [ filter-live ] change-in-d [ filter-live ] change-out-d + [ filter-live ] change-in-r + [ filter-live ] change-out-r [ filter-mapping ] change-mapping - dup in-d>> empty? [ drop f ] when ; + dup [ in-d>> empty? ] [ in-r>> empty? ] bi and [ drop f ] when ; M: #copy remove-dead-code* [ in-d>> ] [ out-d>> ] bi - 2dup swap zip #shuffle + 2dup swap zip #data-shuffle remove-dead-code* ; M: #terminate remove-dead-code* diff --git a/basis/compiler/tree/debugger/debugger.factor b/basis/compiler/tree/debugger/debugger.factor index 59a028a4f4..8d764a2833 100644 --- a/basis/compiler/tree/debugger/debugger.factor +++ b/basis/compiler/tree/debugger/debugger.factor @@ -3,7 +3,7 @@ USING: kernel assocs match fry accessors namespaces make effects sequences sequences.private quotations generic macros arrays prettyprint prettyprint.backend prettyprint.sections math words -combinators io sorting hints qualified +combinators combinators.short-circuit io sorting hints qualified compiler.tree compiler.tree.recursive compiler.tree.normalization @@ -57,11 +57,43 @@ TUPLE: shuffle-node { effect effect } ; M: shuffle-node pprint* effect>> effect>string text ; -M: #shuffle node>quot - shuffle-effect dup pretty-shuffle - [ % ] [ shuffle-node boa , ] ?if ; +: (shuffle-effect) ( in out #shuffle -- effect ) + mapping>> '[ _ at ] map ; -M: #push node>quot literal>> , ; +: shuffle-effect ( #shuffle -- effect ) + [ in-d>> ] [ out-d>> ] [ ] tri (shuffle-effect) ; + +: #>r? ( #shuffle -- ? ) + { + [ in-d>> length 1 = ] + [ out-r>> length 1 = ] + [ in-r>> empty? ] + [ out-d>> empty? ] + } 1&& ; + +: #r>? ( #shuffle -- ? ) + { + [ in-d>> empty? ] + [ out-r>> empty? ] + [ in-r>> length 1 = ] + [ out-d>> length 1 = ] + } 1&& ; + +M: #shuffle node>quot + { + { [ dup #>r? ] [ drop \ >r , ] } + { [ dup #r>? ] [ drop \ r> , ] } + { + [ dup [ in-r>> empty? ] [ out-r>> empty? ] bi and ] + [ + shuffle-effect dup pretty-shuffle + [ % ] [ shuffle-node boa , ] ?if + ] + } + [ drop "COMPLEX SHUFFLE" , ] + } cond ; + +M: #push node>quot literal>> literalize , ; M: #call node>quot word>> , ; @@ -82,16 +114,6 @@ M: #if node>quot M: #dispatch node>quot children>> [ nodes>quot ] map , \ dispatch , ; -M: #>r node>quot - [ in-d>> length ] [ out-r>> empty? \ drop \ >r ? ] bi - % ; - -DEFER: rdrop - -M: #r> node>quot - [ in-r>> length ] [ out-d>> empty? \ rdrop \ r> ? ] bi - % ; - M: #alien-invoke node>quot params>> , \ #alien-invoke , ; M: #alien-indirect node>quot params>> , \ #alien-indirect , ; @@ -103,9 +125,13 @@ M: node node>quot drop ; : nodes>quot ( node -- quot ) [ [ node>quot ] each ] [ ] make ; -: optimized. ( quot/word -- ) - dup word? [ specialized-def ] when - build-tree optimize-tree nodes>quot . ; +GENERIC: optimized. ( quot/word -- ) + +M: method-spec optimized. first2 method optimized. ; + +M: word optimized. specialized-def optimized. ; + +M: callable optimized. build-tree optimize-tree nodes>quot . ; SYMBOL: words-called SYMBOL: generics-called diff --git a/basis/compiler/tree/def-use/def-use.factor b/basis/compiler/tree/def-use/def-use.factor index 2379f3918d..705f44eeb6 100644 --- a/basis/compiler/tree/def-use/def-use.factor +++ b/basis/compiler/tree/def-use/def-use.factor @@ -18,12 +18,16 @@ TUPLE: definition value node uses ; swap >>node V{ } clone >>uses ; +ERROR: no-def-error value ; + : def-of ( value -- definition ) - def-use get at* [ "No def" throw ] unless ; + dup def-use get at* [ nip ] [ no-def-error ] if ; + +ERROR: multiple-defs-error ; : def-value ( node value -- ) def-use get 2dup key? [ - "Multiple defs" throw + multiple-defs-error ] [ [ [ ] keep ] dip set-at ] if ; @@ -38,16 +42,16 @@ GENERIC: node-uses-values ( node -- values ) M: #introduce node-uses-values drop f ; M: #push node-uses-values drop f ; -M: #r> node-uses-values in-r>> ; M: #phi node-uses-values phi-in-d>> concat remove-bottom prune ; M: #declare node-uses-values declaration>> keys ; M: #terminate node-uses-values [ in-d>> ] [ in-r>> ] bi append ; +M: #shuffle node-uses-values [ in-d>> ] [ in-r>> ] bi append ; M: #alien-callback node-uses-values drop f ; M: node node-uses-values in-d>> ; GENERIC: node-defs-values ( node -- values ) -M: #>r node-defs-values out-r>> ; +M: #shuffle node-defs-values [ out-d>> ] [ out-r>> ] bi append ; M: #branch node-defs-values drop f ; M: #declare node-defs-values drop f ; M: #return node-defs-values drop f ; diff --git a/basis/compiler/tree/escape-analysis/allocations/allocations.factor b/basis/compiler/tree/escape-analysis/allocations/allocations.factor index 4c197d7fc0..5d34eaad15 100644 --- a/basis/compiler/tree/escape-analysis/allocations/allocations.factor +++ b/basis/compiler/tree/escape-analysis/allocations/allocations.factor @@ -1,7 +1,7 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: accessors assocs namespaces sequences kernel math -combinators sets disjoint-sets fry stack-checker.state ; +combinators sets disjoint-sets fry stack-checker.values ; IN: compiler.tree.escape-analysis.allocations ! A map from values to one of the following: diff --git a/basis/compiler/tree/escape-analysis/escape-analysis-tests.factor b/basis/compiler/tree/escape-analysis/escape-analysis-tests.factor index 4ed194e81d..9a226b954f 100644 --- a/basis/compiler/tree/escape-analysis/escape-analysis-tests.factor +++ b/basis/compiler/tree/escape-analysis/escape-analysis-tests.factor @@ -8,6 +8,7 @@ math.private kernel tools.test accessors slots.private quotations.private prettyprint classes.tuple.private classes classes.tuple namespaces compiler.tree.propagation.info stack-checker.errors +compiler.tree.checker kernel.private ; \ escape-analysis must-infer @@ -34,6 +35,7 @@ M: node count-unboxed-allocations* drop ; propagate cleanup escape-analysis + dup check-nodes 0 swap [ count-unboxed-allocations* ] each-node ; [ 0 ] [ [ [ + ] curry ] count-unboxed-allocations ] unit-test @@ -307,7 +309,7 @@ C: ro-box : bleach-node ( quot: ( node -- ) -- ) [ bleach-node ] curry [ ] compose impeach-node ; inline recursive -[ 2 ] [ [ [ ] bleach-node ] count-unboxed-allocations ] unit-test +[ 3 ] [ [ [ ] bleach-node ] count-unboxed-allocations ] unit-test [ 0 ] [ [ dup -1 over >= [ 0 >= [ "A" throw ] unless ] [ drop ] if ] diff --git a/basis/compiler/tree/finalization/finalization.factor b/basis/compiler/tree/finalization/finalization.factor index 2d2e429994..16a27e020a 100644 --- a/basis/compiler/tree/finalization/finalization.factor +++ b/basis/compiler/tree/finalization/finalization.factor @@ -1,6 +1,7 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: kernel accessors sequences words memoize classes.builtin +fry assocs compiler.tree compiler.tree.combinators compiler.tree.propagation.info @@ -27,9 +28,10 @@ GENERIC: finalize* ( node -- nodes ) M: #copy finalize* drop f ; M: #shuffle finalize* - dup shuffle-effect - [ in>> ] [ out>> ] bi sequence= - [ drop f ] when ; + dup + [ [ in-d>> ] [ out-d>> ] [ mapping>> ] tri '[ _ at ] map sequence= ] + [ [ in-r>> ] [ out-r>> ] [ mapping>> ] tri '[ _ at ] map sequence= ] + bi and [ drop f ] when ; : builtin-predicate? ( #call -- ? ) word>> "predicating" word-prop builtin-class? ; diff --git a/basis/compiler/tree/identities/identities.factor b/basis/compiler/tree/identities/identities.factor index d6ed59cbaa..00632ec6f6 100644 --- a/basis/compiler/tree/identities/identities.factor +++ b/basis/compiler/tree/identities/identities.factor @@ -79,7 +79,7 @@ GENERIC: apply-identities* ( node -- node ) : select-input ( node n -- #shuffle ) [ [ in-d>> ] [ out-d>> ] bi ] dip - pick nth over first associate #shuffle ; + pick nth over first associate #data-shuffle ; M: #call apply-identities* dup word>> "identities" word-prop [ diff --git a/basis/compiler/tree/normalization/normalization-tests.factor b/basis/compiler/tree/normalization/normalization-tests.factor index c4a97fcc92..5ac3c57abe 100644 --- a/basis/compiler/tree/normalization/normalization-tests.factor +++ b/basis/compiler/tree/normalization/normalization-tests.factor @@ -34,8 +34,8 @@ sequences accessors tools.test kernel math ; [ ] [ [ [ 1 ] [ 2 ] if + * ] test-normalization ] unit-test DEFER: bbb -: aaa ( x -- ) dup [ dup >r bbb r> aaa ] [ drop ] if ; inline recursive -: bbb ( x -- ) >r drop 0 r> aaa ; inline recursive +: aaa ( x -- ) dup [ dup [ bbb ] dip aaa ] [ drop ] if ; inline recursive +: bbb ( x -- ) [ drop 0 ] dip aaa ; inline recursive [ ] [ [ bbb ] test-normalization ] unit-test diff --git a/basis/compiler/tree/normalization/renaming/renaming.factor b/basis/compiler/tree/normalization/renaming/renaming.factor index 3050df2611..9d68f4a733 100644 --- a/basis/compiler/tree/normalization/renaming/renaming.factor +++ b/basis/compiler/tree/normalization/renaming/renaming.factor @@ -10,7 +10,7 @@ SYMBOL: rename-map [ rename-map get at ] keep or ; : rename-values ( values -- values' ) - rename-map get '[ [ _ at ] keep or ] map ; + dup empty? [ rename-map get '[ [ _ at ] keep or ] map ] unless ; : add-renamings ( old new -- ) [ rename-values ] dip @@ -22,13 +22,11 @@ M: #introduce rename-node-values* ; M: #shuffle rename-node-values* [ rename-values ] change-in-d + [ rename-values ] change-in-r [ [ rename-value ] assoc-map ] change-mapping ; M: #push rename-node-values* ; -M: #r> rename-node-values* - [ rename-values ] change-in-r ; - M: #terminate rename-node-values* [ rename-values ] change-in-d [ rename-values ] change-in-r ; diff --git a/basis/compiler/tree/propagation/branches/branches.factor b/basis/compiler/tree/propagation/branches/branches.factor index c76217f8ae..424cd8a01c 100644 --- a/basis/compiler/tree/propagation/branches/branches.factor +++ b/basis/compiler/tree/propagation/branches/branches.factor @@ -40,8 +40,8 @@ M: #dispatch live-branches SYMBOL: infer-children-data : copy-value-info ( -- ) - value-infos [ clone ] change - constraints [ clone ] change ; + value-infos [ H{ } clone suffix ] change + constraints [ H{ } clone suffix ] change ; : no-value-info ( -- ) value-infos off diff --git a/basis/compiler/tree/propagation/constraints/constraints.factor b/basis/compiler/tree/propagation/constraints/constraints.factor index cfdf7f5169..2652547aad 100644 --- a/basis/compiler/tree/propagation/constraints/constraints.factor +++ b/basis/compiler/tree/propagation/constraints/constraints.factor @@ -32,7 +32,7 @@ TUPLE: true-constraint value ; M: true-constraint assume* [ \ f class-not swap value>> refine-value-info ] - [ constraints get at [ assume ] when* ] + [ constraints get assoc-stack [ assume ] when* ] bi ; M: true-constraint satisfied? @@ -44,7 +44,7 @@ TUPLE: false-constraint value ; M: false-constraint assume* [ \ f swap value>> refine-value-info ] - [ constraints get at [ assume ] when* ] + [ constraints get assoc-stack [ assume ] when* ] bi ; M: false-constraint satisfied? @@ -83,7 +83,7 @@ TUPLE: implication p q ; C: --> implication : assume-implication ( p q -- ) - [ constraints get [ swap suffix ] change-at ] + [ constraints get [ assoc-stack swap suffix ] 2keep peek set-at ] [ satisfied? [ assume ] [ drop ] if ] 2bi ; M: implication assume* diff --git a/basis/compiler/tree/propagation/info/info-tests.factor b/basis/compiler/tree/propagation/info/info-tests.factor index 24f4ca59dc..2c3314994b 100644 --- a/basis/compiler/tree/propagation/info/info-tests.factor +++ b/basis/compiler/tree/propagation/info/info-tests.factor @@ -70,3 +70,7 @@ TUPLE: test-tuple { x read-only } ; f f 3 3array test-tuple dup object-info value-info-intersect = ] unit-test + +[ t ] [ + null-info 3 value-info<= +] unit-test diff --git a/basis/compiler/tree/propagation/info/info.factor b/basis/compiler/tree/propagation/info/info.factor index d1d8189f7a..771d3800df 100644 --- a/basis/compiler/tree/propagation/info/info.factor +++ b/basis/compiler/tree/propagation/info/info.factor @@ -2,7 +2,7 @@ ! See http://factorcode.org/license.txt for BSD license. USING: assocs classes classes.algebra classes.tuple classes.tuple.private kernel accessors math math.intervals -namespaces sequences words combinators combinators.short-circuit +namespaces sequences words combinators arrays compiler.tree.propagation.copy ; IN: compiler.tree.propagation.info @@ -34,7 +34,7 @@ slots ; : null-info T{ value-info f null empty-interval } ; inline -: object-info T{ value-info f object T{ interval f { -1.0/0.0 t } { 1.0/0.0 t } } } ; inline +: object-info T{ value-info f object full-interval } ; inline : class-interval ( class -- interval ) dup real class<= @@ -43,7 +43,7 @@ slots ; : interval>literal ( class interval -- literal literal? ) #! If interval has zero length and the class is sufficiently #! precise, we can turn it into a literal - dup empty-interval eq? [ + dup special-interval? [ 2drop f f ] [ dup from>> first { @@ -243,7 +243,7 @@ DEFER: (value-info-union) : literals<= ( info1 info2 -- ? ) { { [ dup literal?>> not ] [ 2drop t ] } - { [ over literal?>> not ] [ 2drop f ] } + { [ over literal?>> not ] [ drop class>> null-class? ] } [ [ literal>> ] bi@ eql? ] } cond ; @@ -253,26 +253,29 @@ DEFER: (value-info-union) { [ over not ] [ 2drop f ] } [ { - [ [ class>> ] bi@ class<= ] - [ [ interval>> ] bi@ interval-subset? ] - [ literals<= ] - [ [ length>> ] bi@ value-info<= ] - [ [ slots>> ] bi@ [ value-info<= ] 2all? ] - } 2&& + { [ 2dup [ class>> ] bi@ class<= not ] [ f ] } + { [ 2dup [ interval>> ] bi@ interval-subset? not ] [ f ] } + { [ 2dup literals<= not ] [ f ] } + { [ 2dup [ length>> ] bi@ value-info<= not ] [ f ] } + { [ 2dup [ slots>> ] bi@ [ value-info<= ] 2all? not ] [ f ] } + [ t ] + } cond 2nip ] } cond ; -! Current value --> info mapping +! Assoc stack of current value --> info mapping SYMBOL: value-infos : value-info ( value -- info ) - resolve-copy value-infos get at null-info or ; + resolve-copy value-infos get assoc-stack null-info or ; : set-value-info ( info value -- ) - resolve-copy value-infos get set-at ; + resolve-copy value-infos get peek set-at ; : refine-value-info ( info value -- ) - resolve-copy value-infos get [ value-info-intersect ] change-at ; + resolve-copy value-infos get + [ assoc-stack value-info-intersect ] 2keep + peek set-at ; : value-literal ( value -- obj ? ) value-info >literal< ; diff --git a/basis/compiler/tree/propagation/inlining/inlining.factor b/basis/compiler/tree/propagation/inlining/inlining.factor index 8397a5fdbb..83a4a7aef7 100644 --- a/basis/compiler/tree/propagation/inlining/inlining.factor +++ b/basis/compiler/tree/propagation/inlining/inlining.factor @@ -3,7 +3,7 @@ USING: accessors kernel arrays sequences math math.order math.partial-dispatch generic generic.standard generic.math classes.algebra classes.union sets quotations assocs combinators -words namespaces continuations +words namespaces continuations classes fry compiler.tree compiler.tree.builder compiler.tree.recursive @@ -26,7 +26,7 @@ GENERIC: splicing-nodes ( #call word/quot/f -- nodes ) M: word splicing-nodes [ [ in-d>> ] [ out-d>> ] bi ] dip #call 1array ; -M: quotation splicing-nodes +M: callable splicing-nodes build-sub-tree analyze-recursive normalize ; : propagate-body ( #call -- ) @@ -85,6 +85,8 @@ DEFER: (flat-length) : word-flat-length ( word -- n ) { + ! special-case + { [ dup { dip 2dip 3dip } memq? ] [ drop 1 ] } ! not inline { [ dup inline? not ] [ drop 1 ] } ! recursive and inline @@ -138,18 +140,21 @@ SYMBOL: history : remember-inlining ( word -- ) history [ swap suffix ] change ; -: inline-word ( #call word -- ? ) - dup history get memq? [ - 2drop f +: inline-word-def ( #call word quot -- ? ) + over history get memq? [ + 3drop f ] [ [ - dup remember-inlining - dupd def>> splicing-nodes >>body + swap remember-inlining + dupd splicing-nodes >>body propagate-body ] with-scope t ] if ; +: inline-word ( #call word -- ? ) + dup def>> inline-word-def ; + : inline-method-body ( #call word -- ? ) 2dup should-inline? [ inline-word ] [ 2drop f ] if ; @@ -163,6 +168,10 @@ SYMBOL: history [ dup 1array ] [ "custom-inlining" word-prop ] bi* with-datastack first object swap eliminate-dispatch ; +: inline-instance-check ( #call word -- ? ) + over in-d>> second value-info literal>> dup class? + [ "predicate" word-prop '[ drop @ ] inline-word-def ] [ 3drop f ] if ; + : do-inlining ( #call word -- ? ) #! If the generic was defined in an outer compilation unit, #! then it doesn't have a definition yet; the definition @@ -175,6 +184,7 @@ SYMBOL: history { { [ dup deferred? ] [ 2drop f ] } { [ dup custom-inlining? ] [ inline-custom ] } + { [ dup \ instance? eq? ] [ inline-instance-check ] } { [ dup always-inline-word? ] [ inline-word ] } { [ dup standard-generic? ] [ inline-standard-method ] } { [ dup math-generic? ] [ inline-math-method ] } diff --git a/basis/compiler/tree/propagation/known-words/known-words.factor b/basis/compiler/tree/propagation/known-words/known-words.factor index 3b698e0001..163b17094a 100644 --- a/basis/compiler/tree/propagation/known-words/known-words.factor +++ b/basis/compiler/tree/propagation/known-words/known-words.factor @@ -37,31 +37,6 @@ most-negative-fixnum most-positive-fixnum [a,b] \ bitnot { integer } "input-classes" set-word-prop -{ - fcosh - flog - fsinh - fexp - fasin - facosh - fasinh - ftanh - fatanh - facos - fpow - fatan - fatan2 - fcos - ftan - fsin - fsqrt -} [ - dup stack-effect - [ in>> length real "input-classes" set-word-prop ] - [ out>> length float "default-output-classes" set-word-prop ] - 2bi -] each - : ?change-interval ( info quot -- quot' ) over interval>> [ [ clone ] dip change-interval ] [ 2drop ] if ; inline @@ -138,6 +113,12 @@ most-negative-fixnum most-positive-fixnum [a,b] \ mod [ [ interval-mod ] [ real-valued ] binary-op ] each-derived-op \ rem [ [ interval-rem ] [ may-overflow real-valued ] binary-op ] each-derived-op +{ /mod fixnum/mod } [ + \ /i \ mod + [ "outputs" word-prop ] bi@ + '[ _ _ 2bi ] "outputs" set-word-prop +] each + \ shift [ [ interval-shift-safe ] [ may-overflow integer-valued ] binary-op ] each-derived-op \ shift [ [ interval-shift-safe ] [ integer-valued ] binary-op ] each-fast-derived-op @@ -216,8 +197,15 @@ generic-comparison-ops [ { { >fixnum fixnum } + { bignum>fixnum fixnum } + { >bignum bignum } + { fixnum>bignum bignum } + { float>bignum bignum } + { >float float } + { fixnum>float float } + { bignum>float float } } [ '[ _ diff --git a/basis/compiler/tree/propagation/propagation-tests.factor b/basis/compiler/tree/propagation/propagation-tests.factor index 101320f92c..2c4769abe0 100644 --- a/basis/compiler/tree/propagation/propagation-tests.factor +++ b/basis/compiler/tree/propagation/propagation-tests.factor @@ -8,7 +8,7 @@ math.functions math.private strings layouts compiler.tree.propagation.info compiler.tree.def-use compiler.tree.debugger compiler.tree.checker slots.private words hashtables classes assocs locals -float-arrays system ; +specialized-arrays.double system sorting math.libm ; IN: compiler.tree.propagation.tests \ propagate must-infer @@ -167,7 +167,8 @@ IN: compiler.tree.propagation.tests [ V{ fixnum } ] [ [ - [ uchar-nth ] 2keep [ uchar-nth ] 2keep uchar-nth + { fixnum byte-array } declare + [ nth-unsafe ] 2keep [ nth-unsafe ] 2keep nth-unsafe >r >r 298 * r> 100 * - r> 208 * - 128 + -8 shift 255 min 0 max ] final-classes @@ -434,7 +435,7 @@ TUPLE: mixed-mutable-immutable { x integer } { y sequence read-only } ; ] unit-test : recursive-test-4 ( i n -- ) - 2dup < [ >r 1+ r> recursive-test-4 ] [ 2drop ] if ; inline recursive + 2dup < [ [ 1+ ] dip recursive-test-4 ] [ 2drop ] if ; inline recursive [ ] [ [ recursive-test-4 ] final-info drop ] unit-test @@ -588,10 +589,16 @@ MIXIN: empty-mixin [ { fixnum integer } declare bitand ] final-classes ] unit-test -[ V{ float-array } ] [ [| | F{ } ] final-classes ] unit-test +[ V{ double-array } ] [ [| | double-array{ } ] final-classes ] unit-test [ V{ t } ] [ [ netbsd unix? ] final-literals ] unit-test +[ V{ array } ] [ [ [ <=> ] sort [ <=> ] sort ] final-classes ] unit-test + +[ V{ float } ] [ [ fsqrt ] final-classes ] unit-test + +[ V{ t } ] [ [ { fixnum } declare 10 mod >float -20 > ] final-literals ] unit-test + ! [ V{ string } ] [ ! [ dup string? t xor [ "A" throw ] [ ] if ] final-classes ! ] unit-test diff --git a/basis/compiler/tree/propagation/propagation.factor b/basis/compiler/tree/propagation/propagation.factor index d82ebed433..b9822d2c6b 100644 --- a/basis/compiler/tree/propagation/propagation.factor +++ b/basis/compiler/tree/propagation/propagation.factor @@ -1,6 +1,6 @@ ! Copyright (C) 2004, 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. -USING: accessors kernel sequences namespaces hashtables +USING: accessors kernel sequences namespaces hashtables arrays compiler.tree compiler.tree.propagation.copy compiler.tree.propagation.info @@ -17,7 +17,7 @@ IN: compiler.tree.propagation : propagate ( node -- node ) H{ } clone copies set - H{ } clone constraints set - H{ } clone value-infos set + H{ } clone 1array value-infos set + H{ } clone 1array constraints set dup count-nodes dup (propagate) ; diff --git a/basis/compiler/tree/propagation/recursive/recursive.factor b/basis/compiler/tree/propagation/recursive/recursive.factor index 53dce813a3..7f10f87016 100644 --- a/basis/compiler/tree/propagation/recursive/recursive.factor +++ b/basis/compiler/tree/propagation/recursive/recursive.factor @@ -17,9 +17,12 @@ IN: compiler.tree.propagation.recursive [ value-info<= ] 2all? [ drop ] [ label>> f >>fixed-point drop ] if ; +: latest-input-infos ( node -- infos ) + in-d>> [ value-info ] map ; + : recursive-stacks ( #enter-recursive -- stacks initial ) [ label>> calls>> [ node-input-infos ] map flip ] - [ in-d>> [ value-info ] map ] bi ; + [ latest-input-infos ] bi ; : generalize-counter-interval ( interval initial-interval -- interval' ) { @@ -46,14 +49,13 @@ IN: compiler.tree.propagation.recursive ] if ; : propagate-recursive-phi ( #enter-recursive -- ) - [ ] [ recursive-stacks unify-recursive-stacks ] [ ] tri - [ node-output-infos check-fixed-point ] - [ out-d>> set-value-infos drop ] - 3bi ; + [ recursive-stacks unify-recursive-stacks ] keep + out-d>> set-value-infos ; M: #recursive propagate-around ( #recursive -- ) + constraints [ H{ } clone suffix ] change [ - constraints [ clone ] change + constraints [ but-last H{ } clone suffix ] change child>> [ first compute-copy-equiv ] @@ -62,6 +64,9 @@ M: #recursive propagate-around ( #recursive -- ) tri ] until-fixed-point ; +: recursive-phi-infos ( node -- infos ) + label>> enter-recursive>> node-output-infos ; + : generalize-return-interval ( info -- info' ) dup [ literal?>> ] [ class>> null-class? ] bi or [ clone [-inf,inf] >>interval ] unless ; @@ -70,12 +75,25 @@ M: #recursive propagate-around ( #recursive -- ) [ generalize-return-interval ] map ; : return-infos ( node -- infos ) - label>> [ return>> node-input-infos ] [ loop?>> ] bi - [ generalize-return ] unless ; + label>> return>> node-input-infos generalize-return ; + +: save-return-infos ( node infos -- ) + swap out-d>> set-value-infos ; + +: unless-loop ( node quot -- ) + [ dup label>> loop?>> [ drop ] ] dip if ; inline M: #call-recursive propagate-before ( #call-recursive -- ) - [ ] [ return-infos ] [ node-output-infos ] tri - [ check-fixed-point ] [ drop swap out-d>> set-value-infos ] 3bi ; + [ + [ ] [ latest-input-infos ] [ recursive-phi-infos ] tri + check-fixed-point + ] + [ + [ + [ ] [ return-infos ] [ node-output-infos ] tri + [ check-fixed-point ] [ drop save-return-infos ] 3bi + ] unless-loop + ] bi ; M: #call-recursive annotate-node dup [ in-d>> ] [ out-d>> ] bi append (annotate-node) ; @@ -83,5 +101,11 @@ M: #call-recursive annotate-node M: #enter-recursive annotate-node dup out-d>> (annotate-node) ; +M: #return-recursive propagate-before ( #return-recursive -- ) + [ + [ ] [ latest-input-infos ] [ node-input-infos ] tri + check-fixed-point + ] unless-loop ; + M: #return-recursive annotate-node dup in-d>> (annotate-node) ; diff --git a/basis/compiler/tree/propagation/simple/simple.factor b/basis/compiler/tree/propagation/simple/simple.factor index d586ff398f..9937c6b9c4 100644 --- a/basis/compiler/tree/propagation/simple/simple.factor +++ b/basis/compiler/tree/propagation/simple/simple.factor @@ -2,7 +2,7 @@ ! See http://factorcode.org/license.txt for BSD license. USING: fry accessors kernel sequences sequences.private assocs words namespaces classes.algebra combinators classes classes.tuple -classes.tuple.private continuations arrays +classes.tuple.private continuations arrays alien.c-types math math.private slots generic definitions stack-checker.state compiler.tree @@ -137,11 +137,12 @@ M: #call propagate-after dup word>> "input-classes" word-prop dup [ propagate-input-classes ] [ 2drop ] if ; -M: #alien-invoke propagate-before - out-d>> [ object-info swap set-value-info ] each ; +: propagate-alien-invoke ( node -- ) + [ out-d>> ] [ params>> return>> ] bi + [ drop ] [ c-type-class swap first set-value-info ] if-void ; -M: #alien-indirect propagate-before - out-d>> [ object-info swap set-value-info ] each ; +M: #alien-invoke propagate-before propagate-alien-invoke ; -M: #return annotate-node - dup in-d>> (annotate-node) ; +M: #alien-indirect propagate-before propagate-alien-invoke ; + +M: #return annotate-node dup in-d>> (annotate-node) ; diff --git a/basis/compiler/tree/recursive/recursive.factor b/basis/compiler/tree/recursive/recursive.factor index d257cd6600..2e40693e69 100644 --- a/basis/compiler/tree/recursive/recursive.factor +++ b/basis/compiler/tree/recursive/recursive.factor @@ -1,7 +1,7 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: kernel assocs arrays namespaces accessors sequences deques -search-deques compiler.tree compiler.tree.combinators ; +search-deques dlists compiler.tree compiler.tree.combinators ; IN: compiler.tree.recursive ! Collect label info diff --git a/basis/compiler/tree/tree.factor b/basis/compiler/tree/tree.factor index 05f33902ec..9f9a43df64 100644 --- a/basis/compiler/tree/tree.factor +++ b/basis/compiler/tree/tree.factor @@ -1,7 +1,7 @@ ! Copyright (C) 2004, 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: fry arrays generic assocs kernel math namespaces parser -sequences words vectors math.intervals effects classes +sequences words vectors math.intervals classes accessors combinators stack-checker.state stack-checker.visitor stack-checker.inlining ; IN: compiler.tree @@ -42,30 +42,21 @@ TUPLE: #push < node literal out-d ; TUPLE: #renaming < node ; -TUPLE: #shuffle < #renaming mapping in-d out-d ; +TUPLE: #shuffle < #renaming mapping in-d out-d in-r out-r ; -: #shuffle ( inputs outputs mapping -- node ) +: #shuffle ( in-d out-d in-r out-r mapping -- node ) \ #shuffle new swap >>mapping + swap >>out-r + swap >>in-r swap >>out-d swap >>in-d ; +: #data-shuffle ( in-d out-d mapping -- node ) + [ f f ] dip #shuffle ; inline + : #drop ( inputs -- node ) - { } { } #shuffle ; - -TUPLE: #>r < #renaming in-d out-r ; - -: #>r ( inputs outputs -- node ) - \ #>r new - swap >>out-r - swap >>in-d ; - -TUPLE: #r> < #renaming in-r out-d ; - -: #r> ( inputs outputs -- node ) - \ #r> new - swap >>out-d - swap >>in-r ; + { } { } #data-shuffle ; TUPLE: #terminate < node in-d in-r ; @@ -171,16 +162,9 @@ TUPLE: #alien-callback < #alien-node ; GENERIC: inputs/outputs ( #renaming -- inputs outputs ) M: #shuffle inputs/outputs mapping>> unzip swap ; -M: #>r inputs/outputs [ in-d>> ] [ out-r>> ] bi ; -M: #r> inputs/outputs [ in-r>> ] [ out-d>> ] bi ; M: #copy inputs/outputs [ in-d>> ] [ out-d>> ] bi ; M: #return-recursive inputs/outputs [ in-d>> ] [ out-d>> ] bi ; -: shuffle-effect ( #shuffle -- effect ) - [ in-d>> ] [ out-d>> ] [ mapping>> ] tri - '[ _ at ] map - ; - : recursive-phi-in ( #enter-recursive -- seq ) [ label>> calls>> [ in-d>> ] map ] [ in-d>> ] bi suffix ; @@ -193,8 +177,8 @@ M: vector #call, #call node, ; M: vector #push, #push node, ; M: vector #shuffle, #shuffle node, ; M: vector #drop, #drop node, ; -M: vector #>r, #>r node, ; -M: vector #r>, #r> node, ; +M: vector #>r, [ [ f f ] dip ] [ swap zip ] 2bi #shuffle, ; +M: vector #r>, [ swap [ f swap ] dip f ] [ swap zip ] 2bi #shuffle, ; M: vector #return, #return node, ; M: vector #enter-recursive, #enter-recursive node, ; M: vector #return-recursive, #return-recursive node, ; diff --git a/basis/compiler/tree/tuple-unboxing/tuple-unboxing.factor b/basis/compiler/tree/tuple-unboxing/tuple-unboxing.factor index 8e07c08194..52903fce8d 100644 --- a/basis/compiler/tree/tuple-unboxing/tuple-unboxing.factor +++ b/basis/compiler/tree/tuple-unboxing/tuple-unboxing.factor @@ -42,7 +42,7 @@ M: #push unbox-tuples* ( #push -- nodes ) [ dup unboxed-allocation [ (flatten-values) ] [ ] ?if ] map ; : flatten-values ( values -- values' ) - (flatten-values) flatten ; + dup empty? [ (flatten-values) flatten ] unless ; : prepare-slot-access ( #call -- tuple-values outputs slot-values ) [ in-d>> flatten-values ] @@ -54,7 +54,7 @@ M: #push unbox-tuples* ( #push -- nodes ) ] tri ; : slot-access-shuffle ( tuple-values outputs slot-values -- #shuffle ) - [ drop ] [ zip ] 2bi #shuffle ; + [ drop ] [ zip ] 2bi #data-shuffle ; : unbox-slot-access ( #call -- nodes ) dup out-d>> first unboxed-slot-access? [ @@ -77,17 +77,11 @@ M: #copy unbox-tuples* [ flatten-values ] change-in-d [ flatten-values ] change-out-d ; -M: #>r unbox-tuples* - [ flatten-values ] change-in-d - [ flatten-values ] change-out-r ; - -M: #r> unbox-tuples* - [ flatten-values ] change-in-r - [ flatten-values ] change-out-d ; - M: #shuffle unbox-tuples* [ flatten-values ] change-in-d [ flatten-values ] change-out-d + [ flatten-values ] change-in-r + [ flatten-values ] change-out-r [ unzip [ flatten-values ] bi@ zip ] change-mapping ; M: #terminate unbox-tuples* diff --git a/basis/concurrency/combinators/combinators-docs.factor b/basis/concurrency/combinators/combinators-docs.factor index a23301c1e2..c61967fc8a 100644 --- a/basis/concurrency/combinators/combinators-docs.factor +++ b/basis/concurrency/combinators/combinators-docs.factor @@ -2,36 +2,42 @@ USING: help.markup help.syntax sequences ; IN: concurrency.combinators HELP: parallel-map -{ $values { "seq" sequence } { "quot" "a quotation with stack effect " { $snippet "( elt -- newelt )" } } { "newseq" sequence } } +{ $values { "seq" sequence } { "quot" { $quotation "( elt -- newelt )" } } { "newseq" sequence } } { $description "Spawns a new thread for applying " { $snippet "quot" } " to every element of " { $snippet "seq" } ", collecting the results at the end." } { $errors "Throws an error if one of the iterations throws an error." } ; HELP: 2parallel-map -{ $values { "seq1" sequence } { "seq2" sequence } { "quot" "a quotation with stack effect " { $snippet "( elt -- newelt )" } } { "newseq" sequence } } +{ $values { "seq1" sequence } { "seq2" sequence } { "quot" { $quotation "( elt -- newelt )" } } { "newseq" sequence } } { $description "Spawns a new thread for applying " { $snippet "quot" } " to pairwise elements of " { $snippet "seq1" } " and " { $snippet "seq2" } ", collecting the results at the end." } { $errors "Throws an error if one of the iterations throws an error." } ; HELP: parallel-each -{ $values { "seq" sequence } { "quot" "a quotation with stack effect " { $snippet "( elt -- )" } } } +{ $values { "seq" sequence } { "quot" { $quotation "( elt -- )" } } } { $description "Spawns a new thread for applying " { $snippet "quot" } " to every element of " { $snippet "seq" } ", blocking until all quotations complete." } { $errors "Throws an error if one of the iterations throws an error." } ; HELP: 2parallel-each -{ $values { "seq1" sequence } { "seq2" sequence } { "quot" "a quotation with stack effect " { $snippet "( elt -- )" } } } +{ $values { "seq1" sequence } { "seq2" sequence } { "quot" { $quotation "( elt -- )" } } } { $description "Spawns a new thread for applying " { $snippet "quot" } " to pairwise elements of " { $snippet "seq1" } " and " { $snippet "seq2" } ", blocking until all quotations complete." } { $errors "Throws an error if one of the iterations throws an error." } ; HELP: parallel-filter -{ $values { "seq" sequence } { "quot" "a quotation with stack effect " { $snippet "( elt -- ? )" } } { "newseq" sequence } } +{ $values { "seq" sequence } { "quot" { $quotation "( elt -- ? )" } } { "newseq" sequence } } { $description "Spawns a new thread for applying " { $snippet "quot" } " to every element of " { $snippet "seq" } ", collecting the elements for which the quotation yielded a true value." } { $errors "Throws an error if one of the iterations throws an error." } ; ARTICLE: "concurrency.combinators" "Concurrent combinators" -"The " { $vocab-link "concurrency.combinators" } " vocabulary provides concurrent variants of " { $link each } ", " { $link map } " and " { $link filter } ":" +"The " { $vocab-link "concurrency.combinators" } " vocabulary provides concurrent variants of various combinators." +$nl +"Concurrent sequence combinators:" { $subsection parallel-each } { $subsection 2parallel-each } { $subsection parallel-map } { $subsection 2parallel-map } -{ $subsection parallel-filter } ; +{ $subsection parallel-filter } +"Concurrent cleave combinators:" +{ $subsection parallel-cleave } +{ $subsection parallel-spread } +{ $subsection parallel-napply } ; ABOUT: "concurrency.combinators" diff --git a/basis/concurrency/combinators/combinators-tests.factor b/basis/concurrency/combinators/combinators-tests.factor index 562111242d..1c2dea2d79 100644 --- a/basis/concurrency/combinators/combinators-tests.factor +++ b/basis/concurrency/combinators/combinators-tests.factor @@ -1,6 +1,7 @@ IN: concurrency.combinators.tests USING: concurrency.combinators tools.test random kernel math -concurrency.mailboxes threads sequences accessors arrays ; +concurrency.mailboxes threads sequences accessors arrays +math.parser ; [ [ drop ] parallel-each ] must-infer { 2 0 } [ [ 2drop ] 2parallel-each ] must-infer-as @@ -10,7 +11,7 @@ concurrency.mailboxes threads sequences accessors arrays ; [ { 1 4 9 } ] [ { 1 2 3 } [ sq ] parallel-map ] unit-test -[ { 1 4 9 } ] [ { 1 2 3 } [ 1000 random sleep sq ] parallel-map ] unit-test +[ { 1 4 9 } ] [ { 1 2 3 } [ 1000000 random sleep sq ] parallel-map ] unit-test [ { 1 2 3 } [ dup 2 mod 0 = [ "Even" throw ] when ] parallel-map ] [ error>> "Even" = ] must-fail-with @@ -45,3 +46,10 @@ concurrency.mailboxes threads sequences accessors arrays ; ] unit-test [ { f } [ "OOPS" throw ] parallel-each ] must-fail + +[ "1a" "4b" "3c" ] [ + 2 + { [ 1- ] [ sq ] [ 1+ ] } parallel-cleave + [ number>string ] 3 parallel-napply + { [ "a" append ] [ "b" append ] [ "c" append ] } parallel-spread +] unit-test diff --git a/basis/concurrency/combinators/combinators.factor b/basis/concurrency/combinators/combinators.factor index ab3ca7ed4a..4608faf79b 100644 --- a/basis/concurrency/combinators/combinators.factor +++ b/basis/concurrency/combinators/combinators.factor @@ -1,34 +1,58 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: concurrency.futures concurrency.count-downs sequences -kernel ; +kernel macros fry combinators generalizations ; IN: concurrency.combinators r r> keep await ; inline + [ ] dip keep await ; inline + PRIVATE> : parallel-each ( seq quot -- ) over length [ - [ >r curry r> spawn-stage ] 2curry each + '[ _ curry _ spawn-stage ] each ] (parallel-each) ; inline : 2parallel-each ( seq1 seq2 quot -- ) 2over min-length [ - [ >r 2curry r> spawn-stage ] 2curry 2each + '[ _ 2curry _ spawn-stage ] 2each ] (parallel-each) ; inline : parallel-filter ( seq quot -- newseq ) - over >r pusher >r each r> r> like ; inline + over [ pusher [ each ] dip ] dip like ; inline : parallel-map ( seq quot -- newseq ) - [ curry future ] curry map future-values ; - inline + [future] map future-values ; inline : 2parallel-map ( seq1 seq2 quot -- newseq ) - [ 2curry future ] curry 2map future-values ; + '[ _ 2curry future ] 2map future-values ; + + ; inline + +: (parallel-cleave) ( quots -- quot-array spread-array ) + [ [future] ] map dup length (parallel-spread) ; inline + +PRIVATE> + +MACRO: parallel-cleave ( quots -- ) + (parallel-cleave) '[ _ cleave _ spread ] ; + +MACRO: parallel-spread ( quots -- ) + (parallel-cleave) '[ _ spread _ spread ] ; + +MACRO: parallel-napply ( quot n -- ) + [ [future] ] dip dup (parallel-spread) '[ _ _ napply _ spread ] ; diff --git a/basis/concurrency/conditions/conditions.factor b/basis/concurrency/conditions/conditions.factor index 43374d3127..11e624110c 100644 --- a/basis/concurrency/conditions/conditions.factor +++ b/basis/concurrency/conditions/conditions.factor @@ -1,6 +1,6 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. -USING: deques threads kernel arrays sequences alarms ; +USING: deques threads kernel arrays sequences alarms fry ; IN: concurrency.conditions : notify-1 ( deque -- ) @@ -12,15 +12,18 @@ IN: concurrency.conditions : queue-timeout ( queue timeout -- alarm ) #! Add an alarm which removes the current thread from the #! queue, and resumes it, passing it a value of t. - >r [ self swap push-front* ] keep [ - [ delete-node ] [ drop node-value ] 2bi - t swap resume-with - ] 2curry r> later ; + [ + [ self swap push-front* ] keep '[ + _ _ + [ delete-node ] [ drop node-value ] 2bi + t swap resume-with + ] + ] dip later ; : wait ( queue timeout status -- ) over [ - >r queue-timeout [ drop ] r> suspend + [ queue-timeout [ drop ] ] dip suspend [ "Timeout" throw ] [ cancel-alarm ] if ] [ - >r drop [ push-front ] curry r> suspend drop + [ drop '[ _ push-front ] ] dip suspend drop ] if ; diff --git a/basis/concurrency/count-downs/count-downs.factor b/basis/concurrency/count-downs/count-downs.factor index c4bc92c688..d79cfbf1c9 100644 --- a/basis/concurrency/count-downs/count-downs.factor +++ b/basis/concurrency/count-downs/count-downs.factor @@ -1,7 +1,7 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: dlists kernel math concurrency.promises -concurrency.mailboxes debugger accessors ; +concurrency.mailboxes debugger accessors fry ; IN: concurrency.count-downs ! http://java.sun.com/j2se/1.5.0/docs/api/java/util/concurrent/CountDownLatch.html @@ -26,12 +26,12 @@ ERROR: count-down-already-done ; [ 1- >>n count-down-check ] if ; : await-timeout ( count-down timeout -- ) - >r promise>> r> ?promise-timeout ?linked t assert= ; + [ promise>> ] dip ?promise-timeout ?linked t assert= ; : await ( count-down -- ) f await-timeout ; : spawn-stage ( quot count-down -- ) - [ [ count-down ] curry compose ] keep + [ '[ @ _ count-down ] ] keep "Count down stage" swap promise>> mailbox>> spawn-linked-to drop ; diff --git a/basis/concurrency/distributed/distributed-tests.factor b/basis/concurrency/distributed/distributed-tests.factor index 528e1956b8..1087823aa0 100644 --- a/basis/concurrency/distributed/distributed-tests.factor +++ b/basis/concurrency/distributed/distributed-tests.factor @@ -15,7 +15,7 @@ concurrency.messaging continuations accessors prettyprint ; [ ] [ [ - receive first2 >r 3 + r> send + receive first2 [ 3 + ] dip send "thread-a" unregister-process ] "Thread A" spawn "thread-a" swap register-process diff --git a/basis/concurrency/exchangers/exchangers.factor b/basis/concurrency/exchangers/exchangers.factor index 6b44886eda..97b3c14fe4 100644 --- a/basis/concurrency/exchangers/exchangers.factor +++ b/basis/concurrency/exchangers/exchangers.factor @@ -1,6 +1,6 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. -USING: kernel threads boxes accessors ; +USING: kernel threads boxes accessors fry ; IN: concurrency.exchangers ! Motivated by @@ -14,8 +14,8 @@ TUPLE: exchanger thread object ; : exchange ( obj exchanger -- newobj ) dup thread>> occupied>> [ dup object>> box> - >r thread>> box> resume-with r> + [ thread>> box> resume-with ] dip ] [ [ object>> >box ] keep - [ thread>> >box ] curry "exchange" suspend + '[ _ thread>> >box ] "exchange" suspend ] if ; diff --git a/basis/concurrency/flags/flags-tests.factor b/basis/concurrency/flags/flags-tests.factor index 9d3f6de98c..a666293316 100644 --- a/basis/concurrency/flags/flags-tests.factor +++ b/basis/concurrency/flags/flags-tests.factor @@ -1,8 +1,8 @@ IN: concurrency.flags.tests USING: tools.test concurrency.flags concurrency.combinators -kernel threads locals accessors ; +kernel threads locals accessors calendar ; -:: flag-test-1 ( -- ) +:: flag-test-1 ( -- val ) [let | f [ ] | [ f raise-flag ] "Flag test" spawn drop f lower-flag @@ -13,14 +13,14 @@ kernel threads locals accessors ; :: flag-test-2 ( -- ) [let | f [ ] | - [ 1000 sleep f raise-flag ] "Flag test" spawn drop + [ 1 seconds sleep f raise-flag ] "Flag test" spawn drop f lower-flag f value>> ] ; [ f ] [ flag-test-2 ] unit-test -:: flag-test-3 ( -- ) +:: flag-test-3 ( -- val ) [let | f [ ] | f raise-flag f value>> @@ -28,7 +28,7 @@ kernel threads locals accessors ; [ t ] [ flag-test-3 ] unit-test -:: flag-test-4 ( -- ) +:: flag-test-4 ( -- val ) [let | f [ ] | [ f raise-flag ] "Flag test" spawn drop f wait-for-flag @@ -37,9 +37,9 @@ kernel threads locals accessors ; [ t ] [ flag-test-4 ] unit-test -:: flag-test-5 ( -- ) +:: flag-test-5 ( -- val ) [let | f [ ] | - [ 1000 sleep f raise-flag ] "Flag test" spawn drop + [ 1 seconds sleep f raise-flag ] "Flag test" spawn drop f wait-for-flag f value>> ] ; @@ -48,6 +48,6 @@ kernel threads locals accessors ; [ ] [ { 1 2 } - [ [ 1000 sleep raise-flag ] curry "Flag test" spawn drop ] + [ [ 1 seconds sleep raise-flag ] curry "Flag test" spawn drop ] [ [ wait-for-flag drop ] curry parallel-each ] bi ] unit-test diff --git a/basis/concurrency/flags/flags.factor b/basis/concurrency/flags/flags.factor index ec260961d0..c65171a3f0 100644 --- a/basis/concurrency/flags/flags.factor +++ b/basis/concurrency/flags/flags.factor @@ -11,7 +11,7 @@ TUPLE: flag value threads ; dup value>> [ drop ] [ t >>value threads>> notify-all ] if ; : wait-for-flag-timeout ( flag timeout -- ) - over value>> [ 2drop ] [ >r threads>> r> "flag" wait ] if ; + over value>> [ 2drop ] [ [ threads>> ] dip "flag" wait ] if ; : wait-for-flag ( flag -- ) f wait-for-flag-timeout ; diff --git a/basis/concurrency/futures/futures-docs.factor b/basis/concurrency/futures/futures-docs.factor index 99b4bb6e81..3d2ac552de 100644 --- a/basis/concurrency/futures/futures-docs.factor +++ b/basis/concurrency/futures/futures-docs.factor @@ -1,18 +1,18 @@ ! Copyright (C) 2005, 2008 Chris Double, Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: concurrency.promises concurrency.messaging kernel arrays -continuations help.markup help.syntax quotations ; +continuations help.markup help.syntax quotations calendar ; IN: concurrency.futures HELP: future -{ $values { "quot" "a quotation with stack effect " { $snippet "( -- value )" } } { "future" future } } +{ $values { "quot" { $quotation "( -- value )" } } { "future" future } } { $description "Creates a deferred computation." $nl "The quotation begins with an empty data stack, an empty catch stack, and a name stack containing the global namespace only. This means that the only way to pass data to the quotation is to partially apply the data, for example using " { $link curry } " or " { $link compose } "." } ; HELP: ?future-timeout -{ $values { "future" future } { "timeout" "a timeout in milliseconds or " { $link f } } { "value" object } } -{ $description "Waits for a deferred computation to complete, blocking indefinitely if " { $snippet "timeout" } " is " { $link f } ", otherwise waiting up to " { $snippet "timeout" } " milliseconds." } +{ $values { "future" future } { "timeout" { $maybe duration } } { "value" object } } +{ $description "Waits for a deferred computation to complete, blocking indefinitely if " { $snippet "timeout" } " is " { $link f } ", otherwise waiting up to the " { $snippet "timeout" } " before throwing an error." } { $errors "Throws an error if the timeout expires before the computation completes. Also throws an error if the future quotation threw an error." } ; HELP: ?future diff --git a/basis/concurrency/futures/futures.factor b/basis/concurrency/futures/futures.factor index 132342aff1..a1f4f57af6 100644 --- a/basis/concurrency/futures/futures.factor +++ b/basis/concurrency/futures/futures.factor @@ -1,12 +1,12 @@ ! Copyright (C) 2005, 2008 Chris Double, Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: concurrency.promises concurrency.mailboxes kernel arrays -continuations accessors ; +continuations accessors fry ; IN: concurrency.futures : future ( quot -- future ) [ - [ [ >r call r> fulfill ] 2curry "Future" ] keep + [ '[ @ _ fulfill ] "Future" ] keep mailbox>> spawn-linked-to drop ] keep ; inline diff --git a/basis/concurrency/locks/locks-docs.factor b/basis/concurrency/locks/locks-docs.factor index a3cf2fc782..b74dcec384 100644 --- a/basis/concurrency/locks/locks-docs.factor +++ b/basis/concurrency/locks/locks-docs.factor @@ -14,7 +14,7 @@ HELP: { $description "Creates a reentrant lock." } ; HELP: with-lock-timeout -{ $values { "lock" lock } { "timeout" "a " { $link duration } " or " { $link f } } { "quot" quotation } } +{ $values { "lock" lock } { "timeout" { $maybe duration } } { "quot" quotation } } { $description "Calls the quotation, ensuring that only one thread executes with the lock held at a time. If another thread is holding the lock, blocks until the thread releases the lock." } { $errors "Throws an error if the lock could not be acquired before the timeout expires. A timeout value of " { $link f } " means the thread is willing to wait indefinitely." } ; @@ -36,7 +36,7 @@ HELP: rw-lock { $class-description "The class of reader/writer locks." } ; HELP: with-read-lock-timeout -{ $values { "lock" lock } { "timeout" "a " { $link duration } " or " { $link f } } { "quot" quotation } } +{ $values { "lock" lock } { "timeout" { $maybe duration } } { "quot" quotation } } { $description "Calls the quotation, ensuring that no other thread is holding a write lock at the same time. If another thread is holding a write lock, blocks until the thread releases the lock." } { $errors "Throws an error if the lock could not be acquired before the timeout expires. A timeout value of " { $link f } " means the thread is willing to wait indefinitely." } ; @@ -45,7 +45,7 @@ HELP: with-read-lock { $description "Calls the quotation, ensuring that no other thread is holding a write lock at the same time. If another thread is holding a write lock, blocks until the thread releases the lock." } ; HELP: with-write-lock-timeout -{ $values { "lock" lock } { "timeout" "a " { $link duration } " or " { $link f } } { "quot" quotation } } +{ $values { "lock" lock } { "timeout" { $maybe duration } } { "quot" quotation } } { $description "Calls the quotation, ensuring that no other thread is holding a read or write lock at the same time. If another thread is holding a read or write lock, blocks until the thread releases the lock." } { $errors "Throws an error if the lock could not be acquired before the timeout expires. A timeout value of " { $link f } " means the thread is willing to wait indefinitely." } ; diff --git a/basis/concurrency/locks/locks-tests.factor b/basis/concurrency/locks/locks-tests.factor index 67f9bbb15a..8f82aa88ba 100644 --- a/basis/concurrency/locks/locks-tests.factor +++ b/basis/concurrency/locks/locks-tests.factor @@ -3,7 +3,7 @@ USING: tools.test concurrency.locks concurrency.count-downs concurrency.messaging concurrency.mailboxes locals kernel threads sequences calendar accessors ; -:: lock-test-0 ( -- ) +:: lock-test-0 ( -- v ) [let | v [ V{ } clone ] c [ 2 ] | @@ -27,7 +27,7 @@ threads sequences calendar accessors ; v ] ; -:: lock-test-1 ( -- ) +:: lock-test-1 ( -- v ) [let | v [ V{ } clone ] l [ ] c [ 2 ] | @@ -79,7 +79,7 @@ threads sequences calendar accessors ; [ ] [ dup [ [ ] with-read-lock ] with-write-lock ] unit-test -:: rw-lock-test-1 ( -- ) +:: rw-lock-test-1 ( -- v ) [let | l [ ] c [ 1 ] c' [ 1 ] @@ -100,7 +100,7 @@ threads sequences calendar accessors ; c await l [ 4 v push - 1000 sleep + 1 seconds sleep 5 v push ] with-write-lock c'' count-down @@ -129,7 +129,7 @@ threads sequences calendar accessors ; [ V{ 1 2 3 4 5 6 } ] [ rw-lock-test-1 ] unit-test -:: rw-lock-test-2 ( -- ) +:: rw-lock-test-2 ( -- v ) [let | l [ ] c [ 1 ] c' [ 2 ] @@ -139,7 +139,7 @@ threads sequences calendar accessors ; l [ 1 v push c count-down - 1000 sleep + 1 seconds sleep 2 v push ] with-write-lock c' count-down @@ -160,7 +160,7 @@ threads sequences calendar accessors ; [ V{ 1 2 3 } ] [ rw-lock-test-2 ] unit-test ! Test lock timeouts -:: lock-timeout-test ( -- ) +:: lock-timeout-test ( -- v ) [let | l [ ] | [ l [ 1 seconds sleep ] with-lock @@ -177,19 +177,6 @@ threads sequences calendar accessors ; thread>> name>> "Lock timeout-er" = ] must-fail-with -:: read/write-test ( -- ) - [let | l [ ] | - [ - l [ 1 seconds sleep ] with-lock - ] "Lock holder" spawn drop - - [ - l 1/10 seconds [ ] with-lock-timeout - ] "Lock timeout-er" spawn-linked drop - - receive - ] ; - [ dup [ 1 seconds [ ] with-write-lock-timeout diff --git a/basis/concurrency/locks/locks.factor b/basis/concurrency/locks/locks.factor index 8c1392dbfb..0094f3323d 100644 --- a/basis/concurrency/locks/locks.factor +++ b/basis/concurrency/locks/locks.factor @@ -1,7 +1,8 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: deques dlists kernel threads continuations math -concurrency.conditions combinators.short-circuit accessors ; +concurrency.conditions combinators.short-circuit accessors +locals ; IN: concurrency.locks ! Simple critical sections @@ -17,16 +18,16 @@ TUPLE: lock threads owner reentrant? ; : acquire-lock ( lock timeout -- ) over owner>> - [ 2dup >r threads>> r> "lock" wait ] when drop + [ 2dup [ threads>> ] dip "lock" wait ] when drop self >>owner drop ; : release-lock ( lock -- ) f >>owner threads>> notify-1 ; -: do-lock ( lock timeout quot acquire release -- ) - >r >r pick rot r> call ! use up timeout acquire - swap r> curry [ ] cleanup ; inline +:: do-lock ( lock timeout quot acquire release -- ) + lock timeout acquire call + quot lock release curry [ ] cleanup ; inline : (with-lock) ( lock timeout quot -- ) [ acquire-lock ] [ release-lock ] do-lock ; inline @@ -60,7 +61,7 @@ TUPLE: rw-lock readers writers reader# writer ; : acquire-read-lock ( lock timeout -- ) over writer>> - [ 2dup >r readers>> r> "read lock" wait ] when drop + [ 2dup [ readers>> ] dip "read lock" wait ] when drop add-reader ; : notify-writer ( lock -- ) @@ -75,7 +76,7 @@ TUPLE: rw-lock readers writers reader# writer ; : acquire-write-lock ( lock timeout -- ) over writer>> pick reader#>> 0 > or - [ 2dup >r writers>> r> "write lock" wait ] when drop + [ 2dup [ writers>> ] dip "write lock" wait ] when drop self >>writer drop ; : release-write-lock ( lock -- ) diff --git a/basis/concurrency/mailboxes/mailboxes-docs.factor b/basis/concurrency/mailboxes/mailboxes-docs.factor index a9b86e3bcd..234fb27d60 100644 --- a/basis/concurrency/mailboxes/mailboxes-docs.factor +++ b/basis/concurrency/mailboxes/mailboxes-docs.factor @@ -1,4 +1,4 @@ -USING: help.markup help.syntax kernel arrays ; +USING: help.markup help.syntax kernel arrays calendar ; IN: concurrency.mailboxes HELP: @@ -18,46 +18,41 @@ HELP: mailbox-put { $description "Put the object into the mailbox. Any threads that have a blocking get on the mailbox are resumed. Only one of those threads will successfully get the object, the rest will immediately block waiting for the next item in the mailbox." } ; HELP: block-unless-pred -{ $values { "pred" "a quotation with stack effect " { $snippet "( X -- bool )" } } - { "mailbox" mailbox } - { "timeout" "a timeout in milliseconds, or " { $link f } } +{ $values { "pred" { $quotation "( obj -- ? )" } } + { "mailbox" mailbox } + { "timeout" "a " { $link duration } " or " { $link f } } } { $description "Block the thread if there are no items in the mailbox that return true when the predicate is called with the item on the stack." } ; HELP: block-if-empty { $values { "mailbox" mailbox } - { "timeout" "a timeout in milliseconds, or " { $link f } } + { "timeout" "a " { $link duration } " or " { $link f } } } { $description "Block the thread if the mailbox is empty." } ; HELP: mailbox-get -{ $values { "mailbox" mailbox } - { "obj" object } -} +{ $values { "mailbox" mailbox } { "obj" object } } { $description "Get the first item put into the mailbox. If it is empty the thread blocks until an item is put into it. The thread then resumes, leaving the item on the stack." } ; HELP: mailbox-get-all -{ $values { "mailbox" mailbox } - { "array" array } -} +{ $values { "mailbox" mailbox } { "array" array } } { $description "Blocks the thread if the mailbox is empty, otherwise removes all objects in the mailbox and returns an array containing the objects." } ; HELP: while-mailbox-empty { $values { "mailbox" mailbox } - { "quot" "a quotation with stack effect " { $snippet "( -- )" } } + { "quot" { $quotation "( -- )" } } } { $description "Repeatedly call the quotation while there are no items in the mailbox." } ; HELP: mailbox-get? { $values { "mailbox" mailbox } - { "pred" "a quotation with stack effect " { $snippet "( X -- bool )" } } + { "pred" { $quotation "( obj -- ? )" } } { "obj" object } } -{ $description "Get the first item in the mailbox which satisfies the predicate. 'pred' will be called repeatedly for each item in the mailbox. When 'pred' returns true that item will be returned. If nothing in the mailbox satisfies the predicate then the thread will block until something does." } ; - +{ $description "Get the first item in the mailbox which satisfies the predicate. When the predicate returns true that item will be returned. If nothing in the mailbox satisfies the predicate then the thread will block until something does." } ; ARTICLE: "concurrency.mailboxes" "Mailboxes" -"A " { $emphasis "mailbox" } " is a first-in-first-out queue where the operation of removing an element blocks if the queue is empty, instead of throwing an error. Mailboxes are implemented in the " { $vocab-link "concurrency.mailboxes" } " vocabulary." +"A " { $emphasis "mailbox" } " is a first-in-first-out queue where the operation of removing an element blocks if the queue is empty. Mailboxes are implemented in the " { $vocab-link "concurrency.mailboxes" } " vocabulary." { $subsection mailbox } { $subsection } "Removing the first element:" diff --git a/basis/concurrency/mailboxes/mailboxes.factor b/basis/concurrency/mailboxes/mailboxes.factor index 39b21e0943..63707041a2 100644 --- a/basis/concurrency/mailboxes/mailboxes.factor +++ b/basis/concurrency/mailboxes/mailboxes.factor @@ -4,7 +4,7 @@ IN: concurrency.mailboxes USING: dlists deques threads sequences continuations destructors namespaces math quotations words kernel arrays assocs init system concurrency.conditions accessors -debugger debugger.threads locals ; +debugger debugger.threads locals fry ; TUPLE: mailbox threads data disposed ; @@ -21,7 +21,7 @@ M: mailbox dispose* threads>> notify-all ; [ threads>> notify-all ] bi yield ; : wait-for-mailbox ( mailbox timeout -- ) - >r threads>> r> "mailbox" wait ; + [ threads>> ] dip "mailbox" wait ; :: block-unless-pred ( mailbox timeout pred: ( message -- ? ) -- ) mailbox check-disposed @@ -57,11 +57,11 @@ M: mailbox dispose* threads>> notify-all ; f mailbox-get-all-timeout ; : while-mailbox-empty ( mailbox quot -- ) - [ [ mailbox-empty? ] curry ] dip [ ] while ; inline + [ '[ _ mailbox-empty? ] ] dip [ ] while ; inline : mailbox-get-timeout? ( mailbox timeout pred -- obj ) [ block-unless-pred ] - [ nip >r data>> r> delete-node-if ] + [ [ drop data>> ] dip delete-node-if ] 3bi ; inline : mailbox-get? ( mailbox pred -- obj ) @@ -90,7 +90,7 @@ M: linked-thread error-in-thread [ ] [ supervisor>> ] bi mailbox-put ; : ( quot name mailbox -- thread' ) - >r linked-thread new-thread r> >>supervisor ; + [ linked-thread new-thread ] dip >>supervisor ; : spawn-linked-to ( quot name mailbox -- thread ) [ (spawn) ] keep ; diff --git a/basis/concurrency/messaging/messaging-docs.factor b/basis/concurrency/messaging/messaging-docs.factor index 6c9e530d9b..25538cd594 100644 --- a/basis/concurrency/messaging/messaging-docs.factor +++ b/basis/concurrency/messaging/messaging-docs.factor @@ -55,7 +55,7 @@ ARTICLE: { "concurrency" "synchronous-sends" } "Synchronous sends" { $example "USING: concurrency.messaging kernel threads ;" ": pong-server ( -- )" - " receive >r \"pong\" r> reply-synchronous ;" + " receive [ \"pong\" ] dip reply-synchronous ;" "[ pong-server t ] \"pong-server\" spawn-server" "\"ping\" swap send-synchronous ." "\"pong\"" diff --git a/basis/concurrency/messaging/messaging.factor b/basis/concurrency/messaging/messaging.factor index 9aeb24ed72..7a00f62e9e 100644 --- a/basis/concurrency/messaging/messaging.factor +++ b/basis/concurrency/messaging/messaging.factor @@ -1,10 +1,7 @@ ! Copyright (C) 2005, 2008 Chris Double, Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. -! -! Concurrency library for Factor, based on Erlang/Termite style -! concurrency. USING: kernel threads concurrency.mailboxes continuations -namespaces assocs accessors summary ; +namespaces assocs accessors summary fry ; IN: concurrency.messaging GENERIC: send ( message thread -- ) @@ -32,7 +29,7 @@ M: thread send ( message thread -- ) my-mailbox -rot mailbox-get-timeout? ?linked ; inline : rethrow-linked ( error process supervisor -- ) - >r r> send ; + [ ] dip send ; : spawn-linked ( quot name -- thread ) my-mailbox spawn-linked-to ; @@ -48,9 +45,7 @@ TUPLE: reply data tag ; tag>> \ reply boa ; : synchronous-reply? ( response synchronous -- ? ) - over reply? - [ >r tag>> r> tag>> = ] - [ 2drop f ] if ; + over reply? [ [ tag>> ] bi@ = ] [ 2drop f ] if ; ERROR: cannot-send-synchronous-to-self message thread ; @@ -61,8 +56,8 @@ M: cannot-send-synchronous-to-self summary dup self eq? [ cannot-send-synchronous-to-self ] [ - >r dup r> send - [ synchronous-reply? ] curry receive-if + [ dup ] dip send + '[ _ synchronous-reply? ] receive-if data>> ] if ; diff --git a/basis/concurrency/promises/promises-docs.factor b/basis/concurrency/promises/promises-docs.factor index 6a4a2bf8d6..8e160842a9 100644 --- a/basis/concurrency/promises/promises-docs.factor +++ b/basis/concurrency/promises/promises-docs.factor @@ -12,8 +12,8 @@ HELP: promise-fulfilled? { $description "Tests if " { $link fulfill } " has previously been called on the promise, in which case " { $link ?promise } " will return immediately without blocking." } ; HELP: ?promise-timeout -{ $values { "promise" promise } { "timeout" "a " { $link duration } " or " { $link f } } { "result" object } } -{ $description "Waits for another thread to fulfill a promise, returning immediately if the promise has already been fulfilled. A timeout of " { $link f } " indicates that the thread may block indefinitely, otherwise it will wait up to " { $snippet "timeout" } " milliseconds." } +{ $values { "promise" promise } { "timeout" { $maybe duration } } { "result" object } } +{ $description "Waits for another thread to fulfill a promise, returning immediately if the promise has already been fulfilled. A timeout of " { $link f } " indicates that the thread may block indefinitely, otherwise it will wait up to the " { $snippet "timeout" } " before throwing an error." } { $errors "Throws an error if the timeout expires before the promise has been fulfilled." } ; HELP: ?promise diff --git a/basis/concurrency/promises/promises.factor b/basis/concurrency/promises/promises.factor index 382697e04f..2ff338c4e3 100644 --- a/basis/concurrency/promises/promises.factor +++ b/basis/concurrency/promises/promises.factor @@ -20,7 +20,7 @@ ERROR: promise-already-fulfilled promise ; ] if ; : ?promise-timeout ( promise timeout -- result ) - >r mailbox>> r> block-if-empty mailbox-peek ; + [ mailbox>> ] dip block-if-empty mailbox-peek ; : ?promise ( promise -- result ) f ?promise-timeout ; diff --git a/basis/concurrency/semaphores/semaphores-docs.factor b/basis/concurrency/semaphores/semaphores-docs.factor index 379fd6a3a0..c86623f86f 100644 --- a/basis/concurrency/semaphores/semaphores-docs.factor +++ b/basis/concurrency/semaphores/semaphores-docs.factor @@ -9,7 +9,7 @@ HELP: { $description "Creates a counting semaphore with the specified initial count." } ; HELP: acquire-timeout -{ $values { "semaphore" semaphore } { "timeout" "a " { $link duration } " or " { $link f } } } +{ $values { "semaphore" semaphore } { "timeout" { $maybe duration } } } { $description "If the semaphore has a non-zero count, decrements it and returns immediately. Otherwise, if the timeout is " { $link f } ", waits indefinitely for the semaphore to be released. If the timeout is not " { $link f } ", waits a certain period of time, and if the semaphore still has not been released, throws an error." } { $errors "Throws an error if the timeout expires before the semaphore is released." } ; @@ -22,7 +22,7 @@ HELP: release { $description "Increments a semaphore's count. If the count was previously zero, any threads waiting on the semaphore are woken up." } ; HELP: with-semaphore-timeout -{ $values { "semaphore" semaphore } { "timeout" "a " { $link duration } " or " { $link f } } { "quot" quotation } } +{ $values { "semaphore" semaphore } { "timeout" { $maybe duration } } { "quot" quotation } } { $description "Calls the quotation with the semaphore held." } ; HELP: with-semaphore diff --git a/basis/concurrency/semaphores/semaphores.factor b/basis/concurrency/semaphores/semaphores.factor index 1b55c7afa5..59518f4c8d 100644 --- a/basis/concurrency/semaphores/semaphores.factor +++ b/basis/concurrency/semaphores/semaphores.factor @@ -1,7 +1,7 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: dlists kernel threads math concurrency.conditions -continuations accessors summary ; +continuations accessors summary locals fry ; IN: concurrency.semaphores TUPLE: semaphore count threads ; @@ -30,9 +30,9 @@ M: negative-count-semaphore summary [ 1+ ] change-count threads>> notify-1 ; -: with-semaphore-timeout ( semaphore timeout quot -- ) - pick rot acquire-timeout swap - [ release ] curry [ ] cleanup ; inline +:: with-semaphore-timeout ( semaphore timeout quot -- ) + semaphore timeout acquire-timeout + quot [ semaphore release ] [ ] cleanup ; inline : with-semaphore ( semaphore quot -- ) - over acquire swap [ release ] curry [ ] cleanup ; inline + swap dup acquire '[ _ release ] [ ] cleanup ; inline diff --git a/basis/core-foundation/core-foundation.factor b/basis/core-foundation/core-foundation.factor index 00bf73e9dd..8e5051e75d 100644 --- a/basis/core-foundation/core-foundation.factor +++ b/basis/core-foundation/core-foundation.factor @@ -90,25 +90,25 @@ FUNCTION: CFTypeID CFGetTypeID ( CFTypeRef cf ) ; : ( seq -- alien ) [ f swap length f CFArrayCreateMutable ] keep [ length ] keep - [ >r dupd r> CFArraySetValueAtIndex ] 2each ; + [ [ dupd ] dip CFArraySetValueAtIndex ] 2each ; : ( string -- alien ) f swap dup length CFStringCreateWithCharacters ; : CF>string ( alien -- string ) dup CFStringGetLength 1+ "ushort" [ - >r 0 over CFStringGetLength r> CFStringGetCharacters + [ 0 over CFStringGetLength ] dip CFStringGetCharacters ] keep utf16n alien>string ; : CF>string-array ( alien -- seq ) CF>array [ CF>string ] map ; : ( seq -- alien ) - [ ] map dup swap [ CFRelease ] each ; + [ ] map [ ] [ [ CFRelease ] each ] bi ; : ( string dir? -- url ) - >r f over kCFURLPOSIXPathStyle - r> CFURLCreateWithFileSystemPath swap CFRelease ; + [ f over kCFURLPOSIXPathStyle ] dip + CFURLCreateWithFileSystemPath swap CFRelease ; : ( string -- url ) diff --git a/basis/core-foundation/fsevents/fsevents.factor b/basis/core-foundation/fsevents/fsevents.factor index 6bec4b23c0..d4d5e88512 100644 --- a/basis/core-foundation/fsevents/fsevents.factor +++ b/basis/core-foundation/fsevents/fsevents.factor @@ -4,7 +4,9 @@ USING: alien alien.c-types alien.strings alien.syntax kernel math sequences namespaces make assocs init accessors continuations combinators core-foundation core-foundation.run-loop core-foundation.run-loop.thread -io.encodings.utf8 destructors ; +io.encodings.utf8 destructors locals arrays +specialized-arrays.direct.alien specialized-arrays.direct.int +specialized-arrays.direct.longlong ; IN: core-foundation.fsevents : kFSEventStreamCreateFlagUseCFTypes 2 ; inline @@ -105,15 +107,14 @@ FUNCTION: CFStringRef FSEventStreamCopyDescription ( FSEventStreamRef streamRef "FSEventStreamContext" [ set-FSEventStreamContext-info ] keep ; -: ( callback info paths latency flags -- event-stream ) - >r >r >r >r >r +:: ( callback info paths latency flags -- event-stream ) f ! allocator - r> ! callback - r> make-FSEventStreamContext - r> ! paths + callback + info make-FSEventStreamContext + paths FSEventStreamEventIdSinceNow ! sinceWhen - r> ! latency - r> ! flags + latency + flags FSEventStreamCreate ; : kCFRunLoopCommonModes ( -- string ) @@ -161,13 +162,12 @@ SYMBOL: event-stream-callbacks : remove-event-source-callback ( id -- ) event-stream-callbacks get delete-at ; -: >event-triple ( n eventPaths eventFlags eventIds -- triple ) - [ - >r >r >r dup dup - r> void*-nth utf8 alien>string , - r> int-nth , - r> longlong-nth , - ] { } make ; +:: (master-event-source-callback) ( eventStream info numEvents eventPaths eventFlags eventIds -- ) + eventPaths numEvents [ utf8 alien>string ] { } map-as + eventFlags numEvents + eventIds numEvents + 3array flip + info event-stream-callbacks get at [ drop ] or call ; : master-event-source-callback ( -- alien ) "void" @@ -179,19 +179,15 @@ SYMBOL: event-stream-callbacks "FSEventStreamEventFlags*" "FSEventStreamEventId*" } - "cdecl" [ - [ >event-triple ] 3curry map - swap event-stream-callbacks get at - dup [ call drop ] [ 3drop ] if - ] alien-callback ; + "cdecl" [ (master-event-source-callback) ] alien-callback ; TUPLE: event-stream info handle disposed ; : ( quot paths latency flags -- event-stream ) - >r >r >r - add-event-source-callback dup - >r master-event-source-callback r> - r> r> r> + [ + add-event-source-callback dup + [ master-event-source-callback ] dip + ] 3dip dup enable-event-stream f event-stream boa ; diff --git a/basis/core-foundation/run-loop/run-loop.factor b/basis/core-foundation/run-loop/run-loop.factor index e30cc2eb60..9a5666b5d3 100644 --- a/basis/core-foundation/run-loop/run-loop.factor +++ b/basis/core-foundation/run-loop/run-loop.factor @@ -1,7 +1,7 @@ ! Copyright (C) 2008 Slava Pestov ! See http://factorcode.org/license.txt for BSD license. USING: alien.syntax kernel threads init namespaces alien -core-foundation ; +core-foundation calendar ; IN: core-foundation.run-loop : kCFRunLoopRunFinished 1 ; inline @@ -30,7 +30,7 @@ FUNCTION: SInt32 CFRunLoopRunInMode ( : run-loop-thread ( -- ) CFRunLoopDefaultMode 0 f CFRunLoopRunInMode - kCFRunLoopRunHandledSource = [ 1000 sleep ] unless + kCFRunLoopRunHandledSource = [ 1 seconds sleep ] unless run-loop-thread ; : start-run-loop-thread ( -- ) diff --git a/basis/cpu/architecture/architecture.factor b/basis/cpu/architecture/architecture.factor index b0b5b048d9..12b6809df9 100644 --- a/basis/cpu/architecture/architecture.factor +++ b/basis/cpu/architecture/architecture.factor @@ -50,7 +50,7 @@ HOOK: %call cpu ( word -- ) HOOK: %jump-label cpu ( label -- ) HOOK: %return cpu ( -- ) -HOOK: %dispatch cpu ( src temp -- ) +HOOK: %dispatch cpu ( src temp offset -- ) HOOK: %dispatch-label cpu ( word -- ) HOOK: %slot cpu ( dst obj slot tag temp -- ) @@ -77,6 +77,13 @@ HOOK: %shr-imm cpu ( dst src1 src2 -- ) HOOK: %sar-imm cpu ( dst src1 src2 -- ) HOOK: %not cpu ( dst src -- ) +HOOK: %fixnum-add cpu ( src1 src2 -- ) +HOOK: %fixnum-add-tail cpu ( src1 src2 -- ) +HOOK: %fixnum-sub cpu ( src1 src2 -- ) +HOOK: %fixnum-sub-tail cpu ( src1 src2 -- ) +HOOK: %fixnum-mul cpu ( src1 src2 temp1 temp2 -- ) +HOOK: %fixnum-mul-tail cpu ( src1 src2 temp1 temp2 -- ) + HOOK: %integer>bignum cpu ( dst src temp -- ) HOOK: %bignum>integer cpu ( dst src temp -- ) @@ -119,9 +126,9 @@ HOOK: %gc cpu ( -- ) HOOK: %prologue cpu ( n -- ) HOOK: %epilogue cpu ( n -- ) -HOOK: %compare cpu ( dst cc src1 src2 -- ) -HOOK: %compare-imm cpu ( dst cc src1 src2 -- ) -HOOK: %compare-float cpu ( dst cc src1 src2 -- ) +HOOK: %compare cpu ( dst temp cc src1 src2 -- ) +HOOK: %compare-imm cpu ( dst temp cc src1 src2 -- ) +HOOK: %compare-float cpu ( dst temp cc src1 src2 -- ) HOOK: %compare-branch cpu ( label cc src1 src2 -- ) HOOK: %compare-imm-branch cpu ( label cc src1 src2 -- ) @@ -141,10 +148,10 @@ HOOK: %loop-entry cpu ( -- ) HOOK: small-enough? cpu ( n -- ? ) ! Is this structure small enough to be returned in registers? -HOOK: struct-small-enough? cpu ( heap-size -- ? ) +HOOK: struct-small-enough? cpu ( c-type -- ? ) -! Do we pass value structs by value or hidden reference? -HOOK: value-structs? cpu ( -- ? ) +! Do we pass this struct by value or hidden reference? +HOOK: value-struct? cpu ( c-type -- ? ) ! If t, all parameters are shadowed by dummy stack parameters HOOK: dummy-stack-params? cpu ( -- ? ) @@ -207,14 +214,3 @@ M: object %callback-return drop %return ; M: stack-params param-reg drop ; M: stack-params param-regs drop f ; - -: if-small-struct ( n size true false -- ? ) - [ 2dup [ not ] [ struct-small-enough? ] bi* and ] 2dip - [ '[ nip @ ] ] dip if ; - inline - -: %unbox-struct ( n c-type -- ) - [ %unbox-small-struct ] [ %unbox-large-struct ] if-small-struct ; - -: %box-struct ( n c-type -- ) - [ %box-small-struct ] [ %box-large-struct ] if-small-struct ; diff --git a/basis/cpu/ppc/bootstrap.factor b/basis/cpu/ppc/bootstrap.factor index 9bf88185c5..d22ff4d615 100644 --- a/basis/cpu/ppc/bootstrap.factor +++ b/basis/cpu/ppc/bootstrap.factor @@ -24,7 +24,6 @@ big-endian on [ 0 6 LOAD32 - 6 dup 0 LWZ 11 6 profile-count-offset LWZ 11 11 1 tag-fixnum ADDI 11 6 profile-count-offset STW @@ -32,7 +31,7 @@ big-endian on 11 11 compiled-header-size ADDI 11 MTCTR BCTR -] rc-absolute-ppc-2/2 rt-literal 1 jit-profiling jit-define +] rc-absolute-ppc-2/2 rt-immediate 1 jit-profiling jit-define [ 0 6 LOAD32 @@ -44,12 +43,6 @@ big-endian on 0 1 lr-save stack-frame + STW ] rc-absolute-ppc-2/2 rt-label 1 jit-prolog jit-define -[ - 0 6 LOAD32 - 6 dup 0 LWZ - 6 ds-reg 4 STWU -] rc-absolute-ppc-2/2 rt-literal 1 jit-push-literal jit-define - [ 0 6 LOAD32 6 ds-reg 4 STWU @@ -57,7 +50,12 @@ big-endian on [ 0 6 LOAD32 - 4 1 MR + 7 6 0 LWZ + 1 7 0 STW +] rc-absolute-ppc-2/2 rt-stack-chain 1 jit-save-stack jit-define + +[ + 0 6 LOAD32 6 MTCTR BCTR ] rc-absolute-ppc-2/2 rt-primitive 1 jit-primitive jit-define @@ -66,7 +64,19 @@ big-endian on [ 0 B ] rc-relative-ppc-3 rt-xt 0 jit-word-jump jit-define -: jit-call-quot ( -- ) +[ + 3 ds-reg 0 LWZ + ds-reg dup 4 SUBI + 0 3 \ f tag-number CMPI + 2 BEQ + 0 B +] rc-relative-ppc-3 rt-xt 4 jit-if-1 jit-define + +[ + 0 B +] rc-relative-ppc-3 rt-xt 0 jit-if-2 jit-define + +: jit-jump-quot ( -- ) 4 3 quot-xt-offset LWZ 4 MTCTR BCTR ; @@ -74,24 +84,76 @@ big-endian on [ 0 3 LOAD32 6 ds-reg 0 LWZ - 0 6 \ f tag-number CMPI - 2 BNE - 3 3 4 ADDI - 3 3 0 LWZ - ds-reg dup 4 SUBI - jit-call-quot -] rc-absolute-ppc-2/2 rt-literal 1 jit-if-jump jit-define - -[ - 0 3 LOAD32 - 3 3 0 LWZ - 6 ds-reg 0 LWZ 6 6 1 SRAWI 3 3 6 ADD 3 3 array-start-offset LWZ ds-reg dup 4 SUBI - jit-call-quot -] rc-absolute-ppc-2/2 rt-literal 1 jit-dispatch jit-define + jit-jump-quot +] rc-absolute-ppc-2/2 rt-immediate 1 jit-dispatch jit-define + +: jit->r ( -- ) + 4 ds-reg 0 LWZ + ds-reg dup 4 SUBI + 4 rs-reg 4 STWU ; + +: jit-2>r ( -- ) + 4 ds-reg 0 LWZ + 5 ds-reg -4 LWZ + ds-reg dup 8 SUBI + rs-reg dup 8 ADDI + 4 rs-reg 0 STW + 5 rs-reg -4 STW ; + +: jit-3>r ( -- ) + 4 ds-reg 0 LWZ + 5 ds-reg -4 LWZ + 6 ds-reg -8 LWZ + ds-reg dup 12 SUBI + rs-reg dup 12 ADDI + 4 rs-reg 0 STW + 5 rs-reg -4 STW + 6 rs-reg -8 STW ; + +: jit-r> ( -- ) + 4 rs-reg 0 LWZ + rs-reg dup 4 SUBI + 4 ds-reg 4 STWU ; + +: jit-2r> ( -- ) + 4 rs-reg 0 LWZ + 5 rs-reg -4 LWZ + rs-reg dup 8 SUBI + ds-reg dup 8 ADDI + 4 ds-reg 0 STW + 5 ds-reg -4 STW ; + +: jit-3r> ( -- ) + 4 rs-reg 0 LWZ + 5 rs-reg -4 LWZ + 6 rs-reg -8 LWZ + rs-reg dup 12 SUBI + ds-reg dup 12 ADDI + 4 ds-reg 0 STW + 5 ds-reg -4 STW + 6 ds-reg -8 STW ; + +[ + jit->r + 0 BL + jit-r> +] rc-relative-ppc-3 rt-xt 3 jit-dip jit-define + +[ + jit-2>r + 0 BL + jit-2r> +] rc-relative-ppc-3 rt-xt 6 jit-2dip jit-define + +[ + jit-3>r + 0 BL + jit-3r> +] rc-relative-ppc-3 rt-xt 8 jit-3dip jit-define [ 0 1 lr-save stack-frame + LWZ @@ -107,7 +169,7 @@ big-endian on [ 3 ds-reg 0 LWZ ds-reg dup 4 SUBI - jit-call-quot + jit-jump-quot ] f f f \ (call) define-sub-primitive [ @@ -240,22 +302,13 @@ big-endian on 4 ds-reg 0 STW ] f f f \ -rot define-sub-primitive -[ - 3 ds-reg 0 LWZ - ds-reg dup 4 SUBI - 3 rs-reg 4 STWU -] f f f \ >r define-sub-primitive +[ jit->r ] f f f \ >r define-sub-primitive -[ - 3 rs-reg 0 LWZ - rs-reg dup 4 SUBI - 3 ds-reg 4 STWU -] f f f \ r> define-sub-primitive +[ jit-r> ] f f f \ r> define-sub-primitive ! Comparisons : jit-compare ( insn -- ) 0 3 LOAD32 - 3 3 0 LWZ 4 ds-reg 0 LWZ 5 ds-reg -4 LWZU 5 0 4 CMP @@ -264,7 +317,7 @@ big-endian on 3 ds-reg 0 STW ; : define-jit-compare ( insn word -- ) - [ [ jit-compare ] curry rc-absolute-ppc-2/2 rt-literal 1 ] dip + [ [ jit-compare ] curry rc-absolute-ppc-2/2 rt-immediate 1 ] dip define-sub-primitive ; \ BEQ \ eq? define-jit-compare @@ -274,6 +327,18 @@ big-endian on \ BLT \ fixnum< define-jit-compare ! Math +[ + 3 ds-reg 0 LWZ + 4 ds-reg -4 LWZ + 3 3 4 OR + 3 3 tag-mask get ANDI + \ f tag-number 4 LI + 0 3 0 CMPI + 2 BNE + 1 tag-fixnum 4 LI + 4 ds-reg 4 STWU +] f f f \ both-fixnums? define-sub-primitive + : jit-math ( insn -- ) 3 ds-reg 0 LWZ 4 ds-reg -4 LWZU @@ -330,12 +395,30 @@ big-endian on 7 ds-reg 0 STW ] f f f \ fixnum-mod define-sub-primitive +[ + 3 ds-reg 0 LWZ + ds-reg ds-reg 4 SUBI + 4 ds-reg 0 LWZ + 5 4 3 DIVW + 5 5 tag-bits get SLWI + 5 ds-reg 0 STW +] f f f \ fixnum/i-fast define-sub-primitive + +[ + 3 ds-reg 0 LWZ + 4 ds-reg -4 LWZ + 5 4 3 DIVW + 6 5 3 MULLW + 7 6 4 SUBF + 5 5 tag-bits get SLWI + 5 ds-reg -4 STW + 7 ds-reg 0 STW +] f f f \ fixnum/mod-fast define-sub-primitive + [ 3 ds-reg 0 LWZ 3 3 1 SRAWI - 4 4 LI - 4 3 4 SUBF - rs-reg 3 4 LWZX + rs-reg 3 3 LWZX 3 ds-reg 0 STW ] f f f \ get-local define-sub-primitive diff --git a/basis/cpu/ppc/linux/linux.factor b/basis/cpu/ppc/linux/linux.factor index 090495aa11..5cfa1391c4 100644 --- a/basis/cpu/ppc/linux/linux.factor +++ b/basis/cpu/ppc/linux/linux.factor @@ -15,7 +15,7 @@ M: linux lr-save 1 cells ; M: float-regs param-regs drop { 1 2 3 4 5 6 7 8 } ; -M: ppc value-structs? f ; +M: ppc value-struct? drop f ; M: ppc dummy-stack-params? f ; diff --git a/basis/cpu/ppc/macosx/macosx.factor b/basis/cpu/ppc/macosx/macosx.factor index 877fb37d31..c742cf2ddc 100644 --- a/basis/cpu/ppc/macosx/macosx.factor +++ b/basis/cpu/ppc/macosx/macosx.factor @@ -16,7 +16,7 @@ M: macosx lr-save 2 cells ; M: float-regs param-regs drop { 1 2 3 4 5 6 7 8 9 10 11 12 13 } ; -M: ppc value-structs? t ; +M: ppc value-struct? drop t ; M: ppc dummy-stack-params? t ; diff --git a/basis/cpu/ppc/ppc.factor b/basis/cpu/ppc/ppc.factor index 49caae4bb8..6b51585750 100644 --- a/basis/cpu/ppc/ppc.factor +++ b/basis/cpu/ppc/ppc.factor @@ -34,10 +34,11 @@ M: ppc two-operand? f ; M: ppc %load-immediate ( reg n -- ) swap LOAD ; -M:: ppc %load-indirect ( reg obj -- ) - 0 reg LOAD32 - obj rc-absolute-ppc-2/2 rel-literal - reg reg 0 LWZ ; +M: ppc %load-indirect ( reg obj -- ) + [ 0 swap LOAD32 ] [ rc-absolute-ppc-2/2 rel-immediate ] bi* ; + +: %load-dlsym ( symbol dll register -- ) + 0 swap LOAD32 rc-absolute-ppc-2/2 rel-dlsym ; : ds-reg 29 ; inline : rs-reg 30 ; inline @@ -111,10 +112,10 @@ M: ppc %call ( label -- ) BL ; M: ppc %jump-label ( label -- ) B ; M: ppc %return ( -- ) BLR ; -M:: ppc %dispatch ( src temp -- ) - 0 temp LOAD32 rc-absolute-ppc-2/2 rel-here - temp temp src ADD - temp temp 5 cells LWZ +M:: ppc %dispatch ( src temp offset -- ) + 0 temp LOAD32 + 4 offset + cells rc-absolute-ppc-2/2 rel-here + temp temp src LWZX temp MTCTR BCTR ; @@ -166,6 +167,91 @@ M: ppc %shr-imm swapd SRWI ; M: ppc %sar-imm SRAWI ; M: ppc %not NOT ; +: %alien-invoke-tail ( func dll -- ) + scratch-reg %load-dlsym scratch-reg MTCTR BCTR ; + +:: exchange-regs ( r1 r2 -- ) + scratch-reg r1 MR + r1 r2 MR + r2 scratch-reg MR ; + +: ?MR ( r1 r2 -- ) 2dup = [ 2drop ] [ MR ] if ; + +:: move>args ( src1 src2 -- ) + { + { [ src1 4 = ] [ 3 src2 ?MR 3 4 exchange-regs ] } + { [ src1 3 = ] [ 4 src2 ?MR ] } + { [ src2 3 = ] [ 4 src1 ?MR 3 4 exchange-regs ] } + { [ src2 4 = ] [ 3 src1 ?MR ] } + [ 3 src1 MR 4 src2 MR ] + } cond ; + +: clear-xer ( -- ) + 0 0 LI + 0 MTXER ; inline + +:: overflow-template ( src1 src2 insn func -- ) + "no-overflow" define-label + clear-xer + scratch-reg src2 src1 insn call + scratch-reg ds-reg 0 STW + "no-overflow" get BNO + src1 src2 move>args + %prepare-alien-invoke + func f %alien-invoke + "no-overflow" resolve-label ; inline + +:: overflow-template-tail ( src1 src2 insn func -- ) + "overflow" define-label + clear-xer + scratch-reg src2 src1 insn call + "overflow" get BO + scratch-reg ds-reg 0 STW + BLR + "overflow" resolve-label + src1 src2 move>args + %prepare-alien-invoke + func f %alien-invoke-tail ; inline + +M: ppc %fixnum-add ( src1 src2 -- ) + [ ADDO. ] "overflow_fixnum_add" overflow-template ; + +M: ppc %fixnum-add-tail ( src1 src2 -- ) + [ ADDO. ] "overflow_fixnum_add" overflow-template-tail ; + +M: ppc %fixnum-sub ( src1 src2 -- ) + [ SUBFO. ] "overflow_fixnum_subtract" overflow-template ; + +M: ppc %fixnum-sub-tail ( src1 src2 -- ) + [ SUBFO. ] "overflow_fixnum_subtract" overflow-template-tail ; + +M:: ppc %fixnum-mul ( src1 src2 temp1 temp2 -- ) + "no-overflow" define-label + clear-xer + temp1 src1 tag-bits get SRAWI + temp2 temp1 src2 MULLWO. + temp2 ds-reg 0 STW + "no-overflow" get BNO + src2 src2 tag-bits get SRAWI + temp1 src2 move>args + %prepare-alien-invoke + "overflow_fixnum_multiply" f %alien-invoke + "no-overflow" resolve-label ; + +M:: ppc %fixnum-mul-tail ( src1 src2 temp1 temp2 -- ) + "overflow" define-label + clear-xer + temp1 src1 tag-bits get SRAWI + temp2 temp1 src2 MULLWO. + "overflow" get BO + temp2 ds-reg 0 STW + BLR + "overflow" resolve-label + src2 src2 tag-bits get SRAWI + temp1 src2 move>args + %prepare-alien-invoke + "overflow_fixnum_multiply" f %alien-invoke-tail ; + : bignum@ ( n -- offset ) cells bignum tag-number - ; inline M:: ppc %integer>bignum ( dst src temp -- ) @@ -320,9 +406,6 @@ M: ppc %set-alien-cell swap 0 STW ; M: ppc %set-alien-float swap 0 STFS ; M: ppc %set-alien-double swap 0 STFD ; -: %load-dlsym ( symbol dll register -- ) - 0 swap LOAD32 rc-absolute-ppc-2/2 rel-dlsym ; - : load-zone-ptr ( reg -- ) [ "nursery" f ] dip %load-dlsym ; @@ -398,14 +481,14 @@ M: ppc %epilogue ( n -- ) 1 1 rot ADDI 0 MTLR ; -:: (%boolean) ( dst word -- ) +:: (%boolean) ( dst temp word -- ) "end" define-label dst \ f tag-number %load-immediate "end" get word execute dst \ t %load-indirect "end" get resolve-label ; inline -: %boolean ( dst cc -- ) +: %boolean ( dst temp cc -- ) negate-cc { { cc< [ \ BLT (%boolean) ] } { cc<= [ \ BLE (%boolean) ] } @@ -540,11 +623,11 @@ M: ppc %prepare-alien-invoke #! Save Factor stack pointers in case the C code calls a #! callback which does a GC, which must reliably trace #! all roots. - "stack_chain" f 11 %load-dlsym - 11 11 0 LWZ - 1 11 0 STW - ds-reg 11 8 STW - rs-reg 11 12 STW ; + "stack_chain" f scratch-reg %load-dlsym + scratch-reg scratch-reg 0 LWZ + 1 scratch-reg 0 STW + ds-reg scratch-reg 8 STW + rs-reg scratch-reg 12 STW ; M: ppc %alien-invoke ( symbol dll -- ) 11 %load-dlsym 11 MTLR BLRL ; diff --git a/basis/cpu/x86/32/32.factor b/basis/cpu/x86/32/32.factor old mode 100644 new mode 100755 index f26d76551a..3df072208d --- a/basis/cpu/x86/32/32.factor +++ b/basis/cpu/x86/32/32.factor @@ -6,7 +6,7 @@ accessors init combinators command-line cpu.x86.assembler cpu.x86 cpu.architecture compiler compiler.units compiler.constants compiler.alien compiler.codegen compiler.codegen.fixup compiler.cfg.instructions -compiler.cfg.builder compiler.cfg.intrinsics ; +compiler.cfg.builder compiler.cfg.intrinsics make ; IN: cpu.x86.32 ! We implement the FFI for Linux, OS X and Windows all at once. @@ -23,8 +23,24 @@ M: x86.32 machine-registers M: x86.32 ds-reg ESI ; M: x86.32 rs-reg EDI ; M: x86.32 stack-reg ESP ; -M: x86.32 temp-reg-1 EAX ; -M: x86.32 temp-reg-2 ECX ; +M: x86.32 temp-reg-1 ECX ; +M: x86.32 temp-reg-2 EDX ; + +M:: x86.32 %dispatch ( src temp offset -- ) + ! Load jump table base. + src HEX: ffffffff ADD + offset cells rc-absolute-cell rel-here + ! Go + src HEX: 7f [+] JMP + ! Fix up the displacement above + cell code-alignment + [ 7 + building get dup pop* push ] + [ align-code ] + bi ; + +! Registers for fastcall +M: x86.32 param-reg-1 EAX ; +M: x86.32 param-reg-2 EDX ; M: x86.32 reserved-area-size 0 ; @@ -32,6 +48,8 @@ M: x86.32 %alien-global 0 [] MOV rc-absolute-cell rel-dlsym ; M: x86.32 %alien-invoke (CALL) rel-dlsym ; +M: x86.32 %alien-invoke-tail (JMP) rel-dlsym ; + M: x86.32 struct-small-enough? ( size -- ? ) heap-size { 1 2 4 8 } member? os { linux netbsd solaris } member? not and ; @@ -76,8 +94,6 @@ M: float-regs store-return-reg [ [ align-sub ] [ call ] bi* ] [ [ align-add ] [ drop ] bi* ] 2bi ; inline -M: x86.32 rel-literal-x86 rc-absolute-cell rel-literal ; - M: x86.32 %prologue ( n -- ) dup PUSH 0 PUSH rc-absolute-cell rel-this @@ -291,7 +307,7 @@ FUNCTION: bool check_sse2 ( ) ; : sse2? ( -- ? ) check_sse2 ; -"-no-sse2" cli-args member? [ +"-no-sse2" (command-line) member? [ [ optimized-recompile-hook ] recompile-hook [ { check_sse2 } compile ] with-variable diff --git a/basis/cpu/x86/32/bootstrap.factor b/basis/cpu/x86/32/bootstrap.factor index 44f840e66a..04bdcca68b 100644 --- a/basis/cpu/x86/32/bootstrap.factor +++ b/basis/cpu/x86/32/bootstrap.factor @@ -1,7 +1,7 @@ ! Copyright (C) 2007 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: bootstrap.image.private kernel namespaces system -cpu.x86.assembler layouts vocabs parser ; +cpu.x86.assembler layouts vocabs parser compiler.constants ; IN: bootstrap.x86 4 \ cell set @@ -12,6 +12,7 @@ IN: bootstrap.x86 : mod-arg ( -- reg ) EDX ; : arg0 ( -- reg ) EAX ; : arg1 ( -- reg ) EDX ; +: arg2 ( -- reg ) ECX ; : temp-reg ( -- reg ) EBX ; : stack-reg ( -- reg ) ESP ; : ds-reg ( -- reg ) ESI ; @@ -19,5 +20,14 @@ IN: bootstrap.x86 : fixnum>slot@ ( -- ) arg0 1 SAR ; : rex-length ( -- n ) 0 ; +[ + arg0 0 [] MOV ! load stack_chain + arg0 [] stack-reg MOV ! save stack pointer +] rc-absolute-cell rt-stack-chain 2 jit-save-stack jit-define + +[ + (JMP) drop +] rc-relative rt-primitive 1 jit-primitive jit-define + << "resource:basis/cpu/x86/bootstrap.factor" parse-file parsed >> call diff --git a/basis/cpu/x86/64/64.factor b/basis/cpu/x86/64/64.factor index 0d20660021..6472ec0edf 100644 --- a/basis/cpu/x86/64/64.factor +++ b/basis/cpu/x86/64/64.factor @@ -2,7 +2,7 @@ ! See http://factorcode.org/license.txt for BSD license. USING: accessors arrays kernel math namespaces make sequences system layouts alien alien.c-types alien.accessors alien.structs -slots splitting assocs combinators cpu.x86.assembler +slots splitting assocs combinators make locals cpu.x86.assembler cpu.x86 cpu.architecture compiler.constants compiler.codegen compiler.codegen.fixup compiler.cfg.instructions compiler.cfg.builder @@ -21,18 +21,27 @@ M: x86.64 machine-registers M: x86.64 ds-reg R14 ; M: x86.64 rs-reg R15 ; M: x86.64 stack-reg RSP ; -M: x86.64 temp-reg-1 RAX ; -M: x86.64 temp-reg-2 RCX ; -: param-reg-1 int-regs param-regs first ; inline -: param-reg-2 int-regs param-regs second ; inline +M:: x86.64 %dispatch ( src temp offset -- ) + ! Load jump table base. + temp HEX: ffffffff MOV + offset cells rc-absolute-cell rel-here + ! Add jump table base + src temp ADD + src HEX: 7f [+] JMP + ! Fix up the displacement above + cell code-alignment + [ 15 + building get dup pop* push ] + [ align-code ] + bi ; + +M: x86.64 param-reg-1 int-regs param-regs first ; +M: x86.64 param-reg-2 int-regs param-regs second ; : param-reg-3 int-regs param-regs third ; inline M: int-regs return-reg drop RAX ; M: float-regs return-reg drop XMM0 ; -M: x86.64 rel-literal-x86 rc-relative rel-literal ; - M: x86.64 %prologue ( n -- ) temp-reg-1 0 MOV rc-absolute-cell rel-this dup PUSH @@ -157,6 +166,11 @@ M: x86.64 %alien-invoke rc-absolute-cell rel-dlsym R11 CALL ; +M: x86.64 %alien-invoke-tail + R11 0 MOV + rc-absolute-cell rel-dlsym + R11 JMP ; + M: x86.64 %prepare-alien-indirect ( -- ) "unbox_alien" f %alien-invoke RBP RAX MOV ; diff --git a/basis/cpu/x86/64/bootstrap.factor b/basis/cpu/x86/64/bootstrap.factor index acac8b55bc..83a72d6dd3 100644 --- a/basis/cpu/x86/64/bootstrap.factor +++ b/basis/cpu/x86/64/bootstrap.factor @@ -1,7 +1,7 @@ ! Copyright (C) 2007 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: bootstrap.image.private kernel namespaces system -cpu.x86.assembler layouts vocabs parser ; +cpu.x86.assembler layouts vocabs parser compiler.constants math ; IN: bootstrap.x86 8 \ cell set @@ -16,5 +16,16 @@ IN: bootstrap.x86 : fixnum>slot@ ( -- ) ; : rex-length ( -- n ) 1 ; +[ + arg0 0 MOV ! load stack_chain + arg0 arg0 [] MOV + arg0 [] stack-reg MOV ! save stack pointer +] rc-absolute-cell rt-stack-chain 1 rex-length + jit-save-stack jit-define + +[ + arg1 0 MOV ! load XT + arg1 JMP ! go +] rc-absolute-cell rt-primitive 1 rex-length + jit-primitive jit-define + << "resource:basis/cpu/x86/bootstrap.factor" parse-file parsed >> call diff --git a/basis/cpu/x86/64/unix/bootstrap.factor b/basis/cpu/x86/64/unix/bootstrap.factor index 29d48bd794..f0ca56da14 100644 --- a/basis/cpu/x86/64/unix/bootstrap.factor +++ b/basis/cpu/x86/64/unix/bootstrap.factor @@ -7,6 +7,7 @@ IN: bootstrap.x86 : stack-frame-size ( -- n ) 4 bootstrap-cells ; : arg0 ( -- reg ) RDI ; : arg1 ( -- reg ) RSI ; +: arg2 ( -- reg ) RDX ; << "resource:basis/cpu/x86/64/bootstrap.factor" parse-file parsed >> call diff --git a/basis/cpu/x86/64/unix/unix.factor b/basis/cpu/x86/64/unix/unix.factor index ddb412873a..f5fb5b9640 100644 --- a/basis/cpu/x86/64/unix/unix.factor +++ b/basis/cpu/x86/64/unix/unix.factor @@ -52,3 +52,7 @@ M: x86.64 dummy-stack-params? f ; M: x86.64 dummy-int-params? f ; M: x86.64 dummy-fp-params? f ; + +M: x86.64 temp-reg-1 R8 ; + +M: x86.64 temp-reg-2 R9 ; diff --git a/basis/cpu/x86/64/winnt/bootstrap.factor b/basis/cpu/x86/64/winnt/bootstrap.factor index a62b946e83..459945d82e 100644 --- a/basis/cpu/x86/64/winnt/bootstrap.factor +++ b/basis/cpu/x86/64/winnt/bootstrap.factor @@ -7,6 +7,7 @@ IN: bootstrap.x86 : stack-frame-size ( -- n ) 8 bootstrap-cells ; : arg0 ( -- reg ) RCX ; : arg1 ( -- reg ) RDX ; +: arg2 ( -- reg ) R8 ; << "resource:basis/cpu/x86/64/bootstrap.factor" parse-file parsed >> call diff --git a/basis/cpu/x86/64/winnt/winnt.factor b/basis/cpu/x86/64/winnt/winnt.factor index 0124c40877..4c6af6c1e7 100644 --- a/basis/cpu/x86/64/winnt/winnt.factor +++ b/basis/cpu/x86/64/winnt/winnt.factor @@ -1,6 +1,6 @@ ! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. -USING: kernel layouts system math alien.c-types +USING: kernel layouts system math alien.c-types sequences compiler.cfg.registers cpu.architecture cpu.x86.assembler cpu.x86 ; IN: cpu.x86.64.winnt @@ -10,8 +10,9 @@ M: float-regs param-regs drop { XMM0 XMM1 XMM2 XMM3 } ; M: x86.64 reserved-area-size 4 cells ; -M: x86.64 struct-small-enough? ( size -- ? ) - heap-size cell <= ; +M: x86.64 struct-small-enough? heap-size { 1 2 4 8 } member? ; + +M: x86.64 value-struct? heap-size { 1 2 4 8 } member? ; M: x86.64 dummy-stack-params? f ; @@ -19,8 +20,13 @@ M: x86.64 dummy-int-params? t ; M: x86.64 dummy-fp-params? t ; +M: x86.64 temp-reg-1 RAX ; + +M: x86.64 temp-reg-2 RCX ; + << "longlong" "ptrdiff_t" typedef -"int" "long" typedef -"uint" "ulong" typedef +"longlong" "intptr_t" typedef +"int" c-type "long" define-primitive-type +"uint" c-type "ulong" define-primitive-type >> diff --git a/basis/cpu/x86/assembler/assembler.factor b/basis/cpu/x86/assembler/assembler.factor index 5c6fff2348..27c00cb3c0 100644 --- a/basis/cpu/x86/assembler/assembler.factor +++ b/basis/cpu/x86/assembler/assembler.factor @@ -130,7 +130,7 @@ M: register modifier drop BIN: 11 ; GENERIC# n, 1 ( value n -- ) M: integer n, >le % ; -M: byte n, >r value>> r> n, ; +M: byte n, [ value>> ] dip n, ; : 1, ( n -- ) 1 n, ; inline : 4, ( n -- ) 4 n, ; inline : 2, ( n -- ) 2 n, ; inline @@ -209,7 +209,7 @@ M: object operand-64? drop f ; : short-operand ( reg rex.w n -- ) #! Some instructions encode their single operand as part of #! the opcode. - >r dupd prefix-1 reg-code r> + , ; + [ dupd prefix-1 reg-code ] dip + , ; : opcode, ( opcode -- ) dup array? [ % ] [ , ] if ; @@ -224,7 +224,7 @@ M: object operand-64? drop f ; : 1-operand ( op reg,rex.w,opcode -- ) #! The 'reg' is not really a register, but a value for the #! 'reg' field of the mod-r/m byte. - first3 >r >r over r> prefix-1 r> opcode, swap addressing ; + first3 [ [ over ] dip prefix-1 ] dip opcode, swap addressing ; : immediate-operand-size-bit ( imm dst reg,rex.w,opcode -- imm dst reg,rex.w,opcode ) pick integer? [ first3 BIN: 1 opcode-or 3array ] when ; @@ -250,7 +250,7 @@ M: object operand-64? drop f ; ] if ; : (2-operand) ( dst src op -- ) - >r 2dup t rex-prefix r> opcode, + [ 2dup t rex-prefix ] dip opcode, reg-code swap addressing ; : direction-bit ( dst src op -- dst' src' op' ) @@ -271,11 +271,11 @@ M: object operand-64? drop f ; PRIVATE> : [] ( reg/displacement -- indirect ) - dup integer? [ >r f f f r> ] [ f f f ] if ; + dup integer? [ [ f f f ] dip ] [ f f f ] if ; : [+] ( reg displacement -- indirect ) dup integer? - [ dup zero? [ drop f ] when >r f f r> ] + [ dup zero? [ drop f ] when [ f f ] dip ] [ f f ] if ; @@ -300,7 +300,7 @@ PREDICATE: callable < word register? not ; GENERIC: MOV ( dst src -- ) M: immediate MOV swap (MOV-I) ; -M: callable MOV 0 rot (MOV-I) rc-absolute-cell rel-word ; +M: callable MOV [ 0 ] 2dip (MOV-I) rc-absolute-cell rel-word ; M: operand MOV HEX: 88 2-operand ; : LEA ( dst src -- ) swap HEX: 8d 2-operand ; @@ -308,18 +308,21 @@ M: operand MOV HEX: 88 2-operand ; ! Control flow GENERIC: JMP ( op -- ) : (JMP) ( -- rel-class ) HEX: e9 , 0 4, rc-relative ; +M: f JMP (JMP) 2drop ; M: callable JMP (JMP) rel-word ; M: label JMP (JMP) label-fixup ; M: operand JMP { BIN: 100 t HEX: ff } 1-operand ; GENERIC: CALL ( op -- ) : (CALL) ( -- rel-class ) HEX: e8 , 0 4, rc-relative ; +M: f CALL (CALL) 2drop ; M: callable CALL (CALL) rel-word ; M: label CALL (CALL) label-fixup ; M: operand CALL { BIN: 010 t HEX: ff } 1-operand ; GENERIC# JUMPcc 1 ( addr opcode -- ) : (JUMPcc) ( n -- rel-class ) extended-opcode, 0 4, rc-relative ; +M: f JUMPcc nip (JUMPcc) drop ; M: callable JUMPcc (JUMPcc) rel-word ; M: label JUMPcc (JUMPcc) label-fixup ; diff --git a/basis/cpu/x86/assembler/syntax/syntax.factor b/basis/cpu/x86/assembler/syntax/syntax.factor index d267baaf4f..6ddec4af07 100644 --- a/basis/cpu/x86/assembler/syntax/syntax.factor +++ b/basis/cpu/x86/assembler/syntax/syntax.factor @@ -4,8 +4,8 @@ USING: kernel words sequences lexer parser fry ; IN: cpu.x86.assembler.syntax : define-register ( name num size -- ) - >r >r "cpu.x86.assembler" create dup define-symbol r> r> - >r dupd "register" set-word-prop r> + [ "cpu.x86.assembler" create dup define-symbol ] 2dip + [ dupd "register" set-word-prop ] dip "register-size" set-word-prop ; : define-registers ( names size -- ) diff --git a/basis/cpu/x86/bootstrap.factor b/basis/cpu/x86/bootstrap.factor index 6dadbc096c..42df1c8437 100644 --- a/basis/cpu/x86/bootstrap.factor +++ b/basis/cpu/x86/bootstrap.factor @@ -13,7 +13,6 @@ big-endian off [ ! Load word temp-reg 0 MOV - temp-reg dup [] MOV ! Bump profiling counter temp-reg profile-count-offset [+] 1 tag-fixnum ADD ! Load word->code @@ -22,7 +21,7 @@ big-endian off temp-reg compiled-header-size ADD ! Jump to XT temp-reg JMP -] rc-absolute-cell rt-literal 1 rex-length + jit-profiling jit-define +] rc-absolute-cell rt-immediate 1 rex-length + jit-profiling jit-define [ temp-reg 0 MOV ! load XT @@ -31,13 +30,6 @@ big-endian off stack-reg stack-frame-size 3 bootstrap-cells - SUB ! alignment ] rc-absolute-cell rt-label 1 rex-length + jit-prolog jit-define -[ - arg0 0 MOV ! load literal - arg0 dup [] MOV - ds-reg bootstrap-cell ADD ! increment datastack pointer - ds-reg [] arg0 MOV ! store literal on datastack -] rc-absolute-cell rt-literal 1 rex-length + jit-push-literal jit-define - [ arg0 0 MOV ! load literal ds-reg bootstrap-cell ADD ! increment datastack pointer @@ -45,39 +37,99 @@ big-endian off ] rc-absolute-cell rt-immediate 1 rex-length + jit-push-immediate jit-define [ - arg0 0 MOV ! load XT - arg1 stack-reg MOV ! pass callstack pointer as arg 2 - arg0 JMP ! go -] rc-absolute-cell rt-primitive 1 rex-length + jit-primitive jit-define - -[ - (JMP) drop + f JMP ] rc-relative rt-xt 1 jit-word-jump jit-define [ - (CALL) drop + f CALL ] rc-relative rt-xt 1 jit-word-call jit-define [ - arg1 0 MOV ! load addr of true quotation arg0 ds-reg [] MOV ! load boolean ds-reg bootstrap-cell SUB ! pop boolean - arg0 \ f tag-number CMP ! compare it with f - arg0 arg1 [] CMOVNE ! load true branch if not equal - arg0 arg1 bootstrap-cell [+] CMOVE ! load false branch if equal - arg0 quot-xt-offset [+] JMP ! jump to quotation-xt -] rc-absolute-cell rt-literal 1 rex-length + jit-if-jump jit-define + arg0 \ f tag-number CMP ! compare boolean with f + f JNE ! jump to true branch if not equal +] rc-relative rt-xt 10 rex-length 3 * + jit-if-1 jit-define + +[ + f JMP ! jump to false branch if equal +] rc-relative rt-xt 1 jit-if-2 jit-define [ arg1 0 MOV ! load dispatch table - arg1 dup [] MOV arg0 ds-reg [] MOV ! load index fixnum>slot@ ! turn it into an array offset ds-reg bootstrap-cell SUB ! pop index arg0 arg1 ADD ! compute quotation location arg0 arg0 array-start-offset [+] MOV ! load quotation arg0 quot-xt-offset [+] JMP ! execute branch -] rc-absolute-cell rt-literal 1 rex-length + jit-dispatch jit-define +] rc-absolute-cell rt-immediate 1 rex-length + jit-dispatch jit-define + +: jit->r ( -- ) + rs-reg bootstrap-cell ADD + arg0 ds-reg [] MOV + ds-reg bootstrap-cell SUB + rs-reg [] arg0 MOV ; + +: jit-2>r ( -- ) + rs-reg 2 bootstrap-cells ADD + arg0 ds-reg [] MOV + arg1 ds-reg -1 bootstrap-cells [+] MOV + ds-reg 2 bootstrap-cells SUB + rs-reg [] arg0 MOV + rs-reg -1 bootstrap-cells [+] arg1 MOV ; + +: jit-3>r ( -- ) + rs-reg 3 bootstrap-cells ADD + arg0 ds-reg [] MOV + arg1 ds-reg -1 bootstrap-cells [+] MOV + arg2 ds-reg -2 bootstrap-cells [+] MOV + ds-reg 3 bootstrap-cells SUB + rs-reg [] arg0 MOV + rs-reg -1 bootstrap-cells [+] arg1 MOV + rs-reg -2 bootstrap-cells [+] arg2 MOV ; + +: jit-r> ( -- ) + ds-reg bootstrap-cell ADD + arg0 rs-reg [] MOV + rs-reg bootstrap-cell SUB + ds-reg [] arg0 MOV ; + +: jit-2r> ( -- ) + ds-reg 2 bootstrap-cells ADD + arg0 rs-reg [] MOV + arg1 rs-reg -1 bootstrap-cells [+] MOV + rs-reg 2 bootstrap-cells SUB + ds-reg [] arg0 MOV + ds-reg -1 bootstrap-cells [+] arg1 MOV ; + +: jit-3r> ( -- ) + ds-reg 3 bootstrap-cells ADD + arg0 rs-reg [] MOV + arg1 rs-reg -1 bootstrap-cells [+] MOV + arg2 rs-reg -2 bootstrap-cells [+] MOV + rs-reg 3 bootstrap-cells SUB + ds-reg [] arg0 MOV + ds-reg -1 bootstrap-cells [+] arg1 MOV + ds-reg -2 bootstrap-cells [+] arg2 MOV ; + +[ + jit->r + f CALL + jit-r> +] rc-relative rt-xt 11 rex-length 4 * + jit-dip jit-define + +[ + jit-2>r + f CALL + jit-2r> +] rc-relative rt-xt 17 rex-length 6 * + jit-2dip jit-define + +[ + jit-3>r + f CALL + jit-3r> +] rc-relative rt-xt 23 rex-length 8 * + jit-3dip jit-define [ stack-reg stack-frame-size bootstrap-cell - ADD ! unwind stack frame @@ -229,25 +281,14 @@ big-endian off ds-reg [] arg1 MOV ] f f f \ -rot define-sub-primitive -[ - rs-reg bootstrap-cell ADD - arg0 ds-reg [] MOV - ds-reg bootstrap-cell SUB - rs-reg [] arg0 MOV -] f f f \ >r define-sub-primitive +[ jit->r ] f f f \ >r define-sub-primitive -[ - ds-reg bootstrap-cell ADD - arg0 rs-reg [] MOV - rs-reg bootstrap-cell SUB - ds-reg [] arg0 MOV -] f f f \ r> define-sub-primitive +[ jit-r> ] f f f \ r> define-sub-primitive ! Comparisons : jit-compare ( insn -- ) - arg1 0 MOV ! load t - arg1 dup [] MOV - temp-reg \ f tag-number MOV ! load f + temp-reg 0 MOV ! load t + arg1 \ f tag-number MOV ! load f arg0 ds-reg [] MOV ! load first value ds-reg bootstrap-cell SUB ! adjust stack pointer ds-reg [] arg0 CMP ! compare with second value @@ -256,14 +297,14 @@ big-endian off ; : define-jit-compare ( insn word -- ) - [ [ jit-compare ] curry rc-absolute-cell rt-literal 1 rex-length + ] dip + [ [ jit-compare ] curry rc-absolute-cell rt-immediate 1 rex-length + ] dip define-sub-primitive ; -\ CMOVNE \ eq? define-jit-compare -\ CMOVL \ fixnum>= define-jit-compare -\ CMOVG \ fixnum<= define-jit-compare -\ CMOVLE \ fixnum> define-jit-compare -\ CMOVGE \ fixnum< define-jit-compare +\ CMOVE \ eq? define-jit-compare +\ CMOVGE \ fixnum>= define-jit-compare +\ CMOVLE \ fixnum<= define-jit-compare +\ CMOVG \ fixnum> define-jit-compare +\ CMOVL \ fixnum< define-jit-compare ! Math : jit-math ( insn -- ) @@ -311,22 +352,48 @@ big-endian off ds-reg [] arg1 MOV ! push to stack ] f f f \ fixnum-shift-fast define-sub-primitive -[ +: jit-fixnum-/mod ( -- ) temp-reg ds-reg [] MOV ! load second parameter - ds-reg bootstrap-cell SUB ! adjust stack pointer - div-arg ds-reg [] MOV ! load first parameter + div-arg ds-reg bootstrap-cell neg [+] MOV ! load first parameter mod-arg div-arg MOV ! make a copy mod-arg bootstrap-cell-bits 1- SAR ! sign-extend - temp-reg IDIV ! divide + temp-reg IDIV ; ! divide + +[ + jit-fixnum-/mod + ds-reg bootstrap-cell SUB ! adjust stack pointer ds-reg [] mod-arg MOV ! push to stack ] f f f \ fixnum-mod define-sub-primitive +[ + jit-fixnum-/mod + ds-reg bootstrap-cell SUB ! adjust stack pointer + div-arg tag-bits get SHL ! tag it + ds-reg [] div-arg MOV ! push to stack +] f f f \ fixnum/i-fast define-sub-primitive + +[ + jit-fixnum-/mod + div-arg tag-bits get SHL ! tag it + ds-reg [] mod-arg MOV ! push to stack + ds-reg bootstrap-cell neg [+] div-arg MOV +] f f f \ fixnum/mod-fast define-sub-primitive + +[ + arg0 ds-reg [] MOV + arg0 ds-reg bootstrap-cell neg [+] OR + ds-reg bootstrap-cell ADD + arg0 tag-mask get AND + arg0 \ f tag-number MOV + arg1 1 tag-fixnum MOV + arg0 arg1 CMOVE + ds-reg [] arg0 MOV +] f f f \ both-fixnums? define-sub-primitive + [ arg0 ds-reg [] MOV ! load local number fixnum>slot@ ! turn local number into offset - arg1 bootstrap-cell MOV ! load base - arg1 arg0 SUB ! turn it into a stack offset - arg0 rs-reg arg1 [+] MOV ! load local value + arg0 rs-reg arg0 [+] MOV ! load local value ds-reg [] arg0 MOV ! push to stack ] f f f \ get-local define-sub-primitive diff --git a/basis/cpu/x86/x86.factor b/basis/cpu/x86/x86.factor index 4f72fe45e1..3dbcd2eabf 100644 --- a/basis/cpu/x86/x86.factor +++ b/basis/cpu/x86/x86.factor @@ -14,11 +14,12 @@ M: x86 two-operand? t ; HOOK: temp-reg-1 cpu ( -- reg ) HOOK: temp-reg-2 cpu ( -- reg ) +HOOK: param-reg-1 cpu ( -- reg ) +HOOK: param-reg-2 cpu ( -- reg ) + M: x86 %load-immediate MOV ; -HOOK: rel-literal-x86 cpu ( literal -- ) - -M: x86 %load-indirect swap 0 [] MOV rel-literal-x86 ; +M: x86 %load-indirect swap 0 MOV rc-absolute-cell rel-immediate ; HOOK: ds-reg cpu ( -- reg ) HOOK: rs-reg cpu ( -- reg ) @@ -60,19 +61,6 @@ M: x86 %return ( -- ) 0 RET ; : align-code ( n -- ) 0 % ; -M:: x86 %dispatch ( src temp -- ) - ! Load jump table base. We use a temporary register - ! since on AMD64 we have to load a 64-bit immediate. On - ! x86, this is redundant. - ! Add jump table base - temp HEX: ffffffff MOV rc-absolute-cell rel-here - src temp ADD - src HEX: 7f [+] JMP - ! Fix up the displacement above - cell code-alignment dup bootstrap-cell 8 = 15 9 ? + - building get dup pop* push - align-code ; - M: x86 %dispatch-label ( word -- ) 0 cell, rc-absolute-cell rel-word ; @@ -105,6 +93,87 @@ M: x86 %shr-imm nip SHR ; M: x86 %sar-imm nip SAR ; M: x86 %not drop NOT ; +: ?MOV ( dst src -- ) + 2dup = [ 2drop ] [ MOV ] if ; inline + +:: move>args ( src1 src2 -- ) + { + { [ src1 param-reg-2 = ] [ param-reg-1 src2 ?MOV param-reg-1 param-reg-2 XCHG ] } + { [ src1 param-reg-1 = ] [ param-reg-2 src2 ?MOV ] } + { [ src2 param-reg-1 = ] [ param-reg-2 src1 ?MOV param-reg-1 param-reg-2 XCHG ] } + { [ src2 param-reg-2 = ] [ param-reg-1 src1 ?MOV ] } + [ + param-reg-1 src1 MOV + param-reg-2 src2 MOV + ] + } cond ; + +HOOK: %alien-invoke-tail cpu ( func dll -- ) + +:: overflow-template ( src1 src2 insn inverse func -- ) +