Skip to content
defsystem.lisp 8.96 KiB
Newer Older
;;;; -------------------------------------------------------------------------
;;;; Defsystem

(asdf/package:define-package :asdf/defsystem
  (:recycle :asdf/defsystem :asdf)
  (:use :common-lisp :asdf/driver :asdf/upgrade
   :asdf/component :asdf/system
   :asdf/find-system :asdf/find-component :asdf/lisp-action :asdf/operate
   :asdf/backward-internals)
  #+gcl<2.7 (:shadowing-import-from :asdf/compatibility #:type-of)
   #:defsystem #:register-system-definition
   #:class-for-type #:*default-component-class*
   #:determine-system-directory #:parse-component-form
   #:duplicate-names #:sysdef-error-component #:check-component-input))
(in-package :asdf/defsystem)

;;; Pathname

(defun* determine-system-directory (pathname)
  ;; The defsystem macro calls this function to determine
  ;; the pathname of a system as follows:
  ;; 1. if the pathname argument is an pathname object (NOT a namestring),
  ;;    that is already an absolute pathname, return it.
  ;; 2. otherwise, the directory containing the LOAD-PATHNAME
  ;;    is considered (as deduced from e.g. *LOAD-PATHNAME*), and
  ;;    if it is indeed available and an absolute pathname, then
  ;;    the PATHNAME argument is normalized to a relative pathname
  ;;    as per PARSE-UNIX-NAMESTRING (with ENSURE-DIRECTORY T)
  ;;    and merged into that DIRECTORY as per SUBPATHNAME.
  ;;    Note: avoid *COMPILE-FILE-PATHNAME* because .asd is loaded,
  ;;    and may be from within the EVAL-WHEN of a file compilation.
  ;; If no absolute pathname was found, we return NIL.
  (check-type pathname (or null string pathname))
   (list pathname (load-pathname) *default-pathname-defaults*
         #-(or abcl gcl genera) (getcwd))
   :resolve-symlinks *resolve-symlinks*))

;;; Component class

(defvar *default-component-class* 'cl-source-file)

(defun* class-for-type (parent type)
  (or (loop :for symbol :in (list
                             type
                             (find-symbol* type *package* nil)
                             (find-symbol* type :asdf/interface nil))
        :for class = (and symbol (find-class* symbol nil))
        :when (and class
                   (#-cormanlisp subtypep #+cormanlisp cl::subclassp
                                 class (find-class* 'component)))
        :return class)
      (and (eq type :file)
           (find-class*
            (or (loop :for p = parent :then (component-parent p) :while p
                      :thereis (module-default-component-class p))
                *default-component-class*) nil))
      (sysdef-error "don't recognize component type ~A" type)))

;;; Check inputs

(define-condition duplicate-names (system-definition-error)
  ((name :initarg :name :reader duplicate-names-name))
  (:report (lambda (c s)
             (format s (compatfmt "~@<Error while defining system: multiple components are given same name ~A~@:>")
                     (duplicate-names-name c)))))

(defun* sysdef-error-component (msg type name value)
  (sysdef-error (strcat msg (compatfmt "~&~@<The value specified for ~(~A~) ~A is ~S~@:>"))
                type name value))

(defun* check-component-input (type name weakly-depends-on
  "A partial test of the values of a component."
  (unless (listp depends-on)
    (sysdef-error-component ":depends-on must be a list."
                            type name depends-on))
  (unless (listp weakly-depends-on)
    (sysdef-error-component ":weakly-depends-on must be a list."
                            type name weakly-depends-on))
  (unless (listp components)
    (sysdef-error-component ":components must be NIL or a list of components."
  (etypecase form
    ((or string null) form)
    (cons
        (destructuring-bind (subpath &key (path 0)) (rest form)
          (safe-read-file-form (subpathname pathname subpath) :path path)))))))

;;; Main parsing function

(defun* parse-component-form (parent options &key previous-serial-component)
  (destructuring-bind
      (type name &rest rest &key
       ;; the following list of keywords is reproduced below in the
       ;; remove-plist-keys form.  important to keep them in sync
       components pathname perform explain output-files operation-done-p
       weakly-depends-on depends-on serial
       do-first if-component-dep-fails (version nil versionp)
       ;; list ends
       &allow-other-keys) options
    (declare (ignorable perform explain output-files operation-done-p))
    (check-component-input type name weakly-depends-on depends-on components)
    (when (and parent
               (find-component parent name)
               (not ;; ignore the same object when rereading the defsystem
                (typep (find-component parent name)
                       (class-for-type parent type))))
      (error 'duplicate-names :name name))
    (when do-first (error "DO-FIRST is not supported anymore as of ASDF 2.27"))
    (let* ((args `(:name ,(coerce-name name)
                   :pathname ,pathname
                   ,@(when parent `(:parent ,parent))
                      '(:components :pathname :if-component-dep-fails :version
                        :perform :explain :output-files :operation-done-p
                        :weakly-depends-on :depends-on :serial)
           (component (find-component parent name)))
        ;; ASDF3: deprecate this feature and remove it.
        (appendf depends-on
                 (remove-if (complement #'(lambda (x) (find-system x nil))) weakly-depends-on)))
      (when previous-serial-component
        (push previous-serial-component depends-on))
      (if component ; preserve identity
          (apply 'reinitialize-instance component args)
          (setf component (apply 'make-instance (class-for-type parent type) args)))
      (component-pathname component) ; eagerly compute the absolute pathname
      (let ((sysdir (system-source-directory (component-system component)))) ;; requires the previous
        (setf version (normalize-version version sysdir)))
      (when (and versionp (not (parse-version version nil)))
        (warn (compatfmt "~@<Invalid version ~S for component ~S~@[ of ~S~]~@:>")
              version name parent))
      (setf (component-version component) version)
      (when (typep component 'parent-component)
        (setf (component-children component)
              (loop
                :with previous-component = nil
                :for c-form :in components
                :for c = (parse-component-form component c-form
                                               :previous-serial-component previous-component)
                :for name = (component-name c)
                :collect c
                :when serial :do (setf previous-component name)))
        (compute-children-by-name component))
      ;; Used by POIU. ASDF3: rename to component-depends-on
      (setf (component-sibling-dependencies component) depends-on)
      (%refresh-component-inline-methods component rest)
      (when if-component-dep-fails
        (%resolve-if-component-dep-fails if-component-dep-fails component))
      component)))
(defun* register-system-definition
    (name &rest options &key pathname (class 'system) (source-file () sfp)
          defsystem-depends-on &allow-other-keys)
  ;; The system must be registered before we parse the body,
  ;; otherwise we recur when trying to find an existing system
  ;; of the same name to reuse options (e.g. pathname) from.
  ;; To avoid infinite recursion in cases where you defsystem a system
  ;; that is registered to a different location to find-system,
  ;; we also need to remember it in a special variable *systems-being-defined*.
  (with-system-definitions ()
    (let* ((name (coerce-name name))
           (source-file (if sfp source-file (resolve-symlinks* (load-pathname))))
           (registered (system-registered-p name))
           (registered! (if registered
                            (rplaca registered (safe-file-write-date source-file))
                            (register-system
                             (make-instance 'system :name name :source-file source-file))))
           (system (reset-system (cdr registered!)
                                 :name name :source-file source-file))
           (component-options (remove-plist-key :class options)))
      (apply 'load-systems defsystem-depends-on)
      ;; We change-class AFTER we loaded the defsystem-depends-on
      ;; since the class might be defined as part of those.
      (let ((class (class-for-type nil class)))
        (unless (eq (type-of system) class)
          (change-class system class)))
      (parse-component-form
       nil (list*
            :module name
            :pathname (determine-system-directory pathname)
            component-options)))))

(defmacro defsystem (name &body options)
  `(apply 'register-system-definition ',name ',options))