1 ;; address after which memory is allocated by the user, therefore not used for
3 ;; in programs, located in the SIXPIC_MEMORY_DIVIDE variable
4 (define memory-divide #f)
6 (define (interference-graph cfg)
8 (define (analyze-liveness cfg)
10 (define (instr-analyze-liveness instr live-after)
14 (let ((def-proc (call-instr-def-proc instr)))
15 (if (and (not (set-empty? live-after))
18 (def-procedure-live-after-calls def-proc)
23 (set-union! (def-procedure-live-after-calls def-proc)
28 (map (lambda (def-var)
30 (value-bytes (def-variable-value def-var))))
31 (def-procedure-params def-proc)))
35 (value-bytes (def-procedure-value def-proc)))))))
36 (if (bb? (def-procedure-entry def-proc))
38 (bb-live-before (def-procedure-entry def-proc))
41 ((return-instr? instr)
42 (let ((def-proc (return-instr-def-proc instr)))
44 (if (def-procedure? def-proc)
45 (def-procedure-live-after-calls def-proc)
46 (list->set (value-bytes def-proc)))))
47 (set! live-after live)
50 (let* ((src1 (instr-src1 instr))
51 (src2 (instr-src2 instr))
52 (dst (instr-dst instr))
53 (use (if (byte-cell? src1)
55 (set-add (new-set src1) src2)
60 (def (if (byte-cell? dst)
64 ;; (and (byte-cell? dst) ; dead instruction?
65 ;; (not (set-member? live-after dst))
66 ;; (not (and (byte-cell? dst) (byte-cell-adr dst))))
69 (set-diff live-after def))))))))
70 (instr-live-before-set! instr live-before)
71 (instr-live-after-set! instr live-after)
73 (define (bb-analyze-liveness bb)
74 (let loop ((rev-instrs (bb-rev-instrs bb))
75 (live-after (set-union-multi
76 (map bb-live-before (bb-succs bb)))))
77 (if (null? rev-instrs)
78 (if (not (set-equal? live-after (bb-live-before bb)))
79 (begin (set! changed? #t)
80 (bb-live-before-set! bb live-after)))
81 (let* ((instr (car rev-instrs))
82 (live-before (instr-analyze-liveness instr live-after)))
83 (loop (cdr rev-instrs)
87 (begin (set! changed? #f)
88 (for-each bb-analyze-liveness (cfg-bbs cfg))
91 (define all-live (new-empty-set))
92 (define (bb-interference-graph bb)
93 (define (interfere x y)
94 (if (not (set-member? (byte-cell-interferes-with y) x))
95 (begin (set-add! (byte-cell-interferes-with x) y)
96 (set-add! (byte-cell-interferes-with y) x))))
97 (define (interfere-pairwise live)
98 (set-union! all-live live) ;; TODO build the live set only once, if not a set already
99 (set-for-each ;; TODO for each cell in live, do the union of live, diff itself, if live is not a set, we win so we can iterate on something better than a hash table
100 ;; TODO since all true variables will be in the low numbers and all temps in the high numbers, we can have sparse bit vectors (if all below n is 0, don't store it) and save (or even have a bit vector that can store data by chunks, leaving empty space)
102 (set-for-each (lambda (y) (if (not (eq? x y)) (interfere x y)))
105 (define (instr-interference-graph instr)
106 (let ((dst (instr-dst instr)))
108 (let ((src1 (instr-src1 instr))
109 (src2 (instr-src2 instr)))
110 (if (byte-cell? src1)
111 (begin (set-add! (byte-cell-coalesceable-with dst) src1)
112 (set-add! (byte-cell-coalesceable-with src1) dst)))
113 (if (byte-cell? src2)
114 (begin (set-add! (byte-cell-coalesceable-with dst) src2)
115 (set-add! (byte-cell-coalesceable-with src2) dst)))
116 (interfere-pairwise (set-add (instr-live-after instr) dst)))))
117 (interfere-pairwise (instr-live-before instr)))
118 (for-each instr-interference-graph (bb-rev-instrs bb)))
120 (pp analyse-liveness:)
121 (time (analyze-liveness cfg))
123 (pp interference-graph:)
124 (time (for-each bb-interference-graph (cfg-bbs cfg)))
128 ;;-----------------------------------------------------------------------------
130 (define register-table (make-table))
131 (define (allocate-registers cfg)
132 (let ((all-live (interference-graph cfg))
133 (max-adr 0)) ; to know how much ram we need
135 (define (color byte-cell)
136 (let (;; (coalesce-candidates ; TODO right now, no coalescing is done
137 ;; (set-filter byte-cell-adr
138 ;; (set-diff (byte-cell-coalesceable-with byte-cell)
139 ;; (byte-cell-interferes-with byte-cell))))
142 ;; (pp (list byte-cell: byte-cell;;;;;;;;;;;;;;;
143 ;; coalesce-candidates
144 ;; ; interferes-with: (byte-cell-interferes-with byte-cell)
145 ;; ; coalesceable-with: (byte-cell-coalesceable-with byte-cell)
148 (if #f #;(not (null? coalesce-candidates))
149 (let ((adr (byte-cell-adr (car (set->list coalesce-candidates))))) ;; TODO have as a set all along
150 (byte-cell-adr-set! byte-cell adr))
151 (let ((neighbours (byte-cell-interferes-with byte-cell)))
153 (if (and memory-divide ; the user wants his own zone
154 (>= adr memory-divide)) ; and we'd use it
155 (error "register allocation would cross the memory divide") ;; TODO fallback ?
156 (let loop2 ((lst (set->list neighbours))) ;; TODO keep using sets, but not urgent, it's not a bottleneck
158 (begin (byte-cell-adr-set! byte-cell adr)
161 (if (and (> adr #x5F) (< adr #xF60))
165 (cons (byte-cell-name byte-cell)
166 (table-ref register-table adr '()))))
168 (if (= adr (byte-cell-adr x))
170 (loop2 (cdr lst)))))))
171 (set! max-adr (max max-adr adr)))))))
173 (define (delete byte-cell1 neighbours)
174 (set-for-each (lambda (byte-cell2)
175 (set-remove! (byte-cell-interferes-with byte-cell2)
179 (define (undelete byte-cell1 neighbours)
180 (set-for-each (lambda (byte-cell2)
181 (set-add! (byte-cell-interferes-with byte-cell2)
185 (define (find-min-neighbours graph)
186 (let loop ((lst graph) (m #f) (byte-cell #f))
190 (n (set-length (byte-cell-interferes-with x))))
191 (if (or (not m) (< n m))
193 (loop (cdr lst) m byte-cell))))))
195 (define (alloc-reg graph)
196 (if (not (null? graph))
197 (let* ((byte-cell (find-min-neighbours graph))
198 (neighbours (byte-cell-interferes-with byte-cell)))
199 (let ((new-graph (remove byte-cell graph)))
200 (delete byte-cell neighbours)
201 (alloc-reg new-graph)
202 (undelete byte-cell neighbours))
203 (if (not (byte-cell-adr byte-cell))
204 (color byte-cell)))))
206 (pp register-allocation:)
207 (time (alloc-reg (set->list all-live))) ;; TODO convert find-min-neighbors and alloc-reg to use tables, not urgent since it's not a bottleneck
208 (display (string-append (number->string (+ max-adr 1)) " RAM bytes\n"))))