195 lines
6.1 KiB
EmacsLisp
195 lines
6.1 KiB
EmacsLisp
;;; comp.el --- compilation of Lisp code into native code -*- lexical-binding: t -*-
|
|
|
|
;; Copyright (C) 2019 Free Software Foundation, Inc.
|
|
|
|
;; Keywords: lisp
|
|
;; Package: emacs
|
|
|
|
;; This file is part of GNU Emacs.
|
|
|
|
;; GNU Emacs is free software: you can redistribute it and/or modify
|
|
;; it under the terms of the GNU General Public License as published by
|
|
;; the Free Software Foundation, either version 3 of the License, or
|
|
;; (at your option) any later version.
|
|
|
|
;; GNU Emacs 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 General Public License for more details.
|
|
|
|
;; You should have received a copy of the GNU General Public License
|
|
;; along with GNU Emacs. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
;;; Code:
|
|
|
|
(require 'bytecomp)
|
|
(eval-when-compile (require 'cl-lib))
|
|
|
|
(defgroup comp nil
|
|
"Emacs Lisp native compiler."
|
|
:group 'lisp)
|
|
|
|
(defconst comp-debug t)
|
|
|
|
(defconst comp-passes '(comp-recuparate-lap
|
|
comp-limplify)
|
|
"Passes to be executed in order.")
|
|
|
|
(cl-defstruct comp-args
|
|
mandatory nonrest rest)
|
|
|
|
(cl-defstruct (comp-func (:copier nil))
|
|
"Internal rapresentation for a function."
|
|
(symbol-name nil
|
|
:documentation "Function symbol's name")
|
|
(func nil
|
|
:documentation "Original form")
|
|
(byte-func nil
|
|
:documentation "Byte compiled version")
|
|
(ir nil
|
|
:documentation "Current intermediate rappresentation")
|
|
(args nil :type 'comp-args))
|
|
|
|
(cl-defstruct (comp-meta-var (:copier nil))
|
|
"A frame slot into the meta-stack."
|
|
(slot nil :type fixnum
|
|
:documentation "Slot position into the meta-stack")
|
|
(const-vld nil
|
|
:documentation "Valid for the following slot")
|
|
(constant nil
|
|
:documentation "When const-vld non nil this is used for constant
|
|
propagation")
|
|
(type nil
|
|
:documentation "When non nil is used for type propagation"))
|
|
|
|
(cl-defstruct (comp-limple-frame (:copier nil))
|
|
"A LIMPLE func."
|
|
(sp 0 :type 'fixnum
|
|
:documentation "Current stack pointer")
|
|
(frame nil :type 'vector
|
|
:documentation "Meta-stack used to flat LAP"))
|
|
|
|
(defun comp-decrypt-lambda-list (x)
|
|
"Decript lambda list X."
|
|
(make-comp-args :rest (not (= (logand x 128) 0))
|
|
:mandatory (logand x 127)
|
|
:nonrest (ash x -8)))
|
|
|
|
(defun comp-recuparate-lap (ir)
|
|
"Byte compile and recuparate LAP rapresentation for IR."
|
|
;; FIXME block timers here, otherwise we could spill the wrong LAP.
|
|
(setf (comp-func-byte-func ir)
|
|
(byte-compile (comp-func-symbol-name ir)))
|
|
(when comp-debug
|
|
(cl-prettyprint byte-compile-lap-output))
|
|
(setf (comp-func-args ir)
|
|
(comp-decrypt-lambda-list (aref (comp-func-byte-func ir) 0)))
|
|
(setf (comp-func-ir ir) byte-compile-lap-output)
|
|
ir)
|
|
|
|
(defmacro comp-sp ()
|
|
"Current stack pointer."
|
|
'(comp-limple-frame-sp frame))
|
|
|
|
(defmacro comp-slot-n (n)
|
|
"Slot N into the meta-stack."
|
|
`(aref (comp-limple-frame-frame frame) ,n))
|
|
|
|
(defmacro comp-slot ()
|
|
"Current slot into the meta-stack pointed by sp."
|
|
'(comp-slot-n (comp-sp)))
|
|
|
|
(defmacro comp-push (x)
|
|
"Push X into frame."
|
|
`(progn
|
|
(cl-incf (comp-sp))
|
|
(list '= (comp-slot) ,x)))
|
|
|
|
(defmacro comp-push-slot-n (n)
|
|
"Push slot number N into frame."
|
|
`(let ((src-slot (comp-slot-n ,n)))
|
|
(cl-incf (comp-sp))
|
|
(setf (comp-slot)
|
|
(copy-sequence src-slot))
|
|
(setf (comp-meta-var-slot (comp-slot)) (comp-sp))
|
|
(list '=slot (comp-slot) src-slot)))
|
|
|
|
(defmacro comp-push-const (x)
|
|
"Push X into frame.
|
|
X value is known at compile time."
|
|
`(progn
|
|
(cl-incf (comp-sp))
|
|
(setf (comp-slot) (make-comp-meta-var :slot (comp-sp)
|
|
:const-vld t
|
|
:constant ,x))
|
|
(list '=const (comp-slot) ,x)))
|
|
|
|
(defmacro comp-pop (n)
|
|
"Pop N elements from the meta-stack."
|
|
`(cl-decf (comp-sp) ,n))
|
|
|
|
(defun comp-limplify-lap-inst (inst frame)
|
|
"Limplify LAP instruction INST in current FRAME."
|
|
(let ((op (car inst)))
|
|
(pcase op
|
|
('byte-dup
|
|
(comp-push-slot-n (comp-sp)))
|
|
('byte-varref
|
|
(comp-push `(call Fsymbol_value ,(second inst))))
|
|
('byte-constant
|
|
(comp-push-const (second inst)))
|
|
('byte-stack-ref
|
|
(comp-push-slot-n (- (comp-sp) (cdr inst))))
|
|
('byte-plus
|
|
(comp-pop 2)
|
|
(comp-push `(callref Fplus 2 ,(comp-sp))))
|
|
('byte-car
|
|
(comp-pop 1)
|
|
(comp-push `(Fcar ,(comp-sp))))
|
|
('byte-return
|
|
`(return ,(comp-slot)))
|
|
(_ 'xxx))))
|
|
|
|
(defun comp-limplify (ir)
|
|
"Given IR and return LIMPLE."
|
|
(let* ((frame-size (aref (comp-func-byte-func ir) 3))
|
|
(frame (make-comp-limple-frame
|
|
:sp -1
|
|
:frame (let ((v (make-vector frame-size nil)))
|
|
(cl-loop for i below frame-size
|
|
do (aset v i (make-comp-meta-var :slot i)))
|
|
v)))
|
|
(limple-ir ()))
|
|
;; Prologue
|
|
(push '(BLOCK prologue) limple-ir)
|
|
(cl-loop for i below (comp-args-mandatory (comp-func-args ir))
|
|
do (progn
|
|
(cl-incf (comp-sp))
|
|
(push `(=par ,(comp-slot) ,i) limple-ir)))
|
|
(push '(BLOCK body) limple-ir)
|
|
(cl-loop for inst in (comp-func-ir ir)
|
|
do (push (comp-limplify-lap-inst inst frame) limple-ir))
|
|
(setq limple-ir (reverse limple-ir))
|
|
(setf (comp-func-ir ir) limple-ir)
|
|
(when comp-debug
|
|
(cl-prettyprint (comp-func-ir ir)))
|
|
ir))
|
|
|
|
(defun native-compile (fun)
|
|
"FUN is the function definition to be compiled to native code."
|
|
(unless lexical-binding
|
|
(error "Can't compile a non lexical binded function"))
|
|
(if-let ((f (symbol-function fun)))
|
|
(progn
|
|
(when (byte-code-function-p f)
|
|
(error "Can't native compile an already bytecompiled function"))
|
|
(cl-loop with ir = (make-comp-func :symbol-name fun
|
|
:func f)
|
|
for pass in comp-passes
|
|
do (funcall pass ir)
|
|
finally return ir))
|
|
(error "Trying to native compile not a function")))
|
|
|
|
(provide 'comp)
|
|
|
|
;;; comp.el ends here
|