1.0.24.27: target-thread cosmetics
[sbcl.git] / src / assembly / x86 / arith.lisp
index 33c0439..535e023 100644 (file)
                  (:res res (descriptor-reg any-reg) edx-offset)
 
                  (:temp eax unsigned-reg eax-offset)
-                 (:temp ebx unsigned-reg ebx-offset)
                  (:temp ecx unsigned-reg ecx-offset))
 
-                (declare (ignorable ebx))
+                (inst mov ecx x)
+                (inst or ecx y)
+                (inst test ecx fixnum-tag-mask)  ; both fixnums?
+                (inst jmp :nz DO-STATIC-FUN)     ; no - do generic
 
-                (inst test x 3)  ; fixnum?
-                (inst jmp :nz DO-STATIC-FUN) ; no - do generic
-                (inst test y 3)  ; fixnum?
-                (inst jmp :z DO-BODY)   ; yes - doit here
+                ,@body
+                (inst clc) ; single-value return
+                (inst ret)
 
                 DO-STATIC-FUN
                 (inst pop eax)
                       (make-ea :dword
                                :disp (+ nil-value
                                         (static-fun-offset
-                                         ',(symbolicate "TWO-ARG-" fun)))))
-
-                DO-BODY
-                ,@body)))
+                                         ',(symbolicate "TWO-ARG-" fun))))))))
 
   (define-generic-arith-routine (+ 10)
     (move res x)
 
   (define-generic-arith-routine (* 30)
     (move eax x)                          ; must use eax for 64-bit result
-    (inst sar eax 2)                  ; remove *4 fixnum bias
-    (inst imul y)                        ; result in edx:eax
-    (inst jmp :no okay)            ; still fixnum
+    (inst sar eax n-fixnum-tag-bits)      ; remove *4 fixnum bias
+    (inst imul y)                         ; result in edx:eax
+    (inst jmp :no okay)                   ; still fixnum
 
     ;; zzz jrd changed edx to ebx in here, as edx isn't listed as a temp, above
     ;;     pfw says that loses big -- edx is target for arg x and result res
     ;;     note that 'edx' is not defined -- using x
-    (inst shrd eax x 2)            ; high bits from edx
-    (inst sar x 2)                      ; now shift edx too
+    (inst shrd eax x n-fixnum-tag-bits)    ; high bits from edx
+    (inst sar x n-fixnum-tag-bits)         ; now shift edx too
 
-    (move ecx x)                          ; save high bits from cdq
-    (inst cdq)                      ; edx:eax <- sign-extend of eax
+    (move ecx x)                           ; save high bits from cdq
+    (inst cdq)                             ; edx:eax <- sign-extend of eax
     (inst cmp x ecx)
     (inst jmp :e SINGLE-WORD-BIGNUM)
 
 
                           (:temp eax unsigned-reg eax-offset)
                           (:temp ecx unsigned-reg ecx-offset))
-  (inst test x 3)
+  (inst test x fixnum-tag-mask)
   (inst jmp :z FIXNUM)
 
   (inst pop eax)
   (move res x)
   (inst neg res)                        ; (- most-negative-fixnum) is BIGNUM
   (inst jmp :no OKAY)
-  (inst shr res 2)                    ; sign bit is data - remove type bits
+  (inst shr res n-fixnum-tag-bits)      ; sign bit is data - remove type bits
   (move ecx res)
 
   (with-fixed-allocation (res bignum-widetag (1+ bignum-digits-offset))
                  (:temp eax unsigned-reg eax-offset)
                  (:temp ecx unsigned-reg ecx-offset))
 
-                ;; KLUDGE: The "3" here is a mask for the bits which will be
-                ;; zero in a fixnum. It should have a symbolic name. (Actually,
-                ;; it might already have a symbolic name which the coder
-                ;; couldn't be bothered to use..) -- WHN 19990917
-                (inst test x 3)
-                (inst jmp :nz TAIL-CALL-TO-STATIC-FN)
-                (inst test y 3)
-                (inst jmp :z INLINE-FIXNUM-COMPARE)
+                (inst mov ecx x)
+                (inst or ecx y)
+                (inst test ecx fixnum-tag-mask)
+                (inst jmp :nz DO-STATIC-FUN)  ; are both fixnums?
+
+                (inst cmp x y)
+                (cond ((member :cmov *backend-subfeatures*)
+                       (load-symbol res t)
+                       (inst mov eax nil-value)
+                       (inst cmov ,test res eax))
+                      (t
+                       (inst mov res nil-value)
+                       (inst jmp ,test RETURN)
+                       (load-symbol res t)))
+                RETURN
+                (inst clc)     ; single-value return
+                (inst ret)
 
-                TAIL-CALL-TO-STATIC-FN
+                DO-STATIC-FUN
                 (inst pop eax)
                 (inst push ebp-tn)
                 (inst lea ebp-tn (make-ea :dword
                                         ; should be named parallelly.
                 (inst jmp (make-ea :dword
                                    :disp (+ nil-value
-                                            (static-fun-offset ',static-fn))))
-
-                INLINE-FIXNUM-COMPARE
-                (inst cmp x y)
-                (inst mov res nil-value)
-                (inst jmp ,test RETURN-FALSE)
-
-                (load-symbol res t)
-
-                RETURN-FALSE
-                DONE)))
+                                            (static-fun-offset ',static-fn)))))))
 
   (define-cond-assem-rtn generic-< < two-arg-< :ge)
   (define-cond-assem-rtn generic-> > two-arg-> :le))
 
                           (:temp eax unsigned-reg eax-offset)
                           (:temp ecx unsigned-reg ecx-offset))
-  (inst cmp x y)
-  (inst jmp :e RETURN-T)
-  (inst test x 3)
-  (inst jmp :z RETURN-NIL)
-  (inst test y 3)
-  (inst jmp :nz DO-STATIC-FN)
+  (inst mov ecx x)
+  (inst and ecx y)
+  (inst test ecx fixnum-tag-mask)
+  (inst jmp :nz DO-STATIC-FUN)
 
-  RETURN-NIL
-  (inst mov res nil-value)
-  (inst jmp DONE)
-
-  DO-STATIC-FN
+  ;; At least one fixnum
+  (inst cmp x y)
+  (load-symbol res t)
+  (cond ((member :cmov *backend-subfeatures*)
+         (inst mov eax nil-value)
+         (inst cmov :ne res eax))
+        (t
+         (inst jmp :e RETURN)
+         (inst mov res nil-value)))
+  RETURN
+  (inst clc)
+  (inst ret)
+
+  ;; FIXME: We could handle all non-numbers here easily enough: go to
+  ;; TWO-ARG-EQL only if lowtags and widetags match, lowtag is
+  ;; other-pointer-lowtag and widetag is < code-header-widetag.
+  DO-STATIC-FUN
   (inst pop eax)
   (inst push ebp-tn)
   (inst lea ebp-tn (make-ea :dword :base esp-tn :disp n-word-bytes))
   (inst push eax)
   (inst mov ecx (fixnumize 2))
   (inst jmp (make-ea :dword
-                     :disp (+ nil-value (static-fun-offset 'eql))))
-
-  RETURN-T
-  (load-symbol res t)
-
-  DONE)
+                     :disp (+ nil-value (static-fun-offset 'eql)))))
 
 (define-assembly-routine (generic-=
                           (:cost 10)
                           (:res res descriptor-reg edx-offset)
 
                           (:temp eax unsigned-reg eax-offset)
-                          (:temp ecx unsigned-reg ecx-offset)
-                          )
-  (inst test x 3)                      ; descriptor?
-  (inst jmp :nz DO-STATIC-FN)          ; yes, do it here
-  (inst test y 3)                      ; descriptor?
-  (inst jmp :nz DO-STATIC-FN)
-  (inst cmp x y)
-  (inst jmp :e RETURN-T)                ; ok
-
-  (inst mov res nil-value)
-  (inst jmp DONE)
+                          (:temp ecx unsigned-reg ecx-offset))
+  (inst mov ecx x)
+  (inst or ecx y)
+  (inst test ecx fixnum-tag-mask)        ; both fixnums?
+  (inst jmp :nz DO-STATIC-FUN)
 
-  DO-STATIC-FN
+  (inst cmp x y)
+  (load-symbol res t)
+  (cond ((member :cmov *backend-subfeatures*)
+         (inst mov eax nil-value)
+         (inst cmov :ne res eax))
+        (t
+         (inst jmp :e RETURN)
+         (inst mov res nil-value)))
+  RETURN
+  (inst clc)
+  (inst ret)
+
+  DO-STATIC-FUN
   (inst pop eax)
   (inst push ebp-tn)
   (inst lea ebp-tn (make-ea :dword :base esp-tn :disp n-word-bytes))
   (inst push eax)
   (inst mov ecx (fixnumize 2))
   (inst jmp (make-ea :dword
-                     :disp (+ nil-value (static-fun-offset 'two-arg-=))))
-
-  RETURN-T
-  (load-symbol res t)
-
-  DONE)
+                     :disp (+ nil-value (static-fun-offset 'two-arg-=)))))
 
 \f
 ;;; Support for the Mersenne Twister, MT19937, random number generator