-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsel-attributes.lisp
86 lines (77 loc) · 3 KB
/
sel-attributes.lisp
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
(defpackage :functional-trees/sel-attributes
(:nicknames :ft/sel-attributes :ft/sel-attrs)
(:use
:gt/full
:software-evolution-library/software/tree-sitter
:software-evolution-library/software/c
:functional-trees
:functional-trees/attrs
:named-readtables
:curry-compose-reader-macros
:named-readtables)
(:shadowing-import-from :software-evolution-library/software/parseable
:source-text :text)
(:shadowing-import-from :fset :set :map :union :empty-set :empty-map
:restrict)
(:export :st :defs :uses)
(:documentation "Package for integration of SEL with FT attributes."))
(in-package :functional-trees/sel-attributes)
(in-readtable :curry-compose-reader-macros)
;;; Define a simple propagator for type information on C terms
(defun st-union (st1 st2) (fset:map-union st1 st2))
(defun st-add (st key value)
(if key (fset:with st key value) st))
;;; There are three attr functions in this example.
;;; The first, st, is the symbol table map coming into the node.
;;; The second, defs, is the map of definitions produced by and
;;; exported by the node.
;;; The third, uses, is a set of names that occur in a subtree.
(def-attr-fun st (in)
"Compute the symbol table at this node."
;; Default method: propagate down
(:method ((node node) &optional in)
;; This passes the full ST down to the subtree
;; (mapc-attrs-children #'st (list in) node)
;; but this prunes off all the symbols not used in the subtree,
;; which may make incrementalization easier.
(mapc (lambda (n) (st n (restrict in (uses n))))
(children node))
in)
;; Perhaps include a superclass that means definitions propagate across
;; the children
(:method ((node c-compound-statement) &optional in)
;; Propagate across children
(reduce (lambda (in2 child) (st-union (st child in2) (defs child)))
(children node)
:initial-value in)
in)
(:method ((node c-translation-unit) &optional in)
;; Propagate across children
(reduce (lambda (in2 child) (st-union (st child in2) (defs child)))
(children node)
:initial-value in))
)
(def-attr-fun defs ()
"Map of definitions from a node"
(:method ((node node))
(empty-map))
(:method ((node c-declaration))
(decl-map node))
)
(def-attr-fun uses ()
"Set of names that occur in a subtree"
(:method ((node node))
(reduce #'union (children node)
:key #'uses :initial-value (fset:empty-set)))
(:method ((node c-identifier))
(fset:set (text node))))
(defgeneric decl-map (node)
(:documentation "Construct a map that gives the declarations produced by NODE")
;; This is a very simple prototype, handling only simple declarations
(:method ((node c-declaration))
(let* ((type (c-type node))
(alist
(iter (for declarator in (c-declarator node))
(when-let ((name declarator))
(collect (cons (text name) (text type)))))))
(convert 'fset:map alist))))