-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsubstitute.fs
90 lines (78 loc) · 2.7 KB
/
substitute.fs
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
\ substitute stuff
require string.fs
wordlist AConstant macros-wordlist
: macro: ( addr u -- ) Create here 0 , $! DOES> [ here >r ] $@ ; r>
Constant macro-does:
: replaces ( addr1 len1 addr2 len2 -- )
\G create a macro with name @var{addr2 len2} and content @var{addr1 len1}.
\G if the macro already exists, just change the content.
2dup macros-wordlist search-wordlist
IF
nip nip dup >does-code macro-does: = IF >body $!
ELSE true [: .name ." is a hard-coded macro" cr ;] ?warning 2drop
THEN
ELSE
get-current >r macros-wordlist set-current
['] macro: execute-parsing
r> set-current
THEN ;
get-current macros-wordlist set-current
: rd ( -- addr u ) sourcefilename extractpath ;
set-current
: .% ( -- ) '%' emit ;
: .substitute ( addr1 len1 -- n / ior )
\G substitute all macros in text @var{addr1 len1} and print the
\G result. @var{n} is the number of substitutions or, if
\G negative, a throwable @var{ior}.
0 >r
BEGIN dup WHILE '%' $split
2swap type dup IF
over c@ '%' = IF
.% 1 /string
ELSE
'%' $split 2swap dup 0= IF
2drop .%
ELSE
2over drop 1- c@ '%' = IF
2dup macros-wordlist search-wordlist IF
nip nip -rot
2>r execute type 2r> r> 1+ >r
ELSE
.% type .%
THEN
ELSE
.% type
THEN
THEN
THEN
ELSE
over 1- c@ '%' = IF .% THEN
THEN
REPEAT 2drop r> ;
: $substitute ( addr1 len1 -- addr2 len2 n/ior )
\G substitute all macros in text @var{addr1 len1}. @var{n} is the
\G number of substitutions, if negative, it's a throwable @{ior},
\G @var{addr2 len2} the result.
['] .substitute $tmp rot ;
: substitute ( addr1 len1 addr2 len2 -- addr2 len3 n/ior )
\G substitute all macros in text @var{addr1 len1}, and copy the
\G result to @var{addr2 len2}. @var{n} is the number of
\G substitutions or, if negative, a throwable @var{ior},
\G @var{addr2 len3} the result.
2>r $substitute over r@ u<= -78 swap select -rot
2r> rot umin 2dup 2>r move 2r> rot -1 tmp$# +! ;
: unescape ( addr1 u1 dest -- dest u2 )
\G double all delimiters in @var{addr1 u1}, so that substitute
\G will result in the original text. Note that the buffer
\G @var{dest} does not have a size, as in worst case, it will need
\G just twice as many characters as @var{u1}. @{dest u2} is the
\G resulting string.
dp @ >r dup >r dp !
bounds ?DO
I c@ dup '%' = IF dup c, THEN c,
LOOP r> here over - r> dp ! ;
: $unescape ( addr1 u1 -- addr2 u2 )
[: bounds ?DO I c@ dup emit '%' = IF '%' emit THEN LOOP ;] $tmp ;
\ file name replacements in include and require
: subst>filename ['] .substitute $tmp rot 0 min throw ;
' subst>filename is >include