-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathlib-stackrng.muf
194 lines (170 loc) · 5.23 KB
/
lib-stackrng.muf
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
@program lib-stackrng
1 99999 d
1 i
( ***** Stack based range handling object -- SRNG ****
offset is how many stack items are between range and parms
pos is the position within the range you wish to deal with.
num is the number of range items to deal with.
A 'range' is defines as a set of related items on the stack with an
integer 'count' of them on the top. ie: "bat" "cat" "dog" 3
extractrng[ {rng} ... offset num pos -- {rng'} ... {subrng} ]
pulls a subrange out of a range buried in the stack, removing them.
copyrng [ {rng} ... offset num pos -- {rng} ... {subrng} ]
copies a subrange out of a range buried in the stack.
deleterng [ {rng} ... offset num pos -- {rng'} ]
deletes a subrange from a range buried on the stack.
insertrng [ {rng1} ... {rng2} offset pos -- {rng} ]
inserts a subrange into the middle of a buried range on the stack.
filterrng [ {rng} funcaddr -- {rng'} {filtrdrng} ]
Takes the given range and tests each item with the given filter
function address. The function takes a single data value and
returns an integer. If the integer is non-zero, it pulls that
data item out of the range and puts it into the filtered range.
The data items can be of any type.
catrng [ {rng1} {rng2} -- {rng} ]
concatenates two ranges into one range.
swaprng [ {rng1} {rng2} -- {rng2} {rng1} ]
takes two ranges on the stack and swaps them.
shortlist [ {rng} -- s ]
converts a range into a string list, with "and" and commas.
)
$doccmd @list __PROG__=!@1-32
: catranges ( {rng1} {rng2} -- {rng} )
dup 2 + rotate +
;
: copyrange ( {rng} ... offset num pos -- {rng} ... {subrng} )
1 - var! pos
var! num
array_make var! stuff
array_make var! range
var subrng
(avoid rangecheck errors to duplicate pre FB6 behaviour.)
num @ 0 <= range @ array_count pos @ < or if
{ }list subrng !
else
(limit operations to the actual size of the range [pre FB6 compat])
pos @ num @ + range @ array_count - dup 0 > if
num @ swap - num !
else
pop
then
range @ pos @ dup num @ + 1 -
array_getrange subrng !
then
range @ array_vals
stuff @ array_vals pop
subrng @ array_vals
;
: extractrange ( {rng} ... offset num pos -- {rng'} ... {subrng} )
1 - var! pos
var! num
array_make var! stuff
array_make var! range
var subrng
(limit operations to the actual size of the range [pre FB6 compat])
pos @ num @ + range @ array_count - dup 0 > if
num @ swap - num !
else
pop
then
(avoid rangecheck errors to duplicate pre FB6 behaviour.)
num @ 0 <= range @ array_count pos @ < or if
{ }list subrng !
else
range @ pos @ dup num @ + 1 -
array_getrange subrng !
range @ pos @ dup num @ + 1 -
array_delrange range !
then
range @ array_vals
stuff @ array_vals pop
subrng @ array_vals
;
: swapranges ( {rng1} {rng2} -- {rng2} {rng1} )
array_make var! tmp
array_make var! tmp2
tmp @ array_vals
tmp2 @ array_vals
;
: deleterange ( {rng} ... offset num pos -- {rng'} )
extractrange popn
;
: insertrange ( {rng1} ... {rng2} offset pos -- {rng} ... )
1 - var! pos
var! offset
array_make var! newrng
offset @ array_make var! stuff
array_make var! range
(limit operations to the actual size of the range [pre FB6 compat])
pos @ range @ array_count - dup 0 > if
pos @ swap - pos !
else
pop
then
range @ pos @ newrng @ array_insertrange
array_vals
stuff @ array_vals pop
;
: filterrange ( {rng} funcaddr -- {rng'} {filtrdrng} )
var! cb
{ }list var! outrng
array_make var! range
range @ foreach
dup cb @ execute if
outrng @ dup array_count array_setitem outrng !
range @ swap array_delitem range !
else
pop
then
repeat
range @ array_vals
outrng @ array_vals
;
: shortlist ( {rng} -- s )
dup not if pop "" exit then
dup 3 <
if
1 - dup 2 + rotate name over
if
" " strcat
then
else
""
begin
over 1 >
while
swap 1 - swap over 3 + rotate name ", " strcat strcat
repeat
then
swap
if
"and " strcat swap name strcat
then
;
public catranges
public extractrange
public swapranges
public copyrange
public deleterange
public insertrange
public filterrange
public shortlist
$pubdef sr-catrng __PROG__ "catranges" call
$pubdef sr-copyrng __PROG__ "copyrange" call
$pubdef sr-deleterng __PROG__ "deleterange" call
$pubdef sr-extractrng __PROG__ "extractrange" call
$pubdef sr-filterrng __PROG__ "filterrange" call
$pubdef sr-insertrng __PROG__ "insertrange" call
$pubdef sr-swaprng __PROG__ "swapranges" call
$pubdef sr-shortlist __PROG__ "shortlist" call
.
c
q
@register lib-stackrng=lib/stackrng
@register #me lib-stackrng=tmp/prog1
@set $tmp/prog1=3
@set $tmp/prog1=H
@set $tmp/prog1=L
@set $tmp/prog1=S
@set $tmp/prog1=V
@register #me =tmp