HEX
Server: Apache/2.4.41 (Ubuntu)
System: Linux vps-320ddf0a 5.4.0-216-generic #236-Ubuntu SMP Fri Apr 11 19:53:21 UTC 2025 x86_64
User: lamphim (1008)
PHP: 7.4.3-4ubuntu2.29
Disabled: pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals,pcntl_unshare,
Upload Files
File: //proc/self/root/proc/thread-self/root/usr/share/guile/2.2/language/cps/specialize-primcalls.scm
;;; Continuation-passing style (CPS) intermediate language (IL)

;; Copyright (C) 2013, 2014, 2015 Free Software Foundation, Inc.

;;;; This library is free software; you can redistribute it and/or
;;;; modify it under the terms of the GNU Lesser General Public
;;;; License as published by the Free Software Foundation; either
;;;; version 3 of the License, or (at your option) any later version.
;;;;
;;;; This library is distributed in the hope that it will be useful,
;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
;;;; Lesser General Public License for more details.
;;;;
;;;; You should have received a copy of the GNU Lesser General Public
;;;; License along with this library; if not, write to the Free Software
;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA

;;; Commentary:
;;;
;;; Some bytecode operations can encode an immediate as an operand.
;;; This pass tranforms generic primcalls to these specialized
;;; primcalls, if possible.
;;;
;;; Code:

(define-module (language cps specialize-primcalls)
  #:use-module (ice-9 match)
  #:use-module (language cps)
  #:use-module (language cps utils)
  #:use-module (language cps intmap)
  #:export (specialize-primcalls))

(define (specialize-primcalls conts)
  (let ((constants (compute-constant-values conts)))
    (define (u6? var)
      (let ((val (intmap-ref constants var (lambda (_) #f))))
        (and (exact-integer? val) (<= 0 val 63))))
    (define (u8? var)
      (let ((val (intmap-ref constants var (lambda (_) #f))))
        (and (exact-integer? val) (<= 0 val 255))))
    (define (u64? var)
      (let ((val (intmap-ref constants var (lambda (_) #f))))
        (and (exact-integer? val) (<= 0 val #xffffFFFFffffFFFF))))
    (define (s64? var)
      (let ((val (intmap-ref constants var (lambda (_) #f))))
        (and (exact-integer? val)
             (<= (- #x8000000000000000) val #x7fffFFFFffffFFFF))))
    (define (f64? var)
      (let ((val (intmap-ref constants var (lambda (_) #f))))
        (and (number? val) (inexact? val) (real? val))))
    (define (specialize-primcall name args)
      (define (rename name)
        (build-exp ($primcall name args)))
      (match (cons name args)
        (('make-vector (? u8? n) init) (rename 'make-vector/immediate))
        (('vector-ref v (? u8? n)) (rename 'vector-ref/immediate))
        (('vector-set! v (? u8? n) x) (rename 'vector-set!/immediate))
        (('allocate-struct v (? u8? n)) (rename 'allocate-struct/immediate))
        (('struct-ref s (? u8? n)) (rename 'struct-ref/immediate))
        (('struct-set! s (? u8? n) x) (rename 'struct-set!/immediate))
        (('add x (? u8? y)) (build-exp ($primcall 'add/immediate (x y))))
        (('add (? u8? x) y) (build-exp ($primcall 'add/immediate (y x))))
        (('sub x (? u8? y)) (build-exp ($primcall 'sub/immediate (x y))))
        (('uadd x (? u8? y)) (build-exp ($primcall 'uadd/immediate (x y))))
        (('uadd (? u8? x) y) (build-exp ($primcall 'uadd/immediate (y x))))
        (('usub x (? u8? y)) (build-exp ($primcall 'usub/immediate (x y))))
        (('umul x (? u8? y)) (build-exp ($primcall 'umul/immediate (x y))))
        (('umul (? u8? x) y) (build-exp ($primcall 'umul/immediate (y x))))
        (('ursh x (? u6? y)) (build-exp ($primcall 'ursh/immediate (x y))))
        (('ulsh x (? u6? y)) (build-exp ($primcall 'ulsh/immediate (x y))))
        (('scm->f64 (? f64?)) (rename 'load-f64))
        (('scm->u64 (? u64?)) (rename 'load-u64))
        (('scm->u64/truncate (? u64?)) (rename 'load-u64))
        (('scm->s64 (? s64?)) (rename 'load-s64))
        (_ #f)))
    (intmap-map
     (lambda (label cont)
       (match cont
         (($ $kargs names vars ($ $continue k src ($ $primcall name args)))
          (let ((exp* (specialize-primcall name args)))
            (if exp*
                (build-cont
                  ($kargs names vars ($continue k src ,exp*)))
                cont)))
         (_ cont)))
     conts)))