! Copyright (C) 2008 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: accessors alien alien.c-types alien.structs byte-arrays classes.struct kernel libc math parser sequences sequences.private words fry memoize compiler.units ; IN: struct-arrays : c-type-struct-class ( c-type -- class ) c-type boxed-class>> ; foldable TUPLE: struct-array { underlying c-ptr read-only } { length array-capacity read-only } { element-size array-capacity read-only } { class read-only } { ctor read-only } ; M: struct-array length length>> ; inline M: struct-array byte-length [ length>> ] [ element-size>> ] bi * ; inline : (nth-ptr) ( i struct-array -- alien ) [ element-size>> * >fixnum ] [ underlying>> ] bi ; inline M: struct-array nth-unsafe [ (nth-ptr) ] [ ctor>> ] bi execute( alien -- object ) ; inline M: struct-array set-nth-unsafe [ (nth-ptr) swap ] [ element-size>> ] bi memcpy ; inline : (struct-element-constructor) ( c-type -- word ) [ "struct-array-ctor" f [ swap dup struct-class? [ '[ _ memory>struct ] [ ] like ] [ drop [ ] ] if (( alien -- object )) define-inline ] keep ] with-compilation-unit ; ! Foldable memo word. This is an optimization; by precompiling a ! constructor for array elements, we avoid memory>struct's slow path. MEMO: struct-element-constructor ( c-type -- word ) (struct-element-constructor) ; foldable : ( alien length c-type -- struct-array ) [ heap-size ] [ c-type-struct-class ] [ struct-element-constructor ] tri struct-array boa ; inline M: struct-array new-sequence [ element-size>> * (byte-array) ] [ length>> ] [ class>> ] tri ; inline M: struct-array resize ( n seq -- newseq ) [ [ element-size>> * ] [ underlying>> ] bi resize ] [ class>> ] 2bi ; inline : ( length c-type -- struct-array ) [ heap-size * ] 2keep ; inline ERROR: bad-byte-array-length byte-array ; : byte-array>struct-array ( byte-array c-type -- struct-array ) [ heap-size [ dup length ] dip /mod 0 = [ drop bad-byte-array-length ] unless ] keep ; inline : struct-array-on ( struct length -- struct-array ) [ [ >c-ptr ] [ class ] bi ] dip swap ; inline : malloc-struct-array ( length c-type -- struct-array ) [ heap-size calloc ] 2keep ; inline INSTANCE: struct-array sequence M: struct-type ( len c-type -- array ) dup c-array-constructor [ execute( len -- array ) ] [ ] ?if ; inline M: struct-type ( alien len c-type -- array ) dup c-direct-array-constructor [ execute( alien len -- array ) ] [ ] ?if ; inline : >struct-array ( sequence class -- struct-array ) [ dup length ] dip [ 0 swap copy ] keep ; inline SYNTAX: struct-array{ \ } scan-word [ >struct-array ] curry parse-literal ; SYNTAX: struct-array@ scan-word [ scan-object scan-object ] dip parsed ; USING: vocabs vocabs.loader ; "prettyprint" vocab [ "struct-arrays.prettyprint" require ] when