;;;; allocating simple objects ;;;; This software is part of the SBCL system. See the README file for ;;;; more information. ;;;; ;;;; This software is derived from the CMU CL system, which was ;;;; written at Carnegie Mellon University and released into the ;;;; public domain. The software is in the public domain and is ;;;; provided with absolutely no warranty. See the COPYING and CREDITS ;;;; files for more information. (in-package "SB!VM") ;;;; Signed and unsigned bignums from word-sized integers. Argument ;;;; and return in the same register. No VOPs, as these are only used ;;;; as out-of-line versions: MOVE-FROM-[UN]SIGNED VOPs handle the ;;;; fixnum cases inline. ;;; #+SB-ASSEMBLING as we don't need VOPS, just the asm routines: ;;; these are out-of-line versions called by VOPs. #+sb-assembling (macrolet ((def (reg) `(define-assembly-routine (,(symbolicate "ALLOC-SIGNED-BIGNUM-IN-" reg)) ((:temp number unsigned-reg ,(symbolicate reg "-OFFSET"))) (inst push number) (with-fixed-allocation (number bignum-widetag (+ bignum-digits-offset 1)) (popw number bignum-digits-offset other-pointer-lowtag)) (inst ret)))) (def rax) (def rcx) (def rdx) (def rbx) (def rsi) (def rdi) (def r8) (def r9) (def r10) (def r12) (def r13) (def r14) (def r15)) #+sb-assembling (macrolet ((def (reg) `(define-assembly-routine (,(symbolicate "ALLOC-UNSIGNED-BIGNUM-IN-" reg)) ((:temp number unsigned-reg ,(symbolicate reg "-OFFSET"))) (inst push number) (inst jmp :ns one-word-bignum) ;; Two word bignum (with-fixed-allocation (number bignum-widetag (+ bignum-digits-offset 2)) (popw number bignum-digits-offset other-pointer-lowtag)) (inst ret) ONE-WORD-BIGNUM (with-fixed-allocation (number bignum-widetag (+ bignum-digits-offset 1)) (popw number bignum-digits-offset other-pointer-lowtag)) (inst ret)))) (def rax) (def rcx) (def rdx) (def rbx) (def rsi) (def rdi) (def r8) (def r9) (def r10) (def r12) (def r13) (def r14) (def r15)) #+sb-assembling (macrolet ((def (reg) (declare (ignorable reg)) #!+sb-thread (let* ((name (intern (format nil "ALLOC-TLS-INDEX-IN-~A" reg))) (target-offset (intern (format nil "~A-OFFSET" reg))) (other-offset (if (eql 'rax reg) 'rcx-offset 'rax-offset))) ;; Symbol starts in TARGET, where the TLS-INDEX ends up in. `(define-assembly-routine ,name ((:temp other descriptor-reg ,other-offset) (:temp target descriptor-reg ,target-offset)) (let ((get-tls-index-lock (gen-label)) (release-tls-index-lock (gen-label))) (pseudo-atomic ;; Save OTHER & push the symbol. RAX is either one of the two. (inst push other) (inst push target) (emit-label get-tls-index-lock) (let ((not-rax ,(if (eql 'rax reg) 'other 'target))) (inst mov not-rax 1) (zeroize rax-tn) (inst cmpxchg (make-ea-for-symbol-value *tls-index-lock*) not-rax :lock) (inst jmp :ne get-tls-index-lock)) ;; The symbol is now in OTHER. (inst pop other) ;; Now with the lock held, see if the symbol's tls index has been ;; set in the meantime. (loadw target other symbol-tls-index-slot other-pointer-lowtag) (inst test target target) (inst jmp :ne release-tls-index-lock) ;; Allocate a new tls-index. (load-symbol-value target *free-tls-index*) (let ((not-error (gen-label)) (error (generate-error-code nil 'tls-exhausted-error))) (inst cmp target (ash tls-size word-shift)) (inst jmp :l not-error) (%clear-pseudo-atomic) (inst jmp error) (emit-label not-error)) (inst add (make-ea-for-symbol-value *free-tls-index*) n-word-bytes) (storew target other symbol-tls-index-slot other-pointer-lowtag) (emit-label release-tls-index-lock) ;; No need for barriers on x86/x86-64 on unlock. (store-symbol-value 0 *tls-index-lock*) ;; Restore OTHER. (inst pop other)) (inst ret)))))) (def rax) (def rcx) (def rdx) (def rbx) (def rsi) (def rdi) (def r8) (def r9) (def r10) (def r12) (def r13) (def r14) (def r15))