-
Notifications
You must be signed in to change notification settings - Fork 30
/
Copy pathchains.mu4
265 lines (224 loc) · 10.6 KB
/
chains.mu4
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
254
255
256
257
258
259
260
261
262
263
264
265
| This file is part of muforth: https://muforth.dev/
|
| Copyright 2002-2025 David Frech. (Read the LICENSE for details.)
loading MSP430 meta-compiler (chains and token consumers)
forth decimal
( Metacompiler vocabulary chains.)
| Since we are now "chaining" chains together, let's do this bottom-up.
|
| .equates. contains chip equates and any other constants that should be
| visible to both the assembler and target compiler. When used interactively,
| or in the assembler, these push their value. When used in a target colon
| word, these compile literals.
|
| .meta. is where meta-compiler specific words go - esp if they would
| otherwise shadow a host word of the same name. @ is a good example - the
| meta version writes into the target image, using a local, image-relative
| address. .meta. contains special versions of : to compile macros (though
| I'm thinking of deprecating or removing this) and to define new defining
| words during target compilation. It will also, once the kernel starts
| loading, contain a : that compiles target colon words!
|
| Labels - like temporary locations defined in assembler that do not
| represent target words - are in .meta., and are invisible to the target
| compiler.
|
| Also in .meta. are a handful of variables that refer to key pieces of
| target infrastructure that won't get defined until the kernel is loaded.
| Having these as forward-reference variables allows us to write the
| metacompiler using variable references; compilation of the kernel will
| patch these with actual addresses of target code.
|
| .meta. is chained to .labels., which is chained to .equates., so
| searching .meta. will also search .labels. and .equates.
|
| .opcodes2. contains an "alternative" set of opcodes whose names end with
| a comma. This is a form preferred by some.
|
| .assembler. contains definitions of instructions, registers, addressing
| modes, and words to build control structures. It is searched when
| building code words, and also when compiling macros. It is chained to
| .opcodes2., so searching .assembler. will also search .opcodes2.
|
| .lex. contains words for delimiting comments and for doing conditional
| interpretation and compilation. They are in a separate chain so we don't
| have to keep making synonyms; instead, we just search the .lex. chain.
|
| .definer. is slightly exotic. It's like .compiler. but it's specifically
| for making new defining words for the meta-compiler. It contains special
| definitions of words like does> and perhaps others - eg, ; [ etc. It
| is chained to .compiler. so searching .definer. will search both. Since
| .compiler. contains all of the conditional compiler words we do not have
| to also search .lex.
|
| .target-compiler. is like .compiler. but for the -target- colon compiler.
| This will contain target versions of [ ; if then begin while etc. It is
| linked to .lex. so we get the conditional compiler words, but is
| otherwise kept "pure". We don't want to get any host words by accident.
|
| .target. contains the actual target words. .target-runtime. - which is
| chained to .target. - contains target words that are compile-only - ie,
| never executed "at the command line" while chatting. By putting stack
| manipulation words on this chain as well, we make it easier to do
| debugging: you'll get the host's swap or dup rather than the target's.
|
|
| Simple, right? ;-)
sealed .equates. ( chip equates and other constants for target)
.equates. chain .labels. ( labels chains to equates)
.labels. chain .meta. ( the `meta' version of .forth.; chains to .labels.)
sealed .opcodes2. ( the alternative opcode set)
.opcodes2. chain .assembler. ( the host-resident target assembler)
.compiler. chain .definer. ( like .compiler. but only for meta defining words)
sealed .lex. ( comments and conditional interpretation)
sealed .target. ( the target words - like `forth' on the host)
.target. chain .target-runtime. ( compile-only target words)
.lex. chain .target-compiler. ( the `meta' version of .compiler.)
: meta .meta. definitions ;
: definer .definer. definitions ;
: assembler .assembler. definitions ;
: target .target. definitions ;
: target-runtime .target-runtime. definitions ;
: target-compiler .target-compiler. definitions ;
( XXX Do we need all of these?)
compiler
: \l ( compile from labels) .labels. \chain ;
: \m ( compile from meta) .meta. \chain ;
: \d ( compile from definer) .definer. \chain ;
: \a ( compile from assembler) .assembler. \chain ;
: \eq ( compile from equates) .equates. \chain ;
: \t ( compile from target/runtime) .target-runtime. \chain ;
: \tc ( compile from target-compiler) .target-compiler. \chain ;
forth
meta
: \f ' execute ; ( execute a forth word from meta)
forth
: \l .labels. chain' execute ; ( execute a label word from forth)
: \m .meta. chain' execute ; ( execute a meta word from forth)
: \eq .equates. chain' execute ; ( execute an equates word from forth)
.lex. definitions
( Comments are nice to have!)
: ( \ ( ;
: | \ | ;
( And conditional intepretation is nice to have too.)
: .if \ .if ;
: .else \ .else ;
: .then \ .then ;
: .def \ .def ;
: .ndef \ .ndef ;
: .ifdef \ .ifdef ;
: .ifndef \ .ifndef ;
: .contains \ .contains ;
( For combining conditional tests.)
: .or .or ;
: .and .and ;
: .not .not ;
forth
| Given a constant value, find the host dictionary entry, if one exists,
| that contains a constant that matches it. It does *not* search through
| muchain connections; if it sees a muchain, it returns "not found".
|
| This is useful for printing out SFRs, CSRs, equates, labels, and such.
: find-constant ( value chain - 0 | 'link -1)
begin
@ =while
dup muchain? if 2drop 0 ^ then ( not found)
dup hidden? 0= if
2dup link> >body @ = if ( found it!) nip -1 ^ then
then
repeat 2drop 0 ( not found) ;
| Similar to find-constant, but searches *through* muchain links, rather
| than stopping when it sees a muchain.
: find-constant-chained ( value chain - 0 | 'link -1)
begin
@ =while
dup hidden? 0= if ( skip muchain and hidden entries)
2dup link> >body @ = if ( found it!) nip -1 ^ then
then
repeat 2drop 0 ( not found) ;
| Metacompiler token consumers. Let's put them all in one place so we can
| understand how they work in relation to each other.
| NOTE: While in general chaining vocab chains together works well, with
| the assembler we have to be careful. Because there are structure words -
| if/then/else, begin/until etc - in both the .assembler. and .compiler. chains,
| and because we definitely want to find the assembler's versions *first*, we
| have to put .assembler. before .definer./.compiler. here.
|
| But there is a subtle problem. What if we decide, in some inline assembler
| code, that we want to specify the use of a .forth. word, say, swap.
| If we use \f - hoping to get the version from .compiler. - and if
| .assembler. is chained to .meta. - we will fail. While searching
| .assembler. for \f we will chain into .meta. and find its \f, which will
| compile itself rather than the following word.
|
| The easiest solution is simply to unchain .assembler. from .meta., and
| search .meta. explicitly. We have to do that both here and in __asm.
-: ." (compiling inline assembler)" ;
-:
.assembler. find if compile, ^ then ( find assembler's if/then etc)
.definer. find if execute ^ then ( need } and ; to exit this mode)
.meta. find if compile, ^ then ( labels are in .meta.)
.runtime. find if compile, ^ then ( utility and runtime words)
number literal ;
mode __inline-asm
-: ." (assembling)" ;
-:
.assembler. find if execute ^ then
.meta. find if execute ^ then ( labels are in .meta.)
.forth. find if execute ^ then ( utility words in .forth.)
number ;
mode __asm
| There are two slightly different "meta" modes we can be in. When
| _building_ the target image, we are in __building. When interacting -
| after connecting via chat - we are in __chatting. Since all the compilation
| words want to return to __meta when they are done - eg, this is what the
| meta ; and ;c do - let's just defer what __meta means.
defer __meta ( will be either __building or __chatting)
| The build meta-interpreter. We're in this mode when we're building the
| target image, and when in between [ and ] when running the target colon
| compiler.
-: ." (building)" ;
-:
.meta. find if execute ^ then ( labels are in .meta.)
.forth. find if execute ^ then
number ;
mode __building
now __building is __meta
| Interacting with a chatty, connected target. This differs from __building
| in two ways:
|
| * we actually try to execute target words remotely
| * we potentially convert numbers in a target-specific way
|
| Both __building and __chatting try, as a last resort, to find and execute
| words in .forth.
defer target-number ( convert token to a number, specific to target)
defer remote
-: error" tried to remotely execute a target word while not chatting" ;
is remote
-: ." (chatting)" ;
-:
( NOTE: if we want to remove .forth., replace it with .lex. .)
.target. find if execute remote ^ then ( execute on target)
.meta. find if execute ^ then ( labels are in .meta.)
.forth. find if execute ^ then
target-number ;
mode __chatting
| __definer-colon is for compiling new defining words that are part of the
| meta-compiler. __definer-colon is to __meta as __inline-asm is to __asm.
-: ." (compiling a meta defining word)" ;
-:
.definer. find if execute ^ then ( does>, ;code, special ;)
.meta. find if compile, ^ then
.runtime. find if compile, ^ then ( utility and runtime words)
number literal ;
mode __definer-colon
defer target-literal ( compile a target literal)
defer target-compile, ( compile a target word into a colon definition)
-: ." (compiling a target colon word)" ;
-:
.target-compiler. find if execute ^ then
.target-runtime. find if execute target-compile, ^ then
.equates. find if execute target-literal ^ then
target-number target-literal ;
mode __target-colon