1
0
mirror of https://git.savannah.gnu.org/git/emacs.git synced 2024-11-23 07:19:15 +00:00
emacs/lisp/loadhist.el

193 lines
7.1 KiB
EmacsLisp
Raw Normal View History

1993-04-16 23:43:57 +00:00
;;; loadhist.el --- lisp functions for working with feature groups
1996-01-14 07:34:30 +00:00
;; Copyright (C) 1995, 1998, 2000 Free Software Foundation, Inc.
1993-04-16 23:43:57 +00:00
;; Author: Eric S. Raymond <esr@snark.thyrsus.com>
;; Maintainer: FSF
1993-04-16 23:43:57 +00:00
;; Keywords: internal
1995-04-08 02:24:20 +00:00
;; 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 2, 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
1996-01-14 07:34:30 +00:00
;; along with GNU Emacs; see the file COPYING. If not, write to the
;; Free Software Foundation, Inc., 59 Temple Place - Suite 330,
;; Boston, MA 02111-1307, USA.
1995-04-08 02:24:20 +00:00
1993-04-16 23:43:57 +00:00
;;; Commentary:
;; These functions exploit the load-history system variable.
;; Entry points include `unload-feature', `symbol-file', and
;; `feature-file', documented in the Emacs Lisp manual.
1993-04-16 23:43:57 +00:00
;;; Code:
(defun feature-symbols (feature)
"Return the file and list of definitions associated with FEATURE.
The value is actually the element of `load-history'
for the file that did (provide FEATURE)."
1993-04-16 23:43:57 +00:00
(catch 'foundit
(mapc (lambda (x)
(if (member (cons 'provide feature) (cdr x))
(throw 'foundit x)))
load-history)
1993-04-16 23:43:57 +00:00
nil))
(defun feature-file (feature)
"Return the file name from which a given FEATURE was loaded.
Actually, return the load argument, if any; this is sometimes the name of a
Lisp file without an extension. If the feature came from an `eval-buffer' on
a buffer with no associated file, or an `eval-region', return nil."
1993-04-16 23:43:57 +00:00
(if (not (featurep feature))
(error "%S is not a currently loaded feature" feature)
1993-04-16 23:43:57 +00:00
(car (feature-symbols feature))))
(defun file-provides (file)
"Return the list of features provided by FILE."
(let ((symbols (cdr (assoc file load-history)))
provides)
(mapc (lambda (x)
(if (and (consp x) (eq (car x) 'provide))
(setq provides (cons (cdr x) provides))))
symbols)
provides))
1993-04-16 23:43:57 +00:00
(defun file-requires (file)
"Return the list of features required by FILE."
(let ((symbols (cdr (assoc file load-history)))
requires)
(mapc (lambda (x)
(if (and (consp x) (eq (car x) 'require))
(setq requires (cons (cdr x) requires))))
symbols)
requires))
(defsubst file-set-intersect (p q)
"Return the set intersection of two lists."
1993-04-16 23:43:57 +00:00
(let ((ret nil))
(dolist (x p ret)
(if (memq x q) (setq ret (cons x ret))))
ret))
1993-04-16 23:43:57 +00:00
(defun file-dependents (file)
"Return the list of loaded libraries that depend on FILE.
This can include FILE itself."
(let ((provides (file-provides file))
(dependents nil))
(dolist (x load-history dependents)
(if (file-set-intersect provides (file-requires (car x)))
(setq dependents (cons (car x) dependents))))
dependents))
1993-04-16 23:43:57 +00:00
(defun read-feature (prompt)
"Read a feature name \(string\) from the minibuffer.
Prompt with PROMPT and completing from `features', and
return the feature \(symbol\)."
(intern (completing-read prompt
(mapcar (lambda (feature)
(list (symbol-name feature)))
features)
nil t)))
(defvar loadhist-hook-functions
'(after-change-functions
after-insert-file-functions auto-fill-function
before-change-functions blink-paren-function
buffer-access-fontify-functions command-line-functions
comment-indent-function kill-buffer-query-functions
kill-emacs-query-functions lisp-indent-function
mouse-position-function
redisplay-end-trigger-functions temp-buffer-show-function
window-scroll-functions window-size-change-functions
write-region-annotate-functions)
"A list of special hooks from Info node `(elisp)Standard Hooks'.
These are symbols with hook-type values whose names don't end in
`-hook' or `-hooks', from which `unload-feature' tries to remove
pertinent symbols.")
1993-04-16 23:43:57 +00:00
;;;###autoload
(defun unload-feature (feature &optional force)
"Unload the library that provided FEATURE, restoring all its autoloads.
If the feature is required by any other loaded code, and prefix arg FORCE
1993-04-16 23:43:57 +00:00
is nil, raise an error."
(interactive (list (read-feature "Feature: ") current-prefix-arg))
1993-04-16 23:43:57 +00:00
(if (not (featurep feature))
(error "%s is not a currently loaded feature" (symbol-name feature)))
1993-04-16 23:43:57 +00:00
(if (not force)
(let* ((file (feature-file feature))
(dependents (delete file (copy-sequence (file-dependents file)))))
1993-04-16 23:43:57 +00:00
(if dependents
(error "Loaded libraries %s depend on %s"
(prin1-to-string dependents) file))))
(let* ((flist (feature-symbols feature))
(file (car flist))
(unload-hook (intern-soft (concat (symbol-name feature)
"-unload-hook"))))
;; Try to avoid losing badly when hooks installed in critical
;; places go away. (Some packages install things on
;; `kill-buffer-hook', `activate-menubar-hook' and the like.)
;; First off, provide a clean way for package `foo' to arrange
;; this by defining `foo-unload-hook'.
(if unload-hook
(run-hooks unload-hook)
;; Otherwise, do our best. Look through the obarray for symbols
;; which seem to be hook variables or special hook functions and
;; remove anything from them which matches the feature-symbols
;; about to get zapped. Obviously this won't get anonymous
;; functions which the package might just have installed, and
;; there might be other important state, but this tactic
;; normally works.
(mapatoms
(lambda (x)
(if (or (and (boundp x) ; Random hooks.
(consp (symbol-value x))
(string-match "-hooks?\\'" (symbol-name x)))
2000-08-16 20:51:48 +00:00
(and (boundp x) ; Known abnormal hooks etc.
(memq x loadhist-hook-functions)))
(dolist (y (cdr flist))
(remove-hook x y))))))
(if (fboundp 'elp-restore-function) ; remove ELP stuff first
(dolist (elt (cdr flist))
(if (symbolp elt)
(elp-restore-function elt))))
(mapc
(lambda (x)
(cond ((stringp x) nil)
((consp x)
;; Remove any feature names that this file provided.
(if (eq (car x) 'provide)
(setq features (delq (cdr x) features)))
(when (eq (car x) 'defvar)
;; Kill local values as much as possible.
(dolist (buf (buffer-list))
(with-current-buffer buf
(kill-local-variable (cdr x))))
;; Get rid of the default binding if we can.
(unless (local-variable-if-set-p (cdr x))
(makunbound (cdr x)))))
(t
(when (fboundp x)
(if (fboundp 'ad-unadvise)
(ad-unadvise x))
(fmakunbound x)
(let ((aload (get x 'autoload)))
(if aload (fset x (cons 'autoload aload))))))))
(cdr flist))
;; Delete the load-history element for this file.
(let ((elt (assoc file load-history)))
(setq load-history (delq elt load-history)))))
1993-04-16 23:43:57 +00:00
(provide 'loadhist)
;;; loadhist.el ends here