-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathclosures.fs
More file actions
253 lines (207 loc) · 8.66 KB
/
closures.fs
File metadata and controls
253 lines (207 loc) · 8.66 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
\ A powerful closure implementation
\ Authors: Bernd Paysan, Anton Ertl
\ Copyright (C) 2018,2019,2020,2021,2022,2023 Free Software Foundation, Inc.
\ This file is part of Gforth.
\ Gforth is free software; you can redistribute it and/or
\ modify it under the terms of the GNU General Public License
\ as published by the Free Software Foundation, either version 3
\ of the License, or (at your option) any later version.
\ This program is distributed in the hope that it will be useful,
\ but WITHOUT ANY WARRANTY; without even the implied warranty of
\ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
\ GNU General Public License for more details.
\ You should have received a copy of the GNU General Public License
\ along with this program. If not, see http://www.gnu.org/licenses/.
\ more information in http://www.complang.tuwien.ac.at/anton/euroforth/ef18/drafts/ertl.pdf
$10 stack: locals-sizes
$10 stack: locals-lists
Defer end-d ( ... xt -- ... )
\ is either EXECUTE (for {: ... :}*) or END-DCLOSURE (for [{: ... :}*).
\ xt is either ' NOOP or [: ]] r> lp! [[ ;], which restores LP.
' execute is end-d
Defer endref, ( -- )
\ pushes a reference to the location
' noop is endref,
false Value 1t-closure?
-2 cells field: >addr ( xt -- addr ) \ gforth-experimental to-addr
\G convert the xt of a closure on the heap to the @var{addr} with can be
\G passed to @code{free} to get rid of the closure
drop
: alloch ( size -- addr )
\ addr is the end of the allocated region
dup allocate throw + ;
: allocd ( size -- addr )
\ addr is the end of the allocated region
dp +! dp @ ;
: >lp ( addr -- r:oldlp ) r> lp@ >r >r lp! ;
opt: drop ]] lp@ >r lp! [[ ;
: lp> ( r:oldlp -- ) r> r> lp! >r ;
opt: drop ]] r> lp! [[ ;
Variable extra-locals ( additional hidden locals size )
locals-types definitions
: :}* ( hmaddr u latest latestnt wid 0 a-addr1 u1 ... xt -- ) \ gforth-internal colon-close-brace-star
0 lit, lits, here cell- >r
compile, ]] >lp [[
:}
locals-size @ extra-locals @ + r> !
['] endref, end-d
['] execute is end-d ['] noop is endref,
extra-locals off activate-locals ;
: :}xt ( hmaddr u latest latestnt wid 0 a-addr1 u1 ... -- ) \ gforth colon-close-brace-x-t
\G end a closure's locals declaration. The closure will be allocated by
\G the xt on the stack, so the closure's run-time stack effect is @code{(
\G xt-alloc -- xt-closure )}.
\ run-time: ( xt size -- ... )
[: swap execute ;] :}* ;
: :}d ( hmaddr u latest latestnt wid 0 a-addr1 u1 ... -- ) \ gforth colon-close-brace-d
\G end a closure's locals declaration. The closure will be allocated in
\G the dictionary.
['] allocd :}* ;
: :}h ( hmaddr u latest latestnt wid 0 a-addr1 u1 ... -- ) \ gforth colon-close-brace-h
\G end a closure's locals declaration. The closure will be allocated on
\G the heap.
['] alloch :}* ;
: :}h1 ( hmaddr u latest latestnt wid 0 a-addr1 u1 ... -- ) \ gforth colon-close-brace-h
\G end a closure's locals declaration. The closure will be allocated on
\G the heap.
true to 1t-closure? ['] alloch :}* ;
forth definitions
: push-locals ( list size -- )
locals-size @ locals-sizes >stack locals-size !
locals-list @ locals-lists >stack locals-list ! ;
: pop-locals ( -- )
locals-lists stack> locals-list !
locals-sizes stack> locals-size ! ;
: dummy-local, ( n -- )
locals-size +!
get-current >r 0 warnings !@ >r [ ' locals >wordlist ]l set-current
s" " nextname create-local locals-size @ locals,
r> warnings ! r> set-current ;
locals-types definitions
: :}l ( hmaddr u latest latestnt wid 0 a-addr1 u1 ... -- ) \ gforth close-brace-locals
\G end a closure's locals declaration. The closure will be allocated on
\G the local's stack.
:}
locals-size @ locals-list @ over 2>r pop-locals
[ 2 cells maxaligned ]L + dummy-local,
2r> push-locals
['] noop end-d
activate-locals ;
forth definitions
: wrap-closure ( xt -- )
dup >extra ! ['] does, set-optimizer
flush-code hm, wrap! hmtemplate off \ dead hmtemplate link
previous-section dead-code off ;
: (closure-;]) ( closure-sys lastxt -- )
dup >r wrap-closure
r> >namehm @ swap !
pop-locals ;
[IFUNDEF] in-colon-def?
0 Value in-colon-def?
[THEN]
: closure-:-hook ( sys -- sys addr xt n )
\ addr is the nfa of the defined word, xt its xt
:-hook1
['] here locals-headers latest latestnt
clear-leave-stack
dead-code off
defstart
true to in-colon-def? ;
: free-closure ( xt -- ) \ gforth-internal
\G free a heap-allocated closure
>addr free throw ;
: closure> ( hmaddr -- addr ) \ gforth-internal closure-end
\G create trampoline head
[ 0 >body ] [IF] dodoes: >l >l lp@ cell+
[ELSE] >l dodoes: >l lp@ cell+ cell+ [THEN] ;
: end-dclosure ( unravel-xt -- closure-sys )
>r
postpone lit here 0 ,
]] closure> [[ r> execute
wrap@ next-section
action-of :-hook >r ['] closure-:-hook is :-hook
:noname
r> is :-hook
1t-closure? IF ]] dup [[ THEN
case locals-size @ \ special optimizations for few locals
cell of ]] @ >l [[ endof
2 cells of ]] 2@ 2>l [[ endof
dup negate ]] literal lp+! lp@ [[ dup ]] literal move [[
endcase
1t-closure? IF ]] free-closure [[ THEN
false to 1t-closure?
['] (closure-;]) colon-sys-xt-offset stick ;
: [{: ( -- hmaddr u latest latestnt wid 0 ) \ gforth-experimental start-closure
\G starts a closure. Closures first declare the locals frame they are
\G going to use, and then the code that is executed with those locals.
\G Closures end like quotations with a @code{;]}. The locals declaration
\G ends depending where the closure's locals are created. At run-time, the
\G closure is created as trampolin xt, and fills the values of its local
\G frame from the stack. At execution time of the xt, the local frame is
\G copied to the locals stack, and used inside the closure's code. After
\G return, those values are removed from the locals stack, and not updated
\G in the closure itself.
[: ] drop ;] defstart
#0. push-locals
['] end-dclosure is end-d [: ]] lp> [[ ;] is endref,
[ 2 cells maxaligned ]L extra-locals !
postpone {:
; immediate compile-only
: <{: ( -- hmaddr u latest latestnt wid 0 ) \ gforth-experimental start-homelocation
\G starts a home location
#0. push-locals postpone {:
; immediate compile-only
: ;> ( -- ) \ gforth-experimental end-homelocation
\G end using a home location
pop-locals ]] lp@ lp> [[
; immediate compile-only
\ stack-based closures without name
: (;]*) ( xt -- hm )
>r ] ]] UNREACHABLE ENDSCOPE [[
r@ wrap-closure r> >namehm @ ;
: (;]l) ( xt1 n xt2 -- )
(;]*) >r dummy-local,
compile, r> lit, ]] closure> [[ ;
: alloc-by-xt, ( xt n -- )
lit, swap compile, ]] >lp [[ ;
: (;]xt) ( xt0 xt1 n xt2 -- )
(;]*) >r alloc-by-xt,
compile, r> lit, ]] closure> lp> [[ ;
: :l ( -- xt ) ['] (;]l) ; immediate restrict
: :h ( -- xt1 xt2 ) ['] alloch ['] (;]xt) ; immediate restrict
: :d ( -- xt1 xt2 ) ['] allocd ['] (;]xt) ; immediate restrict
: [*:: [{: xt@ xt>l size :}d
>r xt>l size [ 2 cells ]L + maxaligned
postpone [: xt@ compile,
r> [ colon-sys-xt-offset 2 + ]L stick ;]
alias immediate restrict ;
cell 4 = [IF] :noname ( n -- xt ) false >l >l ; [ELSE] ' >l [THEN]
' @ swap 1 cells [*:: [n: ( xt -- colon-sys )
' 2@ ' 2>l 2 cells [*:: [d: ( xt -- colon-sys )
' f@ ' f>l 1 floats [*:: [f: ( xt -- colon-sys )
\ combined names (used in existing code)
: [n:l ( -- colon-sys ) ]] :l [n: [[ ; immediate restrict
: [d:l ( -- colon-sys ) ]] :l [d: [[ ; immediate restrict
: [f:l ( -- colon-sys ) ]] :l [f: [[ ; immediate restrict
: [n:d ( -- colon-sys ) ]] :d [n: [[ ; immediate restrict
: [d:d ( -- colon-sys ) ]] :d [d: [[ ; immediate restrict
: [f:d ( -- colon-sys ) ]] :d [f: [[ ; immediate restrict
: [n:h ( -- colon-sys ) ]] :h [n: [[ ; immediate restrict
: [d:h ( -- colon-sys ) ]] :h [d: [[ ; immediate restrict
: [f:h ( -- colon-sys ) ]] :h [f: [[ ; immediate restrict
[IFDEF] test-it
: foo [{: a f: b d: c xt: d :}d a . b f. c d. d ;] ;
5 3.3e #1234. ' cr foo execute
: homeloc <{: w^ a w^ b w^ c :}h a b c ;> ;
1 2 3 homeloc >r ? ? ? r> free throw cr
: A {: w^ k x1 x2 x3 xt: x4 xt: x5 | w^ B :} recursive
k @ 0<= IF x4 x5 + ELSE
B k x1 x2 x3 action-of x4 [{: B k x1 x2 x3 x4 :}L
-1 k +!
k @ B @ x1 x2 x3 x4 A ;] dup B !
execute THEN ;
: man-or-boy? ( n -- n' ) [: 1 ;] [: -1 ;] 2dup swap [: 0 ;] A ;
\ start with: gforth -l64M -r8M closures.fs
\ start with: gforth-fast -l6G -r768M closures.fs if you want to go up to 26
20 0 [DO] [i] dup . !time man-or-boy? . .time cr [LOOP]
[THEN]