d0a705c8c35d4e90a3eae597d0bb9e3822204dfe
[sbcl.git] / src / compiler / mips / insts.lisp
1 ;;; the instruction set definition for MIPS
2
3 ;;;; This software is part of the SBCL system. See the README file for
4 ;;;; more information.
5 ;;;;
6 ;;;; This software is derived from the CMU CL system, which was
7 ;;;; written at Carnegie Mellon University and released into the
8 ;;;; public domain. The software is in the public domain and is
9 ;;;; provided with absolutely no warranty. See the COPYING and CREDITS
10 ;;;; files for more information.
11
12 (in-package "SB!VM")
13
14 (setf *assem-scheduler-p* t)
15 (setf *assem-max-locations* 68)
16 \f
17 ;;;; Constants, types, conversion functions, some disassembler stuff.
18
19 (defun reg-tn-encoding (tn)
20   (declare (type tn tn))
21   (sc-case tn
22     (zero zero-offset)
23     (null null-offset)
24     (t
25      (if (eq (sb-name (sc-sb (tn-sc tn))) 'registers)
26          (tn-offset tn)
27          (error "~S isn't a register." tn)))))
28
29 (defun fp-reg-tn-encoding (tn)
30   (declare (type tn tn))
31   (unless (eq (sb-name (sc-sb (tn-sc tn))) 'float-registers)
32     (error "~S isn't a floating-point register." tn))
33   (tn-offset tn))
34
35 ;;;(sb!disassem:set-disassem-params :instruction-alignment 32)
36
37 (defvar *disassem-use-lisp-reg-names* t)
38
39 (!def-vm-support-routine location-number (loc)
40   (etypecase loc
41     (null)
42     (number)
43     (label)
44     (fixup)
45     (tn
46      (ecase (sb-name (sc-sb (tn-sc loc)))
47        (immediate-constant
48         ;; Can happen if $ZERO or $NULL are passed in.
49         nil)
50        (registers
51         (unless (zerop (tn-offset loc))
52           (tn-offset loc)))
53        (float-registers
54         (+ (tn-offset loc) 32))))
55     (symbol
56      (ecase loc
57        (:memory 0)
58        (:hi-reg 64)
59        (:low-reg 65)
60        (:float-status 66)
61        (:ctrl-stat-reg 67)
62        (:r31 31)))))
63
64 (defparameter reg-symbols
65   (map 'vector
66        #'(lambda (name)
67            (cond ((null name) nil)
68                  (t (make-symbol (concatenate 'string "$" name)))))
69        *register-names*))
70
71 (sb!disassem:define-arg-type reg
72   :printer #'(lambda (value stream dstate)
73                (declare (stream stream) (fixnum value))
74                (let ((regname (aref reg-symbols value)))
75                  (princ regname stream)
76                  (sb!disassem:maybe-note-associated-storage-ref
77                   value
78                   'registers
79                   regname
80                   dstate))))
81
82 (defparameter float-reg-symbols
83   #.(coerce
84      (loop for n from 0 to 31 collect (make-symbol (format nil "$F~d" n)))
85      'vector))
86
87 (sb!disassem:define-arg-type fp-reg
88   :printer #'(lambda (value stream dstate)
89                (declare (stream stream) (fixnum value))
90                (let ((regname (aref float-reg-symbols value)))
91                  (princ regname stream)
92                  (sb!disassem:maybe-note-associated-storage-ref
93                   value
94                   'float-registers
95                   regname
96                   dstate))))
97
98 (sb!disassem:define-arg-type control-reg
99   :printer "(CR:#x~X)")
100
101 (sb!disassem:define-arg-type relative-label
102   :sign-extend t
103   :use-label #'(lambda (value dstate)
104                  (declare (type (signed-byte 16) value)
105                           (type sb!disassem:disassem-state dstate))
106                  (+ (ash (1+ value) 2) (sb!disassem:dstate-cur-addr dstate))))
107
108 (deftype float-format ()
109   '(member :s :single :d :double :w :word))
110
111 (defun float-format-value (format)
112   (ecase format
113     ((:s :single) 0)
114     ((:d :double) 1)
115     ((:w :word) 4)))
116
117 (sb!disassem:define-arg-type float-format
118   :printer #'(lambda (value stream dstate)
119                (declare (ignore dstate)
120                         (stream stream)
121                         (fixnum value))
122                (princ (case value
123                         (0 's)
124                         (1 'd)
125                         (4 'w)
126                         (t '?))
127                       stream)))
128
129 (defconstant-eqx compare-kinds
130   '(:f :un :eq :ueq :olt :ult :ole :ule :sf :ngle :seq :ngl :lt :nge :le :ngt)
131   #'equalp)
132
133 (defconstant-eqx compare-kinds-vec
134   (apply #'vector compare-kinds)
135   #'equalp)
136
137 (deftype compare-kind ()
138   `(member ,@compare-kinds))
139
140 (defun compare-kind (kind)
141   (or (position kind compare-kinds)
142       (error "Unknown floating point compare kind: ~S~%Must be one of: ~S"
143              kind
144              compare-kinds)))
145
146 (sb!disassem:define-arg-type compare-kind
147   :printer compare-kinds-vec)
148
149 (defconstant-eqx float-operations '(+ - * /) #'equalp)
150
151 (deftype float-operation ()
152   `(member ,@float-operations))
153
154 (defconstant-eqx float-operation-names
155   ;; this gets used for output only
156   #(add sub mul div)
157   #'equalp)
158
159 (defun float-operation (op)
160   (or (position op float-operations)
161       (error "Unknown floating point operation: ~S~%Must be one of: ~S"
162              op
163              float-operations)))
164
165 (sb!disassem:define-arg-type float-operation
166   :printer float-operation-names)
167
168
169 \f
170 ;;;; Constants used by instruction emitters.
171
172 (defconstant special-op #b000000)
173 (defconstant bcond-op #b000001)
174 (defconstant cop0-op #b010000)
175 (defconstant cop1-op #b010001)
176 (defconstant cop2-op #b010010)
177 (defconstant cop3-op #b010011)
178
179
180 \f
181 ;;;; dissassem:define-instruction-formats
182
183 (defconstant-eqx immed-printer
184   '(:name :tab rt (:unless (:same-as rt) ", " rs) ", " immediate)
185   #'equalp)
186
187 ;;; for things that use rt=0 as a nop
188 (defconstant-eqx immed-zero-printer
189   '(:name :tab rt (:unless (:constant 0) ", " rs) ", " immediate)
190   #'equalp)
191
192 (sb!disassem:define-instruction-format
193     (immediate 32 :default-printer immed-printer)
194   (op :field (byte 6 26))
195   (rs :field (byte 5 21) :type 'reg)
196   (rt :field (byte 5 16) :type 'reg)
197   (immediate :field (byte 16 0) :sign-extend t))
198
199 (eval-when (:compile-toplevel :load-toplevel :execute)
200   (defparameter jump-printer
201     #'(lambda (value stream dstate)
202         (let ((addr (ash value 2)))
203           (sb!disassem:maybe-note-assembler-routine addr t dstate)
204           (write addr :base 16 :radix t :stream stream)))))
205
206 (sb!disassem:define-instruction-format
207     (jump 32 :default-printer '(:name :tab target))
208   (op :field (byte 6 26))
209   (target :field (byte 26 0) :printer jump-printer))
210
211 (defconstant-eqx reg-printer
212   '(:name :tab rd (:unless (:same-as rd) ", " rs) ", " rt)
213   #'equalp)
214
215 (sb!disassem:define-instruction-format
216     (register 32 :default-printer reg-printer)
217   (op :field (byte 6 26))
218   (rs :field (byte 5 21) :type 'reg)
219   (rt :field (byte 5 16) :type 'reg)
220   (rd :field (byte 5 11) :type 'reg)
221   (shamt :field (byte 5 6) :value 0)
222   (funct :field (byte 6 0)))
223
224 (sb!disassem:define-instruction-format
225     (break 32 :default-printer
226            '(:name :tab code (:unless (:constant 0) subcode)))
227   (op :field (byte 6 26) :value special-op)
228   (code :field (byte 10 16))
229   (subcode :field (byte 10 6) :value 0)
230   (funct :field (byte 6 0) :value #b001101))
231
232 (sb!disassem:define-instruction-format
233     (coproc-branch 32 :default-printer '(:name :tab offset))
234   (op :field (byte 6 26))
235   (funct :field (byte 10 16))
236   (offset :field (byte 16 0)))
237
238 (defconstant-eqx float-fmt-printer
239   '((:unless :constant funct)
240     (:choose (:unless :constant sub-funct) nil)
241     "." format)
242   #'equalp)
243
244 (defconstant-eqx float-printer
245   `(:name ,@float-fmt-printer
246           :tab
247           fd
248           (:unless (:same-as fd) ", " fs)
249           ", " ft)
250   #'equalp)
251
252 (sb!disassem:define-instruction-format
253     (float 32 :default-printer float-printer)
254   (op :field (byte 6 26) :value cop1-op)
255   (filler :field (byte 1 25) :value 1)
256   (format :field (byte 4 21) :type 'float-format)
257   (ft :field (byte 5 16) :value 0)
258   (fs :field (byte 5 11) :type 'fp-reg)
259   (fd :field (byte 5 6) :type 'fp-reg)
260   (funct :field (byte 6 0)))
261
262 (sb!disassem:define-instruction-format
263     (float-aux 32 :default-printer float-printer)
264   (op :field (byte 6 26) :value cop1-op)
265   (filler-1 :field (byte 1 25) :value 1)
266   (format :field (byte 4 21) :type 'float-format)
267   (ft :field (byte 5 16) :type 'fp-reg)
268   (fs :field (byte 5 11) :type 'fp-reg)
269   (fd :field (byte 5 6) :type 'fp-reg)
270   (funct :field (byte 2 4))
271   (sub-funct :field (byte 4 0)))
272
273 (sb!disassem:define-instruction-format
274     (float-op 32
275               :include 'float
276               :default-printer
277                 '('f funct "." format
278                   :tab
279                   fd
280                   (:unless (:same-as fd) ", " fs)
281                   ", " ft))
282   (funct        :field (byte 2 0) :type 'float-operation)
283   (funct-filler :field (byte 4 2) :value 0)
284   (ft           :value nil :type 'fp-reg))
285
286 \f
287 ;;;; Primitive emitters.
288
289 (define-bitfield-emitter emit-word 32
290   (byte 32 0))
291
292 (define-bitfield-emitter emit-short 16
293   (byte 16 0))
294
295 (define-bitfield-emitter emit-immediate-inst 32
296   (byte 6 26) (byte 5 21) (byte 5 16) (byte 16 0))
297
298 (define-bitfield-emitter emit-jump-inst 32
299   (byte 6 26) (byte 26 0))
300
301 (define-bitfield-emitter emit-register-inst 32
302   (byte 6 26) (byte 5 21) (byte 5 16) (byte 5 11) (byte 5 6) (byte 6 0))
303
304 (define-bitfield-emitter emit-break-inst 32
305   (byte 6 26) (byte 10 16) (byte 10 6) (byte 6 0))
306
307 (define-bitfield-emitter emit-float-inst 32
308   (byte 6 26) (byte 1 25) (byte 4 21) (byte 5 16)
309   (byte 5 11) (byte 5 6) (byte 6 0))
310
311
312 \f
313 ;;;; Math instructions.
314
315 (defun emit-math-inst (segment dst src1 src2 reg-opcode immed-opcode
316                                &optional allow-fixups)
317   (unless src2
318     (setf src2 src1)
319     (setf src1 dst))
320   (etypecase src2
321     (tn
322      (emit-register-inst segment special-op (reg-tn-encoding src1)
323                          (reg-tn-encoding src2) (reg-tn-encoding dst)
324                          0 reg-opcode))
325     (integer
326      (emit-immediate-inst segment immed-opcode (reg-tn-encoding src1)
327                           (reg-tn-encoding dst) src2))
328     (fixup
329      (unless allow-fixups
330        (error "Fixups aren't allowed."))
331      (note-fixup segment :addi src2)
332      (emit-immediate-inst segment immed-opcode (reg-tn-encoding src1)
333                           (reg-tn-encoding dst) 0))))
334
335 (define-instruction add (segment dst src1 &optional src2)
336   (:declare (type tn dst)
337             (type (or tn (signed-byte 16) null) src1 src2))
338   (:printer register ((op special-op) (funct #b100000)))
339   (:printer immediate ((op #b001000)))
340   (:dependencies (reads src1) (if src2 (reads src2) (reads dst)) (writes dst))
341   (:delay 0)
342   (:emitter
343    (emit-math-inst segment dst src1 src2 #b100000 #b001000)))
344
345 (define-instruction addu (segment dst src1 &optional src2)
346   (:declare (type tn dst)
347             (type (or tn (signed-byte 16) fixup null) src1 src2))
348   (:printer register ((op special-op) (funct #b100001)))
349   (:printer immediate ((op #b001001)))
350   (:dependencies (reads src1) (if src2 (reads src2) (reads dst)) (writes dst))
351   (:delay 0)
352   (:emitter
353    (emit-math-inst segment dst src1 src2 #b100001 #b001001 t)))
354
355 (define-instruction sub (segment dst src1 &optional src2)
356   (:declare
357    (type tn dst)
358    (type (or tn (integer #.(- 1 (ash 1 15)) #.(ash 1 15)) null) src1 src2))
359   (:printer register ((op special-op) (funct #b100010)))
360   (:dependencies (reads src1) (if src2 (reads src2) (reads dst)) (writes dst))
361   (:delay 0)
362   (:emitter
363    (unless src2
364      (setf src2 src1)
365      (setf src1 dst))
366    (emit-math-inst segment dst src1
367                    (if (integerp src2) (- src2) src2)
368                    #b100010 #b001000)))
369
370 (define-instruction subu (segment dst src1 &optional src2)
371   (:declare
372    (type tn dst)
373    (type
374     (or tn (integer #.(- 1 (ash 1 15)) #.(ash 1 15)) fixup null) src1 src2))
375   (:printer register ((op special-op) (funct #b100011)))
376   (:dependencies (reads src1) (if src2 (reads src2) (reads dst)) (writes dst))
377   (:delay 0)
378   (:emitter
379    (unless src2
380      (setf src2 src1)
381      (setf src1 dst))
382    (emit-math-inst segment dst src1
383                    (if (integerp src2) (- src2) src2)
384                    #b100011 #b001001 t)))
385
386 (define-instruction and (segment dst src1 &optional src2)
387   (:declare (type tn dst)
388             (type (or tn (unsigned-byte 16) null) src1 src2))
389   (:printer register ((op special-op) (funct #b100100)))
390   (:printer immediate ((op #b001100) (immediate nil :sign-extend nil)))
391   (:dependencies (reads src1) (if src2 (reads src2) (reads dst)) (writes dst))
392   (:delay 0)
393   (:emitter
394    (emit-math-inst segment dst src1 src2 #b100100 #b001100)))
395
396 (define-instruction or (segment dst src1 &optional src2)
397   (:declare (type tn dst)
398             (type (or tn (unsigned-byte 16) null) src1 src2))
399   (:printer register ((op special-op) (funct #b100101)))
400   (:printer immediate ((op #b001101)))
401   (:dependencies (reads src1) (if src2 (reads src2) (reads dst)) (writes dst))
402   (:delay 0)
403   (:emitter
404    (emit-math-inst segment dst src1 src2 #b100101 #b001101)))
405
406 (define-instruction xor (segment dst src1 &optional src2)
407   (:declare (type tn dst)
408             (type (or tn (unsigned-byte 16) null) src1 src2))
409   (:printer register ((op special-op) (funct #b100110)))
410   (:printer immediate ((op #b001110)))
411   (:dependencies (reads src1) (if src2 (reads src2) (reads dst)) (writes dst))
412   (:delay 0)
413   (:emitter
414    (emit-math-inst segment dst src1 src2 #b100110 #b001110)))
415
416 (define-instruction nor (segment dst src1 &optional src2)
417   (:declare (type tn dst src1) (type (or tn null) src2))
418   (:printer register ((op special-op) (funct #b100111)))
419   (:dependencies (reads src1) (if src2 (reads src2) (reads dst)) (writes dst))
420   (:delay 0)
421   (:emitter
422    (emit-math-inst segment dst src1 src2 #b100111 #b000000)))
423
424 (define-instruction slt (segment dst src1 &optional src2)
425   (:declare (type tn dst)
426             (type (or tn (signed-byte 16) null) src1 src2))
427   (:printer register ((op special-op) (funct #b101010)))
428   (:printer immediate ((op #b001010)))
429   (:dependencies (reads src1) (if src2 (reads src2) (reads dst)) (writes dst))
430   (:delay 0)
431   (:emitter
432    (emit-math-inst segment dst src1 src2 #b101010 #b001010)))
433
434 (define-instruction sltu (segment dst src1 &optional src2)
435   (:declare (type tn dst)
436             (type (or tn (signed-byte 16) null) src1 src2))
437   (:printer register ((op special-op) (funct #b101011)))
438   (:printer immediate ((op #b001011)))
439   (:dependencies (reads src1) (if src2 (reads src2) (reads dst)) (writes dst))
440   (:delay 0)
441   (:emitter
442    (emit-math-inst segment dst src1 src2 #b101011 #b001011)))
443
444 (defconstant-eqx divmul-printer '(:name :tab rs ", " rt) #'equalp)
445
446 (define-instruction div (segment src1 src2)
447   (:declare (type tn src1 src2))
448   (:printer register ((op special-op) (rd 0) (funct #b011010)) divmul-printer)
449   (:dependencies (reads src1) (reads src2) (writes :hi-reg) (writes :low-reg))
450   (:delay 1)
451   (:emitter
452    (emit-register-inst segment special-op (reg-tn-encoding src1)
453                        (reg-tn-encoding src2) 0 0 #b011010)))
454
455 (define-instruction divu (segment src1 src2)
456   (:declare (type tn src1 src2))
457   (:printer register ((op special-op) (rd 0) (funct #b011011))
458             divmul-printer)
459   (:dependencies (reads src1) (reads src2) (writes :hi-reg) (writes :low-reg))
460   (:delay 1)
461   (:emitter
462    (emit-register-inst segment special-op (reg-tn-encoding src1)
463                        (reg-tn-encoding src2) 0 0 #b011011)))
464
465 (define-instruction mult (segment src1 src2)
466   (:declare (type tn src1 src2))
467   (:printer register ((op special-op) (rd 0) (funct #b011000)) divmul-printer)
468   (:dependencies (reads src1) (reads src2) (writes :hi-reg) (writes :low-reg))
469   (:delay 1)
470   (:emitter
471    (emit-register-inst segment special-op (reg-tn-encoding src1)
472                        (reg-tn-encoding src2) 0 0 #b011000)))
473
474 (define-instruction multu (segment src1 src2)
475   (:declare (type tn src1 src2))
476   (:printer register ((op special-op) (rd 0) (funct #b011001)))
477   (:dependencies (reads src1) (reads src2) (writes :hi-reg) (writes :low-reg))
478   (:delay 1)
479   (:emitter
480    (emit-register-inst segment special-op (reg-tn-encoding src1)
481                        (reg-tn-encoding src2) 0 0 #b011001)))
482
483 (defun emit-shift-inst (segment opcode dst src1 src2)
484   (unless src2
485     (setf src2 src1)
486     (setf src1 dst))
487   (etypecase src2
488     (tn
489      (emit-register-inst segment special-op (reg-tn-encoding src2)
490                          (reg-tn-encoding src1) (reg-tn-encoding dst)
491                          0 (logior #b000100 opcode)))
492     ((unsigned-byte 5)
493      (emit-register-inst segment special-op 0 (reg-tn-encoding src1)
494                          (reg-tn-encoding dst) src2 opcode))))
495
496 (defconstant-eqx shift-printer
497   '(:name :tab
498           rd
499           (:unless (:same-as rd) ", " rt)
500           ", " (:cond ((rs :constant 0) shamt)
501                       (t rs)))
502   #'equalp)
503
504 (define-instruction sll (segment dst src1 &optional src2)
505   (:declare (type tn dst)
506             (type (or tn (unsigned-byte 5) null) src1 src2))
507   (:printer register ((op special-op) (rs 0) (shamt nil) (funct #b000000))
508             shift-printer)
509   (:printer register ((op special-op) (funct #b000100)) shift-printer)
510   (:dependencies (reads src1) (if src2 (reads src2) (reads dst)) (writes dst))
511   (:delay 0)
512   (:emitter
513    (emit-shift-inst segment #b00 dst src1 src2)))
514
515 (define-instruction sra (segment dst src1 &optional src2)
516   (:declare (type tn dst)
517             (type (or tn (unsigned-byte 5) null) src1 src2))
518   (:printer register ((op special-op) (rs 0) (shamt nil) (funct #b000011))
519             shift-printer)
520   (:printer register ((op special-op) (funct #b000111)) shift-printer)
521   (:dependencies (reads src1) (if src2 (reads src2) (reads dst)) (writes dst))
522   (:delay 0)
523   (:emitter
524    (emit-shift-inst segment #b11 dst src1 src2)))
525
526 (define-instruction srl (segment dst src1 &optional src2)
527   (:declare (type tn dst)
528             (type (or tn (unsigned-byte 5) null) src1 src2))
529   (:printer register ((op special-op) (rs 0) (shamt nil) (funct #b000010))
530             shift-printer)
531   (:printer register ((op special-op) (funct #b000110)) shift-printer)
532   (:dependencies (reads src1) (if src2 (reads src2) (reads dst)) (writes dst))
533   (:delay 0)
534   (:emitter
535    (emit-shift-inst segment #b10 dst src1 src2)))
536
537 \f
538 ;;;; Floating point math.
539
540 (define-instruction float-op (segment operation format dst src1 src2)
541   (:declare (type float-operation operation)
542             (type float-format format)
543             (type tn dst src1 src2))
544   (:printer float-op ())
545   (:dependencies (reads src1) (reads src2) (writes dst))
546   (:delay 0)
547   (:emitter
548    (emit-float-inst segment cop1-op 1 (float-format-value format)
549                     (fp-reg-tn-encoding src2) (fp-reg-tn-encoding src1)
550                     (fp-reg-tn-encoding dst) (float-operation operation))))
551
552 (defconstant-eqx float-unop-printer
553   `(:name ,@float-fmt-printer :tab fd (:unless (:same-as fd) ", " fs))
554   #'equalp)
555
556 (define-instruction fabs (segment format dst &optional (src dst))
557   (:declare (type float-format format) (type tn dst src))
558   (:printer float ((funct #b000101)) float-unop-printer)
559   (:dependencies (reads src) (writes dst))
560   (:delay 0)
561   (:emitter
562    (emit-float-inst segment cop1-op 1 (float-format-value format)
563                     0 (fp-reg-tn-encoding src) (fp-reg-tn-encoding dst)
564                     #b000101)))
565
566 (define-instruction fneg (segment format dst &optional (src dst))
567   (:declare (type float-format format) (type tn dst src))
568   (:printer float ((funct #b000111)) float-unop-printer)
569   (:dependencies (reads src) (writes dst))
570   (:delay 0)
571   (:emitter
572    (emit-float-inst segment cop1-op 1 (float-format-value format)
573                     0 (fp-reg-tn-encoding src) (fp-reg-tn-encoding dst)
574                     #b000111)))
575
576 (define-instruction fcvt (segment format1 format2 dst src)
577   (:declare (type float-format format1 format2) (type tn dst src))
578   (:printer float-aux ((funct #b10) (sub-funct nil :type 'float-format))
579            `(:name "." sub-funct "." format :tab fd ", " fs))
580   (:dependencies (reads src) (writes dst))
581   (:delay 0)
582   (:emitter
583    (emit-float-inst segment cop1-op 1 (float-format-value format2) 0
584                     (fp-reg-tn-encoding src) (fp-reg-tn-encoding dst)
585                     (logior #b100000 (float-format-value format1)))))
586
587 (define-instruction fcmp (segment operation format fs ft)
588   (:declare (type compare-kind operation)
589             (type float-format format)
590             (type tn fs ft))
591   (:printer float-aux ((fd 0) (funct #b11) (sub-funct nil :type 'compare-kind))
592             `(:name "-" sub-funct "." format :tab fs ", " ft))
593   (:dependencies (reads fs) (reads ft) (writes :float-status))
594   (:delay 1)
595   (:emitter
596    (emit-float-inst segment cop1-op 1 (float-format-value format)
597                     (fp-reg-tn-encoding ft) (fp-reg-tn-encoding fs) 0
598                     (logior #b110000 (compare-kind operation)))))
599
600 \f
601 ;;;; Branch/Jump instructions.
602
603 (defun emit-relative-branch (segment opcode r1 r2 target)
604   (emit-chooser
605    segment 20 2
606       #'(lambda (segment posn magic-value)
607           (declare (ignore magic-value))
608           (let ((delta (ash (- (label-position target) (+ posn 4)) -2)))
609             (when (typep delta '(signed-byte 16))
610               (emit-back-patch segment 4
611                 #'(lambda (segment posn)
612                     (emit-immediate-inst segment
613                                          opcode
614                                          (if (fixnump r1)
615                                              r1
616                                              (reg-tn-encoding r1))
617                                          (if (fixnump r2)
618                                              r2
619                                              (reg-tn-encoding r2))
620                                          (ash (- (label-position target)
621                                                  (+ posn 4))
622                                               -2))))
623               t)))
624       #'(lambda (segment posn)
625           (declare (ignore posn))
626           (let ((linked))
627             ;; invert branch condition
628             (if (or (= opcode bcond-op) (= opcode cop1-op))
629                 (setf r2 (logxor r2 #b00001))
630                 (setf opcode (logxor opcode #b00001)))
631             ;; check link flag
632             (if (= opcode bcond-op)
633                 (if (logand r2 #b10000)
634                     (progn (setf r2 (logand r2 #b01111))
635                            (setf linked t))))
636             (emit-immediate-inst segment
637                                  opcode
638                                  (if (fixnump r1) r1 (reg-tn-encoding r1))
639                                  (if (fixnump r2) r2 (reg-tn-encoding r2))
640                                  4)
641             (emit-nop segment)
642             (emit-back-patch segment 8
643               #'(lambda (segment posn)
644                   (declare (ignore posn))
645                   (emit-immediate-inst segment #b001111 0
646                                        (reg-tn-encoding lip-tn)
647                                        (ldb (byte 16 16)
648                                             (label-position target)))
649                   (emit-immediate-inst segment #b001101 0
650                                        (reg-tn-encoding lip-tn)
651                                        (ldb (byte 16 0)
652                                             (label-position target)))))
653             (emit-register-inst segment special-op (reg-tn-encoding lip-tn)
654                                 0 (if linked 31 0) 0
655                                 (if linked #b001001 #b001000))))))
656
657 (define-instruction b (segment target)
658   (:declare (type label target))
659   (:printer immediate ((op #b000100) (rs 0) (rt 0)
660                        (immediate nil :type 'relative-label))
661             '(:name :tab immediate))
662   (:attributes branch)
663   (:delay 1)
664   (:emitter
665    (emit-relative-branch segment #b000100 0 0 target)))
666
667 (define-instruction bal (segment target)
668   (:declare (type label target))
669   (:printer immediate ((op bcond-op) (rs 0) (rt #b01001)
670                        (immediate nil :type 'relative-label))
671             '(:name :tab immediate))
672   (:attributes branch)
673   (:dependencies (writes :r31))
674   (:delay 1)
675   (:emitter
676    (emit-relative-branch segment bcond-op 0 #b10001 target)))
677
678 (define-instruction beq (segment r1 r2-or-target &optional target)
679   (:declare (type tn r1)
680             (type (or tn fixnum label) r2-or-target)
681             (type (or label null) target))
682   (:printer immediate ((op #b000100) (immediate nil :type 'relative-label)))
683   (:attributes branch)
684   (:dependencies (reads r1) (if target (reads r2-or-target)))
685   (:delay 1)
686   (:emitter
687    (unless target
688      (setf target r2-or-target)
689      (setf r2-or-target 0))
690    (emit-relative-branch segment #b000100 r1 r2-or-target target)))
691
692 (define-instruction bne (segment r1 r2-or-target &optional target)
693   (:declare (type tn r1)
694             (type (or tn fixnum label) r2-or-target)
695             (type (or label null) target))
696   (:printer immediate ((op #b000101) (immediate nil :type 'relative-label)))
697   (:attributes branch)
698   (:dependencies (reads r1) (if target (reads r2-or-target)))
699   (:delay 1)
700   (:emitter
701    (unless target
702      (setf target r2-or-target)
703      (setf r2-or-target 0))
704    (emit-relative-branch segment #b000101 r1 r2-or-target target)))
705
706 (defconstant-eqx cond-branch-printer
707   '(:name :tab rs ", " immediate)
708   #'equalp)
709
710 (define-instruction blez (segment reg target)
711   (:declare (type label target) (type tn reg))
712   (:printer
713    immediate ((op #b000110) (rt 0) (immediate nil :type 'relative-label))
714             cond-branch-printer)
715   (:attributes branch)
716   (:dependencies (reads reg))
717   (:delay 1)
718   (:emitter
719    (emit-relative-branch segment #b000110 reg 0 target)))
720
721 (define-instruction bgtz (segment reg target)
722   (:declare (type label target) (type tn reg))
723   (:printer
724    immediate ((op #b000111) (rt 0) (immediate nil :type 'relative-label))
725             cond-branch-printer)
726   (:attributes branch)
727   (:dependencies (reads reg))
728   (:delay 1)
729   (:emitter
730    (emit-relative-branch segment #b000111 reg 0 target)))
731
732 (define-instruction bltz (segment reg target)
733   (:declare (type label target) (type tn reg))
734   (:printer
735    immediate ((op bcond-op) (rt 0) (immediate nil :type 'relative-label))
736             cond-branch-printer)
737   (:attributes branch)
738   (:dependencies (reads reg))
739   (:delay 1)
740   (:emitter
741    (emit-relative-branch segment bcond-op reg #b00000 target)))
742
743 (define-instruction bgez (segment reg target)
744   (:declare (type label target) (type tn reg))
745   (:printer
746    immediate ((op bcond-op) (rt 1) (immediate nil :type 'relative-label))
747             cond-branch-printer)
748   (:attributes branch)
749   (:dependencies (reads reg))
750   (:delay 1)
751   (:emitter
752    (emit-relative-branch segment bcond-op reg #b00001 target)))
753
754 (define-instruction bltzal (segment reg target)
755   (:declare (type label target) (type tn reg))
756   (:printer
757    immediate ((op bcond-op) (rt #b01000) (immediate nil :type 'relative-label))
758             cond-branch-printer)
759   (:attributes branch)
760   (:dependencies (reads reg) (writes :r31))
761   (:delay 1)
762   (:emitter
763    (emit-relative-branch segment bcond-op reg #b10000 target)))
764
765 (define-instruction bgezal (segment reg target)
766   (:declare (type label target) (type tn reg))
767   (:printer
768    immediate ((op bcond-op) (rt #b01001) (immediate nil :type 'relative-label))
769             cond-branch-printer)
770   (:attributes branch)
771   (:delay 1)
772   (:dependencies (reads reg) (writes :r31))
773   (:emitter
774    (emit-relative-branch segment bcond-op reg #b10001 target)))
775
776 (defconstant-eqx j-printer
777   '(:name :tab (:choose rs target))
778   #'equalp)
779
780 (define-instruction j (segment target)
781   (:declare (type (or tn fixup) target))
782   (:printer register ((op special-op) (rt 0) (rd 0) (funct #b001000))
783             j-printer)
784   (:printer jump ((op #b000010)) j-printer)
785   (:attributes branch)
786   (:dependencies (reads target))
787   (:delay 1)
788   (:emitter
789    (etypecase target
790      (tn
791       (emit-register-inst segment special-op (reg-tn-encoding target)
792                           0 0 0 #b001000))
793      (fixup
794       (note-fixup segment :jump target)
795       (emit-jump-inst segment #b000010 0)))))
796
797 (define-instruction jal (segment reg-or-target &optional target)
798   (:declare (type (or null tn fixup) target)
799             (type (or tn fixup (integer -16 31)) reg-or-target))
800   (:printer register ((op special-op) (rt 0) (funct #b001001)) j-printer)
801   (:printer jump ((op #b000011)) j-printer)
802   (:attributes branch)
803   (:dependencies (if target (writes reg-or-target) (writes :r31)))
804   (:delay 1)
805   (:emitter
806    (unless target
807      (setf target reg-or-target)
808      (setf reg-or-target 31))
809    (etypecase target
810      (tn
811       (emit-register-inst segment special-op (reg-tn-encoding target) 0
812                           reg-or-target 0 #b001001))
813      (fixup
814       (note-fixup segment :jump target)
815       (emit-jump-inst segment #b000011 0)))))
816
817 (define-instruction bc1f (segment target)
818   (:declare (type label target))
819   (:printer coproc-branch ((op cop1-op) (funct #x100)
820                            (offset nil :type 'relative-label)))
821   (:attributes branch)
822   (:dependencies (reads :float-status))
823   (:delay 1)
824   (:emitter
825    (emit-relative-branch segment cop1-op #b01000 #b00000 target)))
826
827 (define-instruction bc1t (segment target)
828   (:declare (type label target))
829   (:printer coproc-branch ((op cop1-op) (funct #x101)
830                            (offset nil :type 'relative-label)))
831   (:attributes branch)
832   (:dependencies (reads :float-status))
833   (:delay 1)
834   (:emitter
835    (emit-relative-branch segment cop1-op #b01000 #b00001 target)))
836
837
838 \f
839 ;;;; Random movement instructions.
840
841 (define-instruction lui (segment reg value)
842   (:declare (type tn reg)
843             (type (or fixup (signed-byte 16) (unsigned-byte 16)) value))
844   (:printer immediate ((op #b001111)
845                        (immediate nil :sign-extend nil :printer "#x~4,'0X")))
846   (:dependencies (writes reg))
847   (:delay 0)
848   (:emitter
849    (when (fixup-p value)
850      (note-fixup segment :lui value)
851      (setf value 0))
852    (emit-immediate-inst segment #b001111 0 (reg-tn-encoding reg) value)))
853
854 (defconstant-eqx mvsreg-printer '(:name :tab rd)
855   #'equalp)
856
857 (define-instruction mfhi (segment reg)
858   (:declare (type tn reg))
859   (:printer register ((op special-op) (rs 0) (rt 0) (funct #b010000))
860             mvsreg-printer)
861   (:dependencies (reads :hi-reg) (writes reg))
862   (:delay 2)
863   (:emitter
864    (emit-register-inst segment special-op 0 0 (reg-tn-encoding reg) 0
865                         #b010000)))
866
867 (define-instruction mthi (segment reg)
868   (:declare (type tn reg))
869   (:printer register ((op special-op) (rs 0) (rt 0) (funct #b010001))
870             mvsreg-printer)
871   (:dependencies (reads reg) (writes :hi-reg))
872   (:delay 0)
873   (:emitter
874    (emit-register-inst segment special-op 0 0 (reg-tn-encoding reg) 0
875                         #b010001)))
876
877 (define-instruction mflo (segment reg)
878   (:declare (type tn reg))
879   (:printer register ((op special-op) (rs 0) (rt 0) (funct #b010010))
880             mvsreg-printer)
881   (:dependencies (reads :low-reg) (writes reg))
882   (:delay 2)
883   (:emitter
884    (emit-register-inst segment special-op 0 0 (reg-tn-encoding reg) 0
885                         #b010010)))
886
887 (define-instruction mtlo (segment reg)
888   (:declare (type tn reg))
889   (:printer register ((op special-op) (rs 0) (rt 0) (funct #b010011))
890             mvsreg-printer)
891   (:dependencies (reads reg) (writes :low-reg))
892   (:delay 0)
893   (:emitter
894    (emit-register-inst segment special-op 0 0 (reg-tn-encoding reg) 0
895                         #b010011)))
896
897 (define-instruction move (segment dst src)
898   (:declare (type tn dst src))
899   (:printer register ((op special-op) (rt 0) (funct #b100001))
900             '(:name :tab rd ", " rs))
901   (:attributes flushable)
902   (:dependencies (reads src) (writes dst))
903   (:delay 0)
904   (:emitter
905    (emit-register-inst segment special-op (reg-tn-encoding src) 0
906                        (reg-tn-encoding dst) 0 #b100001)))
907
908 (define-instruction fmove (segment format dst src)
909   (:declare (type float-format format) (type tn dst src))
910   (:printer float ((funct #b000110)) '(:name "." format :tab fd ", " fs))
911   (:attributes flushable)
912   (:dependencies (reads src) (writes dst))
913   (:delay 0)
914   (:emitter
915    (emit-float-inst segment cop1-op 1 (float-format-value format) 0
916                     (fp-reg-tn-encoding src) (fp-reg-tn-encoding dst)
917                     #b000110)))
918
919 (defun %li (reg value)
920   (etypecase value
921     ((unsigned-byte 16)
922      (inst or reg zero-tn value))
923     ((signed-byte 16)
924      (inst addu reg zero-tn value))
925     ((or (signed-byte 32) (unsigned-byte 32))
926      (inst lui reg (ldb (byte 16 16) value))
927      (inst or reg (ldb (byte 16 0) value)))
928     (fixup
929      (inst lui reg value)
930      (inst addu reg value))))
931
932 (define-instruction-macro li (reg value)
933   `(%li ,reg ,value))
934
935 (defconstant-eqx sub-op-printer '(:name :tab rd ", " rt) #'equalp)
936
937 (define-instruction mtc1 (segment to from)
938   (:declare (type tn to from))
939   (:printer register ((op cop1-op) (rs #b00100) (funct 0)) sub-op-printer)
940   (:dependencies (reads from) (writes to))
941   (:delay 1)
942   (:emitter
943    (emit-register-inst segment cop1-op #b00100 (reg-tn-encoding from)
944                        (fp-reg-tn-encoding to) 0 0)))
945
946 (define-instruction mtc1-odd (segment to from)
947   (:declare (type tn to from))
948   (:dependencies (reads from) (writes to))
949   (:delay 1)
950   (:emitter
951    (emit-register-inst segment cop1-op #b00100 (reg-tn-encoding from)
952                        (1+ (fp-reg-tn-encoding to)) 0 0)))
953
954 (define-instruction mfc1 (segment to from)
955   (:declare (type tn to from))
956   (:printer register ((op cop1-op) (rs 0) (rd nil :type 'fp-reg) (funct 0))
957             sub-op-printer)
958   (:dependencies (reads from) (writes to))
959   (:delay 1)
960   (:emitter
961    (emit-register-inst segment cop1-op #b00000 (reg-tn-encoding to)
962                        (fp-reg-tn-encoding from) 0 0)))
963
964 (define-instruction mfc1-odd (segment to from)
965   (:declare (type tn to from))
966   (:dependencies (reads from) (writes to))
967   (:delay 1)
968   (:emitter
969    (emit-register-inst segment cop1-op #b00000 (reg-tn-encoding to)
970                        (1+ (fp-reg-tn-encoding from)) 0 0)))
971
972 (define-instruction mfc1-odd2 (segment to from)
973   (:declare (type tn to from))
974   (:dependencies (reads from) (writes to))
975   (:delay 1)
976   (:emitter
977    (emit-register-inst segment cop1-op #b00000 (1+ (reg-tn-encoding to))
978                        (fp-reg-tn-encoding from) 0 0)))
979
980 (define-instruction mfc1-odd3 (segment to from)
981   (:declare (type tn to from))
982   (:dependencies (reads from) (writes to))
983   (:delay 1)
984   (:emitter
985    (emit-register-inst segment cop1-op #b00000 (1+ (reg-tn-encoding to))
986                        (1+ (fp-reg-tn-encoding from)) 0 0)))
987
988 (define-instruction cfc1 (segment reg cr)
989   (:declare (type tn reg) (type (unsigned-byte 5) cr))
990   (:printer register ((op cop1-op) (rs #b00010) (rd nil :type 'control-reg)
991                       (funct 0)) sub-op-printer)
992   (:dependencies (reads :ctrl-stat-reg) (writes reg))
993   (:delay 1)
994   (:emitter
995    (emit-register-inst segment cop1-op #b00010 (reg-tn-encoding reg)
996                        cr 0 0)))
997
998 (define-instruction ctc1 (segment reg cr)
999   (:declare (type tn reg) (type (unsigned-byte 5) cr))
1000   (:printer register ((op cop1-op) (rs #b00110) (rd nil :type 'control-reg)
1001                       (funct 0)) sub-op-printer)
1002   (:dependencies (reads reg) (writes :ctrl-stat-reg))
1003   (:delay 1)
1004   (:emitter
1005    (emit-register-inst segment cop1-op #b00110 (reg-tn-encoding reg)
1006                        cr 0 0)))
1007
1008
1009 \f
1010 ;;;; Random system hackery and other noise
1011
1012 (define-instruction-macro entry-point ()
1013   nil)
1014
1015 (defun snarf-error-junk (sap offset &optional length-only)
1016   (let* ((length (sb!sys:sap-ref-8 sap offset))
1017          (vector (make-array length :element-type '(unsigned-byte 8))))
1018     (declare (type sb!sys:system-area-pointer sap)
1019              (type (unsigned-byte 8) length)
1020              (type (simple-array (unsigned-byte 8) (*)) vector))
1021     (cond (length-only
1022            (values 0 (1+ length) nil nil))
1023           (t
1024            (sb!kernel:copy-ub8-from-system-area sap (1+ offset)
1025                                                 vector 0 length)
1026            (collect ((sc-offsets)
1027                      (lengths))
1028              (lengths 1)                ; the length byte
1029              (let* ((index 0)
1030                     (error-number (sb!c:read-var-integer vector index)))
1031                (lengths index)
1032                (loop
1033                  (when (>= index length)
1034                    (return))
1035                  (let ((old-index index))
1036                    (sc-offsets (sb!c:read-var-integer vector index))
1037                    (lengths (- index old-index))))
1038                (values error-number
1039                        (1+ length)
1040                        (sc-offsets)
1041                        (lengths))))))))
1042
1043 (defmacro break-cases (breaknum &body cases)
1044   (let ((bn-temp (gensym)))
1045     (collect ((clauses))
1046       (dolist (case cases)
1047         (clauses `((= ,bn-temp ,(car case)) ,@(cdr case))))
1048       `(let ((,bn-temp ,breaknum))
1049          (cond ,@(clauses))))))
1050
1051 (defun break-control (chunk inst stream dstate)
1052   (declare (ignore inst))
1053   (flet ((nt (x) (if stream (sb!disassem:note x dstate))))
1054     (case (break-code chunk dstate)
1055       (#.error-trap
1056        (nt "Error trap")
1057        (sb!disassem:handle-break-args #'snarf-error-junk stream dstate))
1058       (#.cerror-trap
1059        (nt "Cerror trap")
1060        (sb!disassem:handle-break-args #'snarf-error-junk stream dstate))
1061       (#.breakpoint-trap
1062        (nt "Breakpoint trap"))
1063       (#.pending-interrupt-trap
1064        (nt "Pending interrupt trap"))
1065       (#.halt-trap
1066        (nt "Halt trap"))
1067       (#.fun-end-breakpoint-trap
1068        (nt "Function end breakpoint trap"))
1069     )))
1070
1071 (define-instruction break (segment code &optional (subcode 0))
1072   (:declare (type (unsigned-byte 10) code subcode))
1073   (:printer break ((op special-op) (funct #b001101))
1074             '(:name :tab code (:unless (:constant 0) subcode))
1075             :control #'break-control )
1076   :pinned
1077   (:cost 0)
1078   (:delay 0)
1079   (:emitter
1080    (emit-break-inst segment special-op code subcode #b001101)))
1081
1082 (define-instruction syscall (segment)
1083   (:printer register ((op special-op) (rd 0) (rt 0) (rs 0) (funct #b001110))
1084             '(:name))
1085   :pinned
1086   (:delay 0)
1087   (:emitter
1088    (emit-register-inst segment special-op 0 0 0 0 #b001110)))
1089
1090 (define-instruction nop (segment)
1091   (:printer register ((op 0) (rd 0) (rd 0) (rs 0) (funct 0)) '(:name))
1092   (:attributes flushable)
1093   (:delay 0)
1094   (:emitter
1095    (emit-word segment 0)))
1096
1097 (!def-vm-support-routine emit-nop (segment)
1098   (emit-word segment 0))
1099
1100 (define-instruction word (segment word)
1101   (:declare (type (or (unsigned-byte 32) (signed-byte 32)) word))
1102   :pinned
1103   (:cost 0)
1104   (:delay 0)
1105   (:emitter
1106    (emit-word segment word)))
1107
1108 (define-instruction short (segment short)
1109   (:declare (type (or (unsigned-byte 16) (signed-byte 16)) short))
1110   :pinned
1111   (:cost 0)
1112   (:delay 0)
1113   (:emitter
1114    (emit-short segment short)))
1115
1116 (define-instruction byte (segment byte)
1117   (:declare (type (or (unsigned-byte 8) (signed-byte 8)) byte))
1118   :pinned
1119   (:cost 0)
1120   (:delay 0)
1121   (:emitter
1122    (emit-byte segment byte)))
1123
1124
1125 (defun emit-header-data (segment type)
1126   (emit-back-patch
1127    segment 4
1128    #'(lambda (segment posn)
1129        (emit-word segment
1130                   (logior type
1131                           (ash (+ posn (component-header-length))
1132                                (- n-widetag-bits word-shift)))))))
1133
1134 (define-instruction fun-header-word (segment)
1135   :pinned
1136   (:cost 0)
1137   (:delay 0)
1138   (:emitter
1139    (emit-header-data segment simple-fun-header-widetag)))
1140
1141 (define-instruction lra-header-word (segment)
1142   :pinned
1143   (:cost 0)
1144   (:delay 0)
1145   (:emitter
1146    (emit-header-data segment return-pc-header-widetag)))
1147
1148
1149 (defun emit-compute-inst (segment vop dst src label temp calc)
1150   (emit-chooser
1151    ;; We emit either 12 or 4 bytes, so we maintain 8 byte alignments.
1152    segment 12 3
1153    #'(lambda (segment posn delta-if-after)
1154        (let ((delta (funcall calc label posn delta-if-after)))
1155           (when (<= (- (ash 1 15)) delta (1- (ash 1 15)))
1156             (emit-back-patch segment 4
1157                              #'(lambda (segment posn)
1158                                  (assemble (segment vop)
1159                                            (inst addu dst src
1160                                                  (funcall calc label posn 0)))))
1161             t)))
1162    #'(lambda (segment posn)
1163        (let ((delta (funcall calc label posn 0)))
1164          (assemble (segment vop)
1165                    (inst lui temp (ldb (byte 16 16) delta))
1166                    (inst or temp (ldb (byte 16 0) delta))
1167                    (inst addu dst src temp))))))
1168
1169 ;; code = fn - header - label-offset + other-pointer-tag
1170 (define-instruction compute-code-from-fn (segment dst src label temp)
1171   (:declare (type tn dst src temp) (type label label))
1172   (:attributes variable-length)
1173   (:dependencies (reads src) (writes dst) (writes temp))
1174   (:delay 0)
1175   (:vop-var vop)
1176   (:emitter
1177    (emit-compute-inst segment vop dst src label temp
1178                       #'(lambda (label posn delta-if-after)
1179                           (- other-pointer-lowtag
1180                              (label-position label posn delta-if-after)
1181                              (component-header-length))))))
1182
1183 ;; code = lra - other-pointer-tag - header - label-offset + other-pointer-tag
1184 ;;      = lra - (header + label-offset)
1185 (define-instruction compute-code-from-lra (segment dst src label temp)
1186   (:declare (type tn dst src temp) (type label label))
1187   (:attributes variable-length)
1188   (:dependencies (reads src) (writes dst) (writes temp))
1189   (:delay 0)
1190   (:vop-var vop)
1191   (:emitter
1192    (emit-compute-inst segment vop dst src label temp
1193                       #'(lambda (label posn delta-if-after)
1194                           (- (+ (label-position label posn delta-if-after)
1195                                 (component-header-length)))))))
1196
1197 ;; lra = code + other-pointer-tag + header + label-offset - other-pointer-tag
1198 (define-instruction compute-lra-from-code (segment dst src label temp)
1199   (:declare (type tn dst src temp) (type label label))
1200   (:attributes variable-length)
1201   (:dependencies (reads src) (writes dst) (writes temp))
1202   (:delay 0)
1203   (:vop-var vop)
1204   (:emitter
1205    (emit-compute-inst segment vop dst src label temp
1206                       #'(lambda (label posn delta-if-after)
1207                           (+ (label-position label posn delta-if-after)
1208                              (component-header-length))))))
1209
1210 \f
1211 ;;;; Loads and Stores
1212
1213 (defun emit-load/store-inst (segment opcode reg base index
1214                                      &optional (oddhack 0))
1215   (when (fixup-p index)
1216     (note-fixup segment :addi index)
1217     (setf index 0))
1218   (emit-immediate-inst segment opcode (reg-tn-encoding reg)
1219                        (+ (reg-tn-encoding base) oddhack) index))
1220
1221 (defconstant-eqx load-store-printer
1222   '(:name :tab
1223           rt ", "
1224           rs
1225           (:unless (:constant 0) "[" immediate "]"))
1226   #'equalp)
1227
1228 (define-instruction lb (segment reg base &optional (index 0))
1229   (:declare (type tn reg base)
1230             (type (or (signed-byte 16) fixup) index))
1231   (:printer immediate ((op #b100000)) load-store-printer)
1232   (:dependencies (reads base) (reads :memory) (writes reg))
1233   (:delay 1)
1234   (:emitter
1235    (emit-load/store-inst segment #b100000 base reg index)))
1236
1237 (define-instruction lh (segment reg base &optional (index 0))
1238   (:declare (type tn reg base)
1239             (type (or (signed-byte 16) fixup) index))
1240   (:printer immediate ((op #b100001)) load-store-printer)
1241   (:dependencies (reads base) (reads :memory) (writes reg))
1242   (:delay 1)
1243   (:emitter
1244    (emit-load/store-inst segment #b100001 base reg index)))
1245
1246 (define-instruction lwl (segment reg base &optional (index 0))
1247   (:declare (type tn reg base)
1248             (type (or (signed-byte 16) fixup) index))
1249   (:printer immediate ((op #b100010)) load-store-printer)
1250   (:dependencies (reads base) (reads :memory) (writes reg))
1251   (:delay 1)
1252   (:emitter
1253    (emit-load/store-inst segment #b100010 base reg index)))
1254
1255 (define-instruction lw (segment reg base &optional (index 0))
1256   (:declare (type tn reg base)
1257             (type (or (signed-byte 16) fixup) index))
1258   (:printer immediate ((op #b100011)) load-store-printer)
1259   (:dependencies (reads base) (reads :memory) (writes reg))
1260   (:delay 1)
1261   (:emitter
1262    (emit-load/store-inst segment #b100011 base reg index)))
1263
1264 ;; next is just for ease of coding double-in-int c-call convention
1265 (define-instruction lw-odd (segment reg base &optional (index 0))
1266   (:declare (type tn reg base)
1267             (type (or (signed-byte 16) fixup) index))
1268   (:dependencies (reads base) (reads :memory) (writes reg))
1269   (:delay 1)
1270   (:emitter
1271    (emit-load/store-inst segment #b100011 base reg index 1)))
1272
1273 (define-instruction lbu (segment reg base &optional (index 0))
1274   (:declare (type tn reg base)
1275             (type (or (signed-byte 16) fixup) index))
1276   (:printer immediate ((op #b100100)) load-store-printer)
1277   (:dependencies (reads base) (reads :memory) (writes reg))
1278   (:delay 1)
1279   (:emitter
1280    (emit-load/store-inst segment #b100100 base reg index)))
1281
1282 (define-instruction lhu (segment reg base &optional (index 0))
1283   (:declare (type tn reg base)
1284             (type (or (signed-byte 16) fixup) index))
1285   (:printer immediate ((op #b100101)) load-store-printer)
1286   (:dependencies (reads base) (reads :memory) (writes reg))
1287   (:delay 1)
1288   (:emitter
1289    (emit-load/store-inst segment #b100101 base reg index)))
1290
1291 (define-instruction lwr (segment reg base &optional (index 0))
1292   (:declare (type tn reg base)
1293             (type (or (signed-byte 16) fixup) index))
1294   (:printer immediate ((op #b100110)) load-store-printer)
1295   (:dependencies (reads base) (reads :memory) (writes reg))
1296   (:delay 1)
1297   (:emitter
1298    (emit-load/store-inst segment #b100110 base reg index)))
1299
1300 (define-instruction sb (segment reg base &optional (index 0))
1301   (:declare (type tn reg base)
1302             (type (or (signed-byte 16) fixup) index))
1303   (:printer immediate ((op #b101000)) load-store-printer)
1304   (:dependencies (reads base) (reads reg) (writes :memory))
1305   (:delay 0)
1306   (:emitter
1307    (emit-load/store-inst segment #b101000 base reg index)))
1308
1309 (define-instruction sh (segment reg base &optional (index 0))
1310   (:declare (type tn reg base)
1311             (type (or (signed-byte 16) fixup) index))
1312   (:printer immediate ((op #b101001)) load-store-printer)
1313   (:dependencies (reads base) (reads reg) (writes :memory))
1314   (:delay 0)
1315   (:emitter
1316    (emit-load/store-inst segment #b101001 base reg index)))
1317
1318 (define-instruction swl (segment reg base &optional (index 0))
1319   (:declare (type tn reg base)
1320             (type (or (signed-byte 16) fixup) index))
1321   (:printer immediate ((op #b101010)) load-store-printer)
1322   (:dependencies (reads base) (reads reg) (writes :memory))
1323   (:delay 0)
1324   (:emitter
1325    (emit-load/store-inst segment #b101010 base reg index)))
1326
1327 (define-instruction sw (segment reg base &optional (index 0))
1328   (:declare (type tn reg base)
1329             (type (or (signed-byte 16) fixup) index))
1330   (:printer immediate ((op #b101011)) load-store-printer)
1331   (:dependencies (reads base) (reads reg) (writes :memory))
1332   (:delay 0)
1333   (:emitter
1334    (emit-load/store-inst segment #b101011 base reg index)))
1335
1336 (define-instruction swr (segment reg base &optional (index 0))
1337   (:declare (type tn reg base)
1338             (type (or (signed-byte 16) fixup) index))
1339   (:printer immediate ((op #b101110)) load-store-printer)
1340   (:dependencies (reads base) (reads reg) (writes :memory))
1341   (:delay 0)
1342   (:emitter
1343    (emit-load/store-inst segment #b101110 base reg index)))
1344
1345
1346 (defun emit-fp-load/store-inst (segment opcode reg odd base index)
1347   (when (fixup-p index)
1348     (note-fixup segment :addi index)
1349     (setf index 0))
1350   (emit-immediate-inst segment opcode (reg-tn-encoding base)
1351                        (+ (fp-reg-tn-encoding reg) odd) index))
1352
1353 (define-instruction lwc1 (segment reg base &optional (index 0))
1354   (:declare (type tn reg base)
1355             (type (or (signed-byte 16) fixup) index))
1356   (:printer immediate ((op #b110001) (rt nil :type 'fp-reg)) load-store-printer)
1357   (:dependencies (reads base) (reads :memory) (writes reg))
1358   (:delay 1)
1359   (:emitter
1360    (emit-fp-load/store-inst segment #b110001 reg 0 base index)))
1361
1362 (define-instruction lwc1-odd (segment reg base &optional (index 0))
1363   (:declare (type tn reg base)
1364             (type (or (signed-byte 16) fixup) index))
1365   (:dependencies (reads base) (reads :memory) (writes reg))
1366   (:delay 1)
1367   (:emitter
1368    (emit-fp-load/store-inst segment #b110001 reg 1 base index)))
1369
1370 (define-instruction swc1 (segment reg base &optional (index 0))
1371   (:declare (type tn reg base)
1372             (type (or (signed-byte 16) fixup) index))
1373   (:printer immediate ((op #b111001) (rt nil :type 'fp-reg)) load-store-printer)
1374   (:dependencies (reads base) (reads reg) (writes :memory))
1375   (:delay 0)
1376   (:emitter
1377    (emit-fp-load/store-inst segment #b111001 reg 0 base index)))
1378
1379 (define-instruction swc1-odd (segment reg base &optional (index 0))
1380   (:declare (type tn reg base)
1381             (type (or (signed-byte 16) fixup) index))
1382   (:dependencies (reads base) (reads reg) (writes :memory))
1383   (:delay 0)
1384   (:emitter
1385    (emit-fp-load/store-inst segment #b111001 reg 1 base index)))
1386