https://github.com/akkartik/mu/blob/master/apps/tile/data.mu
1 type sandbox {
2 setup: (handle line)
3 data: (handle line)
4
5 cursor-call-path: (handle call-path-element)
6 expanded-words: (handle call-path)
7 partial-name-for-cursor-word: (handle word)
8 partial-name-for-function: (handle word)
9
10 next: (handle sandbox)
11 prev: (handle sandbox)
12 }
13
14 type function {
15 name: (handle array byte)
16 args: (handle word)
17 body: (handle line)
18
19 next: (handle function)
20 }
21
22 type line {
23 name: (handle array byte)
24 data: (handle word)
25 result: (handle result)
26 next: (handle line)
27 prev: (handle line)
28 }
29
30 type word {
31 scalar-data: (handle gap-buffer)
32 next: (handle word)
33 prev: (handle word)
34 }
35
36
37 type value {
38 type: int
39 int-data: int
40 text-data: (handle array byte)
41 array-data: (handle array value)
42 file-data: (handle buffered-file)
43 filename: (handle array byte)
44 screen-data: (handle screen)
45 }
46
47 type table {
48 data: (handle array bind)
49 next: (handle table)
50 }
51
52 type bind {
53 key: (handle array byte)
54 value: (handle value)
55 }
56
57
58
59 type call-path {
60 data: (handle call-path-element)
61 next: (handle call-path)
62 }
63
64
65 type call-path-element {
66 word: (handle word)
67 next: (handle call-path-element)
68 }
69
70 type result {
71 data: value-stack
72 error: (handle array byte)
73 }
74
75 fn initialize-sandbox _sandbox: (addr sandbox) {
76 var sandbox/esi: (addr sandbox) <- copy _sandbox
77 var line-ah/eax: (addr handle line) <- get sandbox, data
78 allocate line-ah
79 var line/eax: (addr line) <- lookup *line-ah
80 initialize-line line
81 var word-ah/ecx: (addr handle word) <- get line, data
82 var cursor-call-path-ah/eax: (addr handle call-path-element) <- get sandbox, cursor-call-path
83 allocate cursor-call-path-ah
84 var cursor-call-path/eax: (addr call-path-element) <- lookup *cursor-call-path-ah
85 var dest/eax: (addr handle word) <- get cursor-call-path, word
86 copy-object word-ah, dest
87 }
88
89
90 fn initialize-line _line: (addr line) {
91 var line/esi: (addr line) <- copy _line
92 var word-ah/eax: (addr handle word) <- get line, data
93 allocate word-ah
94 var word/eax: (addr word) <- lookup *word-ah
95 initialize-word word
96 }
97
98 fn create-primitive-functions _self: (addr handle function) {
99
100 var self/esi: (addr handle function) <- copy _self
101 allocate self
102 var _f/eax: (addr function) <- lookup *self
103 var f/esi: (addr function) <- copy _f
104 var name-ah/eax: (addr handle array byte) <- get f, name
105 populate-text-with name-ah, "2*"
106 var args-ah/eax: (addr handle word) <- get f, args
107 allocate args-ah
108 var args/eax: (addr word) <- lookup *args-ah
109 initialize-word-with args, "x"
110 var body-ah/eax: (addr handle line) <- get f, body
111 allocate body-ah
112 var body/eax: (addr line) <- lookup *body-ah
113 initialize-line body
114 var curr-word-ah/ecx: (addr handle word) <- get body, data
115
116 allocate curr-word-ah
117 var tmp/eax: (addr word) <- lookup *curr-word-ah
118 var curr-word/edx: (addr word) <- copy tmp
119 initialize-word-with curr-word, "x"
120
121 var next-word-ah/ebx: (addr handle word) <- get curr-word, next
122 allocate next-word-ah
123 tmp <- lookup *next-word-ah
124 initialize-word-with tmp, "2"
125
126 var prev-word-ah/edi: (addr handle word) <- get tmp, prev
127 copy-object curr-word-ah, prev-word-ah
128
129 curr-word-ah <- copy next-word-ah
130 curr-word <- copy tmp
131
132 next-word-ah <- get curr-word, next
133 allocate next-word-ah
134 tmp <- lookup *next-word-ah
135 initialize-word-with tmp, "*"
136
137 prev-word-ah <- get tmp, prev
138 copy-object curr-word-ah, prev-word-ah
139 tmp <- lookup *prev-word-ah
140
141 var next/esi: (addr handle function) <- get f, next
142 allocate next
143 var _f/eax: (addr function) <- lookup *next
144 var f/esi: (addr function) <- copy _f
145 var name-ah/eax: (addr handle array byte) <- get f, name
146 populate-text-with name-ah, "1+"
147 var args-ah/eax: (addr handle word) <- get f, args
148 allocate args-ah
149 var args/eax: (addr word) <- lookup *args-ah
150 initialize-word-with args, "x"
151 var body-ah/eax: (addr handle line) <- get f, body
152 allocate body-ah
153 var body/eax: (addr line) <- lookup *body-ah
154 initialize-line body
155 var curr-word-ah/ecx: (addr handle word) <- get body, data
156
157 allocate curr-word-ah
158 var tmp/eax: (addr word) <- lookup *curr-word-ah
159 curr-word <- copy tmp
160 initialize-word-with curr-word, "x"
161
162 next-word-ah <- get curr-word, next
163 allocate next-word-ah
164 tmp <- lookup *next-word-ah
165 initialize-word-with tmp, "1"
166
167 prev-word-ah <- get tmp, prev
168 copy-object curr-word-ah, prev-word-ah
169
170 curr-word-ah <- copy next-word-ah
171 curr-word <- copy tmp
172
173 next-word-ah <- get curr-word, next
174 allocate next-word-ah
175 tmp <- lookup *next-word-ah
176 initialize-word-with tmp, "+"
177
178 prev-word-ah <- get tmp, prev
179 copy-object curr-word-ah, prev-word-ah
180 tmp <- lookup *prev-word-ah
181
182 var next/esi: (addr handle function) <- get f, next
183 allocate next
184 var _f/eax: (addr function) <- lookup *next
185 var f/esi: (addr function) <- copy _f
186 var name-ah/eax: (addr handle array byte) <- get f, name
187 populate-text-with name-ah, "2+"
188 var args-ah/eax: (addr handle word) <- get f, args
189 allocate args-ah
190 var args/eax: (addr word) <- lookup *args-ah
191 initialize-word-with args, "x"
192 var body-ah/eax: (addr handle line) <- get f, body
193 allocate body-ah
194 var body/eax: (addr line) <- lookup *body-ah
195 initialize-line body
196 var curr-word-ah/ecx: (addr handle word) <- get body, data
197
198 allocate curr-word-ah
199 var tmp/eax: (addr word) <- lookup *curr-word-ah
200 curr-word <- copy tmp
201 initialize-word-with curr-word, "x"
202
203 next-word-ah <- get curr-word, next
204 allocate next-word-ah
205 tmp <- lookup *next-word-ah
206 initialize-word-with tmp, "1+"
207
208 prev-word-ah <- get tmp, prev
209 copy-object curr-word-ah, prev-word-ah
210
211 curr-word-ah <- copy next-word-ah
212 curr-word <- copy tmp
213
214 next-word-ah <- get curr-word, next
215 allocate next-word-ah
216 tmp <- lookup *next-word-ah
217 initialize-word-with tmp, "1+"
218
219 prev-word-ah <- get tmp, prev
220 copy-object curr-word-ah, prev-word-ah
221 tmp <- lookup *prev-word-ah
222
223 var next/esi: (addr handle function) <- get f, next
224 allocate next
225 var _f/eax: (addr function) <- lookup *next
226 var f/esi: (addr function) <- copy _f
227 var name-ah/eax: (addr handle array byte) <- get f, name
228 populate-text-with name-ah, "square"
229 var args-ah/eax: (addr handle word) <- get f, args
230 allocate args-ah
231 var args/eax: (addr word) <- lookup *args-ah
232 initialize-word-with args, "x"
233 var body-ah/eax: (addr handle line) <- get f, body
234 allocate body-ah
235 var body/eax: (addr line) <- lookup *body-ah
236 initialize-line body
237 var curr-word-ah/ecx: (addr handle word) <- get body, data
238
239 allocate curr-word-ah
240 var tmp/eax: (addr word) <- lookup *curr-word-ah
241 var curr-word/edx: (addr word) <- copy tmp
242 initialize-word-with curr-word, "x"
243
244 var next-word-ah/ebx: (addr handle word) <- get curr-word, next
245 allocate next-word-ah
246 tmp <- lookup *next-word-ah
247 initialize-word-with tmp, "x"
248
249 var prev-word-ah/edi: (addr handle word) <- get tmp, prev
250 copy-object curr-word-ah, prev-word-ah
251
252 curr-word-ah <- copy next-word-ah
253 curr-word <- copy tmp
254
255 next-word-ah <- get curr-word, next
256 allocate next-word-ah
257 tmp <- lookup *next-word-ah
258 initialize-word-with tmp, "*"
259
260 prev-word-ah <- get tmp, prev
261 copy-object curr-word-ah, prev-word-ah
262 tmp <- lookup *prev-word-ah
263
264 var next/esi: (addr handle function) <- get f, next
265 allocate next
266 var _f/eax: (addr function) <- lookup *next
267 var f/esi: (addr function) <- copy _f
268 var name-ah/eax: (addr handle array byte) <- get f, name
269 populate-text-with name-ah, "1-"
270 var args-ah/eax: (addr handle word) <- get f, args
271 allocate args-ah
272 var args/eax: (addr word) <- lookup *args-ah
273 initialize-word-with args, "x"
274 var body-ah/eax: (addr handle line) <- get f, body
275 allocate body-ah
276 var body/eax: (addr line) <- lookup *body-ah
277 initialize-line body
278 var curr-word-ah/ecx: (addr handle word) <- get body, data
279
280 allocate curr-word-ah
281 var tmp/eax: (addr word) <- lookup *curr-word-ah
282 curr-word <- copy tmp
283 initialize-word-with curr-word, "x"
284
285 next-word-ah <- get curr-word, next
286 allocate next-word-ah
287 tmp <- lookup *next-word-ah
288 initialize-word-with tmp, "1"
289
290 prev-word-ah <- get tmp, prev
291 copy-object curr-word-ah, prev-word-ah
292
293 curr-word-ah <- copy next-word-ah
294 curr-word <- copy tmp
295
296 next-word-ah <- get curr-word, next
297 allocate next-word-ah
298 tmp <- lookup *next-word-ah
299 initialize-word-with tmp, "-"
300
301 prev-word-ah <- get tmp, prev
302 copy-object curr-word-ah, prev-word-ah
303 tmp <- lookup *prev-word-ah
304
305 var next/esi: (addr handle function) <- get f, next
306 allocate next
307 var _f/eax: (addr function) <- lookup *next
308 var f/esi: (addr function) <- copy _f
309 var name-ah/eax: (addr handle array byte) <- get f, name
310 populate-text-with name-ah, "sub"
311
312 var args-ah/eax: (addr handle word) <- get f, args
313 allocate args-ah
314 var args/eax: (addr word) <- lookup *args-ah
315 initialize-word-with args, "y"
316 var next-arg-ah/eax: (addr handle word) <- get args, next
317 allocate next-arg-ah
318 var next-arg/eax: (addr word) <- lookup *next-arg-ah
319 initialize-word-with next-arg, "x"
320 var body-ah/eax: (addr handle line) <- get f, body
321 allocate body-ah
322 var body/eax: (addr line) <- lookup *body-ah
323 initialize-line body
324 var curr-word-ah/ecx: (addr handle word) <- get body, data
325
326 allocate curr-word-ah
327 var tmp/eax: (addr word) <- lookup *curr-word-ah
328 curr-word <- copy tmp
329 initialize-word-with curr-word, "x"
330
331 next-word-ah <- get curr-word, next
332 allocate next-word-ah
333 tmp <- lookup *next-word-ah
334 initialize-word-with tmp, "y"
335
336 prev-word-ah <- get tmp, prev
337 copy-object curr-word-ah, prev-word-ah
338
339 curr-word-ah <- copy next-word-ah
340 curr-word <- copy tmp
341
342 next-word-ah <- get curr-word, next
343 allocate next-word-ah
344 tmp <- lookup *next-word-ah
345 initialize-word-with tmp, "-"
346
347 prev-word-ah <- get tmp, prev
348 copy-object curr-word-ah, prev-word-ah
349 tmp <- lookup *prev-word-ah
350 }
351
352 fn function-body functions: (addr handle function), _word: (addr handle word), out: (addr handle line) {
353 var function-name-storage: (handle array byte)
354 var function-name-ah/ecx: (addr handle array byte) <- address function-name-storage
355 var word-ah/esi: (addr handle word) <- copy _word
356 var word/eax: (addr word) <- lookup *word-ah
357 var gap-ah/eax: (addr handle gap-buffer) <- get word, scalar-data
358 var gap/eax: (addr gap-buffer) <- lookup *gap-ah
359 gap-buffer-to-string gap, function-name-ah
360 var _function-name/eax: (addr array byte) <- lookup *function-name-ah
361 var function-name/esi: (addr array byte) <- copy _function-name
362 var curr-ah/ecx: (addr handle function) <- copy functions
363 $function-body:loop: {
364 var _curr/eax: (addr function) <- lookup *curr-ah
365 var curr/edx: (addr function) <- copy _curr
366 compare curr, 0
367 break-if-=
368 var curr-name-ah/eax: (addr handle array byte) <- get curr, name
369 var curr-name/eax: (addr array byte) <- lookup *curr-name-ah
370 var found?/eax: boolean <- string-equal? curr-name, function-name
371 compare found?, 0
372 {
373 break-if-=
374 var src/eax: (addr handle line) <- get curr, body
375 copy-object src, out
376 break $function-body:loop
377 }
378 curr-ah <- get curr, next
379 loop
380 }
381 }
382
383 fn body-length functions: (addr handle function), function-name: (addr handle word) -> _/eax: int {
384 var body-storage: (handle line)
385 var body-ah/edi: (addr handle line) <- address body-storage
386 function-body functions, function-name, body-ah
387 var body/eax: (addr line) <- lookup *body-ah
388 var result/eax: int <- line-length body
389 return result
390 }
391
392 fn line-length _in: (addr line) -> _/eax: int {
393 var in/esi: (addr line) <- copy _in
394 var curr-ah/ecx: (addr handle word) <- get in, data
395 var result/edi: int <- copy 0
396 {
397 var curr/eax: (addr word) <- lookup *curr-ah
398 compare curr, 0
399 break-if-=
400 curr-ah <- get curr, next
401 result <- increment
402 loop
403 }
404 return result
405 }
406
407 fn populate-text-with _out: (addr handle array byte), _in: (addr array byte) {
408 var in/esi: (addr array byte) <- copy _in
409 var n/ecx: int <- length in
410 var out/edx: (addr handle array byte) <- copy _out
411 populate out, n
412 var _out-addr/eax: (addr array byte) <- lookup *out
413 var out-addr/edx: (addr array byte) <- copy _out-addr
414 var i/eax: int <- copy 0
415 {
416 compare i, n
417 break-if->=
418 var src/esi: (addr byte) <- index in, i
419 var val/ecx: byte <- copy-byte *src
420 var dest/edi: (addr byte) <- index out-addr, i
421 copy-byte-to *dest, val
422 i <- increment
423 loop
424 }
425 }
426
427 fn initialize-path-from-sandbox _in: (addr sandbox), _out: (addr handle call-path-element) {
428 var sandbox/esi: (addr sandbox) <- copy _in
429 var line-ah/eax: (addr handle line) <- get sandbox, data
430 var line/eax: (addr line) <- lookup *line-ah
431 var src/esi: (addr handle word) <- get line, data
432 var out-ah/edi: (addr handle call-path-element) <- copy _out
433 var out/eax: (addr call-path-element) <- lookup *out-ah
434 var dest/edi: (addr handle word) <- get out, word
435 copy-object src, dest
436 }
437
438 fn initialize-path-from-line _line: (addr line), _out: (addr handle call-path-element) {
439 var line/eax: (addr line) <- copy _line
440 var src/esi: (addr handle word) <- get line, data
441 var out-ah/edi: (addr handle call-path-element) <- copy _out
442 var out/eax: (addr call-path-element) <- lookup *out-ah
443 var dest/edi: (addr handle word) <- get out, word
444 copy-object src, dest
445 }
446
447 fn find-in-call-paths call-paths: (addr handle call-path), needle: (addr handle call-path-element) -> _/eax: boolean {
448 var curr-ah/esi: (addr handle call-path) <- copy call-paths
449 $find-in-call-path:loop: {
450 var curr/eax: (addr call-path) <- lookup *curr-ah
451 compare curr, 0
452 break-if-=
453 {
454 var curr-data/eax: (addr handle call-path-element) <- get curr, data
455 var match?/eax: boolean <- call-path-element-match? curr-data, needle
456 compare match?, 0
457 {
458 break-if-=
459 return 1
460 }
461 }
462 curr-ah <- get curr, next
463 loop
464 }
465 return 0
466 }
467
468 fn call-path-element-match? _x: (addr handle call-path-element), _y: (addr handle call-path-element) -> _/eax: boolean {
469 var x-ah/eax: (addr handle call-path-element) <- copy _x
470 var x-a/eax: (addr call-path-element) <- lookup *x-ah
471 var x/esi: (addr call-path-element) <- copy x-a
472 var y-ah/eax: (addr handle call-path-element) <- copy _y
473 var y-a/eax: (addr call-path-element) <- lookup *y-ah
474 var y/edi: (addr call-path-element) <- copy y-a
475 compare x, y
476 {
477 break-if-!=
478 return 1
479 }
480 compare x, 0
481 {
482 break-if-!=
483 return 0
484 }
485 compare y, 0
486 {
487 break-if-!=
488 return 0
489 }
490
491 var x-data-ah/ecx: (addr handle word) <- get x, word
492 var x-data-a/eax: (addr word) <- lookup *x-data-ah
493 var x-data/ecx: int <- copy x-data-a
494 var y-data-ah/eax: (addr handle word) <- get y, word
495 var y-data-a/eax: (addr word) <- lookup *y-data-ah
496 var y-data/eax: int <- copy y-data-a
497
498
499
500
501
502 compare x-data, y-data
503 {
504 break-if-=
505 return 0
506 }
507 var x-next/ecx: (addr handle call-path-element) <- get x, next
508 var y-next/eax: (addr handle call-path-element) <- get y, next
509 var result/eax: boolean <- call-path-element-match? x-next, y-next
510 return result
511 }
512
513
514 fn insert-in-call-path list: (addr handle call-path), new: (addr handle call-path-element) {
515 var new-path-storage: (handle call-path)
516 var new-path-ah/edi: (addr handle call-path) <- address new-path-storage
517 allocate new-path-ah
518 var new-path/eax: (addr call-path) <- lookup *new-path-ah
519 var next/ecx: (addr handle call-path) <- get new-path, next
520 copy-object list, next
521 var dest/ecx: (addr handle call-path-element) <- get new-path, data
522 deep-copy-call-path-element new, dest
523 copy-object new-path-ah, list
524 }
525
526
527 fn deep-copy-call-path-element _src: (addr handle call-path-element), _dest: (addr handle call-path-element) {
528 var src/esi: (addr handle call-path-element) <- copy _src
529
530 var _src-addr/eax: (addr call-path-element) <- lookup *src
531 compare _src-addr, 0
532 break-if-=
533
534 var src-addr/esi: (addr call-path-element) <- copy _src-addr
535 var dest/eax: (addr handle call-path-element) <- copy _dest
536 allocate dest
537
538 var dest-addr/eax: (addr call-path-element) <- lookup *dest
539 {
540 var dest-data-addr/ecx: (addr handle word) <- get dest-addr, word
541 var src-data-addr/eax: (addr handle word) <- get src-addr, word
542 copy-object src-data-addr, dest-data-addr
543 }
544
545 var src-next/esi: (addr handle call-path-element) <- get src-addr, next
546 var dest-next/eax: (addr handle call-path-element) <- get dest-addr, next
547 deep-copy-call-path-element src-next, dest-next
548 }
549
550 fn delete-in-call-path list: (addr handle call-path), needle: (addr handle call-path-element) {
551 var curr-ah/esi: (addr handle call-path) <- copy list
552 $delete-in-call-path:loop: {
553 var _curr/eax: (addr call-path) <- lookup *curr-ah
554 var curr/ecx: (addr call-path) <- copy _curr
555 compare curr, 0
556 break-if-=
557 {
558 var curr-data/eax: (addr handle call-path-element) <- get curr, data
559 var match?/eax: boolean <- call-path-element-match? curr-data, needle
560 compare match?, 0
561 {
562 break-if-=
563 var next-ah/ecx: (addr handle call-path) <- get curr, next
564 copy-object next-ah, curr-ah
565 loop $delete-in-call-path:loop
566 }
567 }
568 curr-ah <- get curr, next
569 loop
570 }
571 }
572
573 fn increment-final-element list: (addr handle call-path-element) {
574 var final-ah/eax: (addr handle call-path-element) <- copy list
575 var final/eax: (addr call-path-element) <- lookup *final-ah
576 var val-ah/ecx: (addr handle word) <- get final, word
577 var val/eax: (addr word) <- lookup *val-ah
578 var new-ah/edx: (addr handle word) <- get val, next
579 var target/eax: (addr word) <- lookup *new-ah
580 compare target, 0
581 break-if-=
582 copy-object new-ah, val-ah
583 }
584
585 fn decrement-final-element list: (addr handle call-path-element) {
586 var final-ah/eax: (addr handle call-path-element) <- copy list
587 var final/eax: (addr call-path-element) <- lookup *final-ah
588 var val-ah/ecx: (addr handle word) <- get final, word
589 var val/eax: (addr word) <- lookup *val-ah
590
591
592
593
594
595 var new-ah/edx: (addr handle word) <- get val, prev
596 var target/eax: (addr word) <- lookup *new-ah
597 compare target, 0
598 break-if-=
599
600
601
602
603
604
605
606 copy-object new-ah, val-ah
607 }
608
609 fn move-final-element-to-start-of-line list: (addr handle call-path-element) {
610 var final-ah/eax: (addr handle call-path-element) <- copy list
611 var final/eax: (addr call-path-element) <- lookup *final-ah
612 var val-ah/ecx: (addr handle word) <- get final, word
613 var val/eax: (addr word) <- lookup *val-ah
614 var new-ah/edx: (addr handle word) <- get val, prev
615 var target/eax: (addr word) <- lookup *new-ah
616 compare target, 0
617 break-if-=
618 copy-object new-ah, val-ah
619 move-final-element-to-start-of-line list
620 }
621
622 fn push-to-call-path-element list: (addr handle call-path-element), new: (addr handle word) {
623 var new-element-storage: (handle call-path-element)
624 var new-element-ah/edi: (addr handle call-path-element) <- address new-element-storage
625 allocate new-element-ah
626 var new-element/eax: (addr call-path-element) <- lookup *new-element-ah
627
628 var dest/ecx: (addr handle word) <- get new-element, word
629 copy-object new, dest
630
631 var dest2/ecx: (addr handle call-path-element) <- get new-element, next
632 copy-object list, dest2
633
634 copy-object new-element-ah, list
635 }
636
637 fn drop-from-call-path-element _list: (addr handle call-path-element) {
638 var list-ah/esi: (addr handle call-path-element) <- copy _list
639 var list/eax: (addr call-path-element) <- lookup *list-ah
640 var next/eax: (addr handle call-path-element) <- get list, next
641 copy-object next, _list
642 }
643
644 fn drop-nested-calls _list: (addr handle call-path-element) {
645 var list-ah/esi: (addr handle call-path-element) <- copy _list
646 var list/eax: (addr call-path-element) <- lookup *list-ah
647 var next-ah/edi: (addr handle call-path-element) <- get list, next
648 var next/eax: (addr call-path-element) <- lookup *next-ah
649 compare next, 0
650 break-if-=
651 copy-object next-ah, _list
652 drop-nested-calls _list
653 }
654
655 fn dump-call-path-element screen: (addr screen), _x-ah: (addr handle call-path-element) {
656 var x-ah/ecx: (addr handle call-path-element) <- copy _x-ah
657 var _x/eax: (addr call-path-element) <- lookup *x-ah
658 var x/esi: (addr call-path-element) <- copy _x
659 var word-ah/eax: (addr handle word) <- get x, word
660 var word/eax: (addr word) <- lookup *word-ah
661 print-word screen, word
662 var next-ah/ecx: (addr handle call-path-element) <- get x, next
663 var next/eax: (addr call-path-element) <- lookup *next-ah
664 compare next, 0
665 {
666 break-if-=
667 print-string screen, " "
668 dump-call-path-element screen, next-ah
669 return
670 }
671 print-string screen, "\n"
672 }
673
674 fn dump-call-paths screen: (addr screen), _x-ah: (addr handle call-path) {
675 var x-ah/ecx: (addr handle call-path) <- copy _x-ah
676 var x/eax: (addr call-path) <- lookup *x-ah
677 compare x, 0
678 break-if-=
679 var src/ecx: (addr handle call-path-element) <- get x, data
680 dump-call-path-element screen, src
681 var next-ah/ecx: (addr handle call-path) <- get x, next
682 var next/eax: (addr call-path) <- lookup *next-ah
683 compare next, 0
684 {
685 break-if-=
686 dump-call-paths screen, next-ah
687 }
688 }