]> code.delx.au - gnu-emacs/blobdiff - lisp/font-lock.el
(mouse-drag-secondary): Make sure to return nil
[gnu-emacs] / lisp / font-lock.el
index edff5980e90ecf179490e3a46fde874531e749ef..fd3ccdc0d1a948a6ddf1cda35cc375579d69c7b6 100644 (file)
@@ -1,6 +1,6 @@
 ;;; font-lock.el --- Electric font lock mode
 
-;; Copyright (C) 1992, 1993, 1994, 1995 Free Software Foundation, Inc.
+;; Copyright (C) 1992, 1993, 1994, 1995, 1996 Free Software Foundation, Inc.
 
 ;; Author: jwz, then rms, then sm <simon@gnu.ai.mit.edu>
 ;; Maintainer: FSF
@@ -19,8 +19,9 @@
 ;; GNU General Public License for more details.
 
 ;; You should have received a copy of the GNU General Public License
-;; along with GNU Emacs; see the file COPYING.  If not, write to
-;; the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
+;; 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.
 
 ;;; Commentary:
 
 ;; When this minor mode is on, the faces of the current line are updated with
 ;; every insertion or deletion.
 ;;
-;; To turn Font Lock mode on automatically, add this to your .emacs file:
+;; To turn Font Lock mode on automatically, add this to your ~/.emacs file:
 ;;
 ;;  (add-hook 'emacs-lisp-mode-hook 'turn-on-font-lock)
 ;;
+;; Or if you want to turn Font Lock mode on in many modes:
+;;
+;;  (global-font-lock-mode t)
+;;
 ;; Fontification for a particular mode may be available in a number of levels
 ;; of decoration.  The higher the level, the more decoration, but the more time
 ;; it takes to fontify.  See the variable `font-lock-maximum-decoration', and
-;; also the variable `font-lock-maximum-size'.
+;; also the variable `font-lock-maximum-size'.  Support modes for Font Lock
+;; mode can be used to speed up Font Lock mode.  See `font-lock-support-mode'.
+\f
+;; Constructing patterns:
+;;
+;; See the documentation for the variable `font-lock-keywords'.
+;;
+;; Nasty regexps of the form "bar\\(\\|lo\\)\\|f\\(oo\\|u\\(\\|bar\\)\\)\\|lo"
+;; are made thusly: (make-regexp '("foo" "fu" "fubar" "bar" "barlo" "lo")) for
+;; efficiency.  See /pub/gnu/emacs/elisp-archive/functions/make-regexp.el.Z on
+;; archive.cis.ohio-state.edu for this and other functions.
+
+;; Adding patterns for modes that already support Font Lock:
+;;
+;; Font Lock mode uses the buffer local variable `font-lock-keywords' for the
+;; highlighting patterns.  This variable is set by Font Lock mode from (a) the
+;; buffer local variable `font-lock-defaults', if non-nil, or (b) the global
+;; variable `font-lock-defaults-alist', if the major mode has an entry. 
+;; Font Lock mode is set up via (a) where a mode's patterns are distributed
+;; with the mode's package library, (b) where a mode's patterns are distributed
+;; with font-lock.el itself.  An example of (a) is Pascal mode, an example of
+;; (b) is C/C++ modes.  (Normally, the mechanism is (a); (b) is used where it
+;; is not clear which package library should contain the pattern definitions.)
+;;
+;; If, for a particular mode, mechanism (a) is used, you need to add your
+;; patterns after that package library has loaded, e.g.:
+;;
+;;  (eval-after-load "pascal" '(add-to-list 'pascal-font-lock-keywords ...))
+;;
+;; (Note that only one pattern can be added with `add-to-list'.  For multiple
+;; patterns, use one `eval-after-load' form with one `setq' and `append' form,
+;; or multiple `eval-after-load' forms each with one `add-to-list' form.)
+;; If mechanism (b) is used, you need to add your patterns after font-lock.el
+;; itself has loaded, e.g.:
+;;
+;;  (eval-after-load "font-lock" '(add-to-list 'c-font-lock-keywords ...))
+;;
+;; Which variable you should add to depends on what level of fontification you
+;; choose and what level is supported.  If you choose the maximum level, by
+;; setting the variable `font-lock-maximum-decoration', you change a different
+;; variable.  Maximum level patterns for C are `c-font-lock-keywords-3', so:
+;;
+;;  (setq font-lock-maximum-decoration t)
+;;  (eval-after-load "font-lock"
+;;   '(add-to-list 'c-font-lock-keywords-3
+;;                '("\\<FILE\\>" . font-lock-type-face)))
+;;
+;; To see which variable to set, see the buffer's value of `font-lock-defaults'
+;; or the mode's entry in the global value of `font-lock-defaults-alist'.
 
+;; Adding patterns for modes that do not support Font Lock:
+;;
 ;; If you add patterns for a new mode, say foo.el's `foo-mode', say in which
 ;; you don't want syntactic fontification to occur, you can make Font Lock mode
 ;; use your regexps when turning on Font Lock by adding to `foo-mode-hook':
 ;;  (add-hook 'foo-mode-hook
 ;;   '(lambda () (make-local-variable 'font-lock-defaults)
 ;;               (setq font-lock-defaults '(foo-font-lock-keywords t))))
-;;
-;; Nasty regexps of the form "bar\\(\\|lo\\)\\|f\\(oo\\|u\\(\\|bar\\)\\)\\|lo"
-;; are made thusly: (make-regexp '("foo" "fu" "fubar" "bar" "barlo" "lo")) for
-;; efficiency.  See /pub/gnu/emacs/elisp-archive/functions/make-regexp.el.Z on
-;; archive.cis.ohio-state.edu for this and other functions.
-
+\f
 ;; What is fontification for?  You might say, "It's to make my code look nice."
 ;; I think it should be for adding information in the form of cues.  These cues
 ;; should provide you with enough information to both (a) distinguish between
 ;;
 ;; So, here are my opinions/advice/guidelines:
 ;; 
+;; - Highlight conceptual objects, such as function and variable names, and
+;;   different objects types differently, i.e., (a) and (b) above, highlight
+;;   function names differently to variable names.
+;; - Keep the faces distinct from each other as far as possible.
+;;   i.e., (a) above.
 ;; - Use the same face for the same conceptual object, across all modes.
 ;;   i.e., (b) above, all modes that have items that can be thought of as, say,
 ;;   keywords, should be highlighted with the same face, etc.
-;; - Keep the faces distinct from each other as far as possible.
-;;   i.e., (a) above.
 ;; - Make the face attributes fit the concept as far as possible.
 ;;   i.e., function names might be a bold colour such as blue, comments might
 ;;   be a bright colour such as red, character strings might be brown, because,
 \f
 ;; User variables.
 
-(defvar font-lock-verbose t
-  "*If non-nil, means show status messages when fontifying.")
+(defvar font-lock-verbose (* 0 1024)
+  "*If non-nil, means show status messages for buffer fontification.
+If a number, only buffers greater than this size have fontification messages.")
 
 ;;;###autoload
 (defvar font-lock-maximum-decoration nil
-  "*If non-nil, the maximum decoration level for fontifying.
+  "*Maximum decoration level for fontification.
 If nil, use the default decoration (typically the minimum available).
 If t, use the maximum decoration available.
 If a number, use that level of decoration (or if not available the maximum).
 If a list, each element should be a cons pair of the form (MAJOR-MODE . LEVEL),
 where MAJOR-MODE is a symbol or t (meaning the default).  For example:
- ((c++-mode . 2) (c-mode . t) (t . 1))
-means use level 2 decoration for buffers in `c++-mode', the maximum decoration
-available for buffers in `c-mode', and level 1 decoration otherwise.")
+ ((c-mode . t) (c++-mode . 2) (t . 1))
+means use the maximum decoration available for buffers in C mode, level 2
+decoration for buffers in C++ mode, and level 1 decoration otherwise.")
 
 ;;;###autoload
 (defvar font-lock-maximum-size (* 250 1024)
-  "*If non-nil, the maximum size for buffers for fontifying.
+  "*Maximum size of a buffer for buffer fontification.
 Only buffers less than this can be fontified when Font Lock mode is turned on.
 If nil, means size is irrelevant.
 If a list, each element should be a cons pair of the form (MAJOR-MODE . SIZE),
 where MAJOR-MODE is a symbol or t (meaning the default).  For example:
- ((c++-mode . 256000) (c-mode . 256000) (rmail-mode . 1048576))
-means that the maximum size is 250K for buffers in `c++-mode' or `c-mode', one
-megabyte for buffers in `rmail-mode', and size is irrelevant otherwise.")
+ ((c-mode . 256000) (c++-mode . 256000) (rmail-mode . 1048576))
+means that the maximum size is 250K for buffers in C or C++ modes, one megabyte
+for buffers in Rmail mode, and size is irrelevant otherwise.")
 \f
 ;; Fontification variables:
 
@@ -142,11 +196,16 @@ Each element should be of the form:
  (MATCHER . FACENAME)
  (MATCHER . HIGHLIGHT)
  (MATCHER HIGHLIGHT ...)
+ (eval . FORM)
 
 where HIGHLIGHT should be either MATCH-HIGHLIGHT or MATCH-ANCHORED.
 
+FORM is an expression, whose value should be a keyword element, evaluated when
+the keyword is (first) used in a buffer.  This feature can be used to provide a
+keyword that can only be generated when Font Lock mode is actually turned on.
+
 For highlighting single items, typically only MATCH-HIGHLIGHT is required.
-However, if an item or (typically) items is to be hightlighted following the
+However, if an item or (typically) items are to be highlighted following the
 instance of another item (the anchor) then MATCH-ANCHORED may be required.
 
 MATCH-HIGHLIGHT should be of the form:
@@ -163,7 +222,7 @@ OVERRIDE and LAXMATCH are flags.  If OVERRIDE is t, existing fontification may
 be overwritten.  If `keep', only parts not already fontified are highlighted.
 If `prepend' or `append', existing fontification is merged with the new, in
 which the new or existing fontification, respectively, takes precedence.
-If LAXMATCH is non-nil, no error is signalled if there is no MATCH in MATCHER.
+If LAXMATCH is non-nil, no error is signaled if there is no MATCH in MATCHER.
 
 For example, an element of the form highlights (if not already highlighted):
 
@@ -222,30 +281,37 @@ The value should be like the `cdr' of an item in `font-lock-defaults-alist'.")
        (c-mode-defaults
         '((c-font-lock-keywords c-font-lock-keywords-1
            c-font-lock-keywords-2 c-font-lock-keywords-3)
-          nil nil ((?_ . "w")) beginning-of-defun))
+          nil nil ((?_ . "w")) beginning-of-defun
+          (font-lock-comment-start-regexp . "/[*/]")
+          (font-lock-mark-block-function . mark-defun)))
        (c++-mode-defaults
         '((c++-font-lock-keywords c++-font-lock-keywords-1 
            c++-font-lock-keywords-2 c++-font-lock-keywords-3)
-          nil nil ((?_ . "w") (?~ . "w")) beginning-of-defun))
+          nil nil ((?_ . "w") (?~ . "w")) beginning-of-defun
+          (font-lock-comment-start-regexp . "/[*/]")
+          (font-lock-mark-block-function . mark-defun)))
        (lisp-mode-defaults
         '((lisp-font-lock-keywords
            lisp-font-lock-keywords-1 lisp-font-lock-keywords-2)
-          nil nil
-          ((?: . "w") (?- . "w") (?* . "w") (?+ . "w") (?. . "w") (?< . "w")
-           (?> . "w") (?= . "w") (?! . "w") (?? . "w") (?$ . "w") (?% . "w")
-           (?_ . "w") (?& . "w") (?~ . "w") (?^ . "w") (?/ . "w"))
-          beginning-of-defun))
+          nil nil (("+-*/.<>=!?$%_&~^:" . "w")) beginning-of-defun
+          (font-lock-comment-start-regexp . ";")
+          (font-lock-mark-block-function . mark-defun)))
        (scheme-mode-defaults
-        '(scheme-font-lock-keywords nil t
-          ((?: . "w") (?- . "w") (?* . "w") (?+ . "w") (?. . "w") (?< . "w")
-           (?> . "w") (?= . "w") (?! . "w") (?? . "w") (?$ . "w") (?% . "w")
-           (?_ . "w") (?& . "w") (?~ . "w") (?^ . "w") (?/ . "w"))
-          beginning-of-defun))
+        '(scheme-font-lock-keywords
+          nil t (("+-*/.<>=!?$%_&~^:" . "w")) beginning-of-defun
+          (font-lock-comment-start-regexp . ";")
+          (font-lock-mark-block-function . mark-defun)))
        ;; For TeX modes we could use `backward-paragraph' for the same reason.
-       (tex-mode-defaults '(tex-font-lock-keywords nil nil ((?$ . "\""))))
+       ;; But we don't, because paragraph breaks are arguably likely enough to
+       ;; occur within a genuine syntactic block to make it too risky.
+       ;; However, we do specify a MARK-BLOCK function as that cannot result
+       ;; in a mis-fontification even if it might not fontify enough.  --sm.
+       (tex-mode-defaults
+        '(tex-font-lock-keywords nil nil ((?$ . "\"")) nil
+          (font-lock-comment-start-regexp . "%")
+          (font-lock-mark-block-function . mark-paragraph)))
        )
     (list
-     (cons 'bibtex-mode                        tex-mode-defaults)
      (cons 'c++-c-mode                 c-mode-defaults)
      (cons 'c++-mode                   c++-mode-defaults)
      (cons 'c-mode                     c-mode-defaults)
@@ -263,15 +329,16 @@ The value should be like the `cdr' of an item in `font-lock-defaults-alist'.")
   "Alist of default major mode and Font Lock defaults.
 Each item should be a list of the form:
 
- (MAJOR-MODE . (KEYWORDS KEYWORDS-ONLY CASE-FOLD SYNTAX-ALIST SYNTAX-BEGIN))
+ (MAJOR-MODE . (KEYWORDS KEYWORDS-ONLY CASE-FOLD SYNTAX-ALIST SYNTAX-BEGIN
+                ...))
 
 where MAJOR-MODE is a symbol.  KEYWORDS may be a symbol (a variable or function
 whose value is the keywords to use for fontification) or a list of symbols.
 If KEYWORDS-ONLY is non-nil, syntactic fontification (strings and comments) is
 not performed.  If CASE-FOLD is non-nil, the case of the keywords is ignored
 when fontifying.  If SYNTAX-ALIST is non-nil, it should be a list of cons pairs
-of the form (CHAR . STRING) used to set the local Font Lock syntax table, for
-keyword and syntactic fontification (see `modify-syntax-entry').
+of the form (CHAR-OR-STRING . STRING) used to set the local Font Lock syntax
+table, for keyword and syntactic fontification (see `modify-syntax-entry').
 
 If SYNTAX-BEGIN is non-nil, it should be a function with no args used to move
 backwards outside any enclosing syntactic block, for syntactic fontification.
@@ -284,7 +351,23 @@ is used as a position outside of a syntactic block, in the worst case.
 These item elements are used by Font Lock mode to set the variables
 `font-lock-keywords', `font-lock-keywords-only',
 `font-lock-keywords-case-fold-search', `font-lock-syntax-table' and
-`font-lock-beginning-of-syntax-function', respectively.")
+`font-lock-beginning-of-syntax-function', respectively.
+
+Further item elements are alists of the form (VARIABLE . VALUE) and are in no
+particular order.  Each VARIABLE is made buffer-local before set to VALUE.
+
+Currently, appropriate variables include `font-lock-mark-block-function'.
+If this is non-nil, it should be a function with no args used to mark any
+enclosing block of text, for fontification via \\[font-lock-fontify-block].
+Typical values are `mark-defun' for programming modes or `mark-paragraph' for
+textual modes (i.e., the mode-dependent function is known to put point and mark
+around a text block relevant to that mode).
+
+Other variables include those for buffer-specialised fontification functions,
+`font-lock-fontify-buffer-function', `font-lock-unfontify-buffer-function',
+`font-lock-fontify-region-function', `font-lock-unfontify-region-function',
+`font-lock-comment-start-regexp', `font-lock-inhibit-thing-lock' and
+`font-lock-maximum-size'.")
 
 (defvar font-lock-keywords-only nil
   "*Non-nil means Font Lock should not fontify comments or strings.
@@ -303,29 +386,65 @@ This is normally set via `font-lock-defaults'.")
 ;; `font-lock-cache-position' and `font-lock-cache-state'.
 (defvar font-lock-beginning-of-syntax-function nil
   "*Non-nil means use this function to move back outside of a syntactic block.
+When called with no args it should leave point at the beginning of any
+enclosing syntactic block.
 If this is nil, the beginning of the buffer is used (in the worst case).
 This is normally set via `font-lock-defaults'.")
 
-;; These record the parse state at a particular position, always the start of a
-;; line.  Used to make `font-lock-fontify-syntactically-region' faster.
-(defvar font-lock-cache-position nil)
-(defvar font-lock-cache-state nil)
-(make-variable-buffer-local 'font-lock-cache-position)
-(make-variable-buffer-local 'font-lock-cache-state)
+(defvar font-lock-mark-block-function nil
+  "*Non-nil means use this function to mark a block of text.
+When called with no args it should leave point at the beginning of any
+enclosing textual block and mark at the end.
+This is normally set via `font-lock-defaults'.")
+
+(defvar font-lock-comment-start-regexp nil
+  "*Regexp to match the start of a comment.
+This need not discriminate between genuine comments and quoted comment
+characters or comment characters within strings.
+If nil, `comment-start-skip' is used instead; see that variable for more info.
+This is normally set via `font-lock-defaults'.")
+
+(defvar font-lock-fontify-buffer-function 'font-lock-default-fontify-buffer
+  "Function to use for fontifying the buffer.
+This is normally set via `font-lock-defaults'.")
+
+(defvar font-lock-unfontify-buffer-function 'font-lock-default-unfontify-buffer
+  "Function to use for unfontifying the buffer.
+This is used when turning off Font Lock mode.
+This is normally set via `font-lock-defaults'.")
+
+(defvar font-lock-fontify-region-function 'font-lock-default-fontify-region
+  "Function to use for fontifying a region.
+It should take two args, the beginning and end of the region, and an optional
+third arg VERBOSE.  If non-nil, the function should print status messages.
+This is normally set via `font-lock-defaults'.")
+
+(defvar font-lock-unfontify-region-function 'font-lock-default-unfontify-region
+  "Function to use for unfontifying a region.
+It should take two args, the beginning and end of the region.
+This is normally set via `font-lock-defaults'.")
+
+(defvar font-lock-inhibit-thing-lock nil
+  "List of Font Lock mode related modes that should not be turned on.
+Currently, valid mode names as `fast-lock-mode' and `lazy-lock-mode'.
+This is normally set via `font-lock-defaults'.")
 
 (defvar font-lock-mode nil)            ; For the modeline.
 (defvar font-lock-fontified nil)       ; Whether we have fontified the buffer.
-(put 'font-lock-fontified 'permanent-local t)
 
 ;;;###autoload
 (defvar font-lock-mode-hook nil
   "Function or functions to run on entry to Font Lock mode.")
 \f
-;; User functions.
+;; Font Lock mode.
+
+(eval-when-compile
+  ;; We don't do this at the top-level as we only use non-autoloaded macros.
+  (require 'cl))
 
 ;;;###autoload
 (defun font-lock-mode (&optional arg)
-  "[pretest] Toggle Font Lock mode.
+  "Toggle Font Lock mode.
 With arg, turn Font Lock mode on if and only if arg is positive.
 
 When Font Lock mode is enabled, text is fontified as you type it:
@@ -340,101 +459,315 @@ the major mode's hook.  For example, put in your ~/.emacs:
 
  (add-hook 'c-mode-hook 'turn-on-font-lock)
 
-Or for any visited file with the following in your ~/.emacs:
+Alternatively, you can use Global Font Lock mode to automagically turn on Font
+Lock mode in buffers whose major mode supports it and whose major mode is one
+of `font-lock-global-modes'.  For example, put in your ~/.emacs:
 
- (add-hook 'find-file-hooks 'turn-on-font-lock)
+ (global-font-lock-mode t)
+
+There are a number of support modes that may be used to speed up Font Lock mode
+in various ways, specified via the variable `font-lock-support-mode'.  Where
+major modes support different levels of fontification, you can use the variable
+`font-lock-maximum-decoration' to specify which level you generally prefer.
+When you turn Font Lock mode on/off the buffer is fontified/defontified, though
+fontification occurs only if the buffer is less than `font-lock-maximum-size'.
+
+For example, to specify that Font Lock mode use use Lazy Lock mode as a support
+mode and use maximum levels of fontification, put in your ~/.emacs:
+
+ (setq font-lock-support-mode 'lazy-lock-mode)
+ (setq font-lock-maximum-decoration t)
+
+To fontify a buffer, without turning on Font Lock mode and regardless of buffer
+size, you can use \\[font-lock-fontify-buffer].
+
+To fontify a block (the function or paragraph containing point, or a number of
+lines around point), perhaps because modification on the current line caused
+syntactic change on other lines, you can use \\[font-lock-fontify-block].
 
 The default Font Lock mode faces and their attributes are defined in the
 variable `font-lock-face-attributes', and Font Lock mode default settings in
 the variable `font-lock-defaults-alist'.  You can set your own default settings
 for some mode, by setting a buffer local value for `font-lock-defaults', via
-its mode hook.
-
-Where modes support different levels of fontification, you can use the variable
-`font-lock-maximum-decoration' to specify which level you generally prefer.
-When you turn Font Lock mode on/off the buffer is fontified/defontified, though
-fontification occurs only if the buffer is less than `font-lock-maximum-size'.
-To fontify a buffer without turning on Font Lock mode, and regardless of buffer
-size, you can use \\[font-lock-fontify-buffer]."
+its mode hook."
   (interactive "P")
-  (let ((on-p (if arg (> (prefix-numeric-value arg) 0) (not font-lock-mode)))
-       (maximum-size (if (not (consp font-lock-maximum-size))
-                         font-lock-maximum-size
-                       (cdr (or (assq major-mode font-lock-maximum-size)
-                                (assq t font-lock-maximum-size))))))
-    (if (equal (buffer-name) " *Compiler Input*") ; hack for bytecomp...
-       (setq on-p nil))
-    (if (not on-p)
-       (remove-hook 'after-change-functions 'font-lock-after-change-function
-                    t)
-      (make-local-hook 'after-change-functions)
-      (add-hook 'after-change-functions 'font-lock-after-change-function
-               nil t))
+  ;; Don't turn on Font Lock mode if we don't have a display (we're running a
+  ;; batch job) or if the buffer is invisible (the name starts with a space).
+  (let ((on-p (and (not noninteractive)
+                  (not (eq (aref (buffer-name) 0) ?\ ))
+                  (if arg
+                      (> (prefix-numeric-value arg) 0)
+                    (not font-lock-mode)))))
     (set (make-local-variable 'font-lock-mode) on-p)
-    (cond (on-p
-          (font-lock-set-defaults)
-          (make-local-hook 'before-revert-hook)
-          (make-local-hook 'after-revert-hook)
-          ;; If buffer is reverted, must clean up the state.
-          (add-hook 'before-revert-hook 'font-lock-revert-setup nil t)
-          (add-hook 'after-revert-hook 'font-lock-revert-cleanup nil t)
-          (run-hooks 'font-lock-mode-hook)
-          (cond (font-lock-fontified
-                 nil)
-                ((or (null maximum-size) (<= (buffer-size) maximum-size))
-                 (font-lock-fontify-buffer))
-                (font-lock-verbose
-                 (message "Fontifying %s... buffer too big." (buffer-name)))))
-         (font-lock-fontified
-          (setq font-lock-fontified nil)
-          (remove-hook 'before-revert-hook 'font-lock-revert-setup t)
-          (remove-hook 'after-revert-hook 'font-lock-revert-cleanup t)
-          (font-lock-unfontify-region (point-min) (point-max))
-          (font-lock-thing-lock-cleanup))
-         (t
-          (remove-hook 'before-revert-hook 'font-lock-revert-setup t)
-          (remove-hook 'after-revert-hook 'font-lock-revert-cleanup t)
-          (font-lock-thing-lock-cleanup)))
+    ;; Turn on Font Lock mode.
+    (when on-p
+      (font-lock-set-defaults)
+      (unless (eq font-lock-fontify-region-function 'ignore)
+       (make-local-hook 'after-change-functions)
+       (add-hook 'after-change-functions 'font-lock-after-change-function nil t))
+      (font-lock-turn-on-thing-lock)
+      (run-hooks 'font-lock-mode-hook)
+      ;; Fontify the buffer if we have to.
+      (let ((max-size (font-lock-value-in-major-mode font-lock-maximum-size)))
+       (cond (font-lock-fontified
+              nil)
+             ((or (null max-size) (> max-size (buffer-size)))
+              (font-lock-fontify-buffer))
+             (font-lock-verbose
+              (message "Fontifying %s...buffer too big" (buffer-name))))))
+    ;; Turn off Font Lock mode.
+    (when (not on-p)
+      (remove-hook 'after-change-functions 'font-lock-after-change-function t)
+      (font-lock-unfontify-buffer)
+      (font-lock-turn-off-thing-lock)
+      (font-lock-unset-defaults))
     (force-mode-line-update)))
 
 ;;;###autoload
 (defun turn-on-font-lock ()
-  "Unconditionally turn on Font Lock mode."
-  (font-lock-mode 1))
+  "Turn on Font Lock mode conditionally.
+Turn on only if the terminal can display it."
+  (when window-system
+    (font-lock-mode t)))
+\f
+;; Global Font Lock mode.
+;;
+;; A few people have hassled in the past for a way to make it easier to turn on
+;; Font Lock mode, without the user needing to know for which modes s/he has to
+;; turn it on, perhaps the same way hilit19.el/hl319.el does.  I've always
+;; balked at that way, as I see it as just re-moulding the same problem in
+;; another form.  That is; some person would still have to keep track of which
+;; modes (which may not even be distributed with Emacs) support Font Lock mode.
+;; The list would always be out of date.  And that person might have to be me.
+
+;; Implementation.
+;;
+;; In a previous discussion the following hack came to mind.  It is a gross
+;; hack, but it generally works.  We use the convention that major modes start
+;; by calling the function `kill-all-local-variables', which in turn runs
+;; functions on the hook variable `change-major-mode-hook'.  We attach our
+;; function `font-lock-change-major-mode' to that hook.  Of course, when this
+;; hook is run, the major mode is in the process of being changed and we do not
+;; know what the final major mode will be.  So, `font-lock-change-major-mode'
+;; only (a) notes the name of the current buffer, and (b) adds our function
+;; `turn-on-font-lock-if-enabled' to the hook variables `find-file-hooks' and
+;; `post-command-hook' (for buffers that are not visiting files).  By the time
+;; the functions on the first of these hooks to be run are run, the new major
+;; mode is assumed to be in place.  This way we get a Font Lock function run
+;; when a major mode is turned on, without knowing major modes or their hooks.
+;;
+;; Naturally this requires that (a) major modes run `kill-all-local-variables',
+;; as they are supposed to do, and (b) the major mode is in place after the
+;; file is visited or the command that ran `kill-all-local-variables' has
+;; finished, whichever the sooner.  Arguably, any major mode that does not
+;; follow the convension (a) is broken, and I can't think of any reason why (b)
+;; would not be met (except `gnudoit' on non-files).  However, it is not clean.
+;;
+;; Probably the cleanest solution is to have each major mode function run some
+;; hook, e.g., `major-mode-hook', but maybe implementing that change is
+;; impractical.  I am personally against making `setq' a macro or be advised,
+;; or have a special function such as `set-major-mode', but maybe someone can
+;; come up with another solution?
+
+;; User interface.
+;;
+;; Although Global Font Lock mode is a pseudo-mode, I think that the user
+;; interface should conform to the usual Emacs convention for modes, i.e., a
+;; command to toggle the feature (`global-font-lock-mode') with a variable for
+;; finer control of the mode's behaviour (`font-lock-global-modes').
+;;
+;; I don't think it is better that the feature be enabled via a variable, since
+;; it does not conform to the usual convention.  I don't think the feature
+;; should be enabled by loading font-lock.el, since other mechanisms such as
+;; M-x font-lock-mode RET or (add-hook 'c-mode-hook 'turn-on-font-lock) would
+;; cause Font Lock mode to be turned on everywhere, and it is not intuitive or
+;; informative because loading a file tells you nothing about the feature or
+;; how to control it.  It would be contrary to the Principle of Least Surprise.
+
+(defvar font-lock-buffers nil)         ; For remembering buffers.
+(defvar global-font-lock-mode nil)
+
+;;;###autoload
+(defvar font-lock-global-modes t
+  "*Modes for which Font Lock mode is automagically turned on.
+Global Font Lock mode is controlled by the `global-font-lock-mode' command.
+If nil, means no modes have Font Lock mode automatically turned on.
+If t, all modes that support Font Lock mode have it automatically turned on.
+If a list, it should be a list of `major-mode' symbol names for which Font Lock
+mode should be automatically turned on.  The sense of the list is negated if it
+begins with `not'.  For example:
+ (c-mode c++-mode)
+means that Font Lock mode is turned on for buffers in C and C++ modes only.")
+
+;;;###autoload
+(defun global-font-lock-mode (&optional arg message)
+  "Toggle Global Font Lock mode.
+With prefix ARG, turn Global Font Lock mode on if and only if ARG is positive.
+Displays a message saying whether the mode is on or off if MESSAGE is non-nil.
+Returns the new status of Global Font Lock mode (non-nil means on).
+
+When Global Font Lock mode is enabled, Font Lock mode is automagically
+turned on in a buffer if its major mode is one of `font-lock-global-modes'."
+  (interactive "P\np")
+  (let ((off-p (if arg
+                  (<= (prefix-numeric-value arg) 0)
+                global-font-lock-mode)))
+    (if off-p
+       (remove-hook 'find-file-hooks 'turn-on-font-lock-if-enabled)
+      (add-hook 'find-file-hooks 'turn-on-font-lock-if-enabled)
+      (add-hook 'post-command-hook 'turn-on-font-lock-if-enabled)
+      (setq font-lock-buffers (buffer-list)))
+    (when message
+      (message "Global Font Lock mode is now %s." (if off-p "OFF" "ON")))
+    (setq global-font-lock-mode (not off-p))))
+
+(defun font-lock-change-major-mode ()
+  ;; Turn off Font Lock mode if it's on.
+  (when font-lock-mode
+    (font-lock-mode nil))
+  ;; Gross hack warning: Delicate readers should avert eyes now.
+  ;; Something is running `kill-all-local-variables', which generally means the
+  ;; major mode is being changed.  Run `turn-on-font-lock-if-enabled' after the
+  ;; file is visited or the current command has finished.
+  (when global-font-lock-mode
+    (add-hook 'post-command-hook 'turn-on-font-lock-if-enabled)
+    (add-to-list 'font-lock-buffers (current-buffer))))
+
+(defun turn-on-font-lock-if-enabled ()
+  ;; Gross hack warning: Delicate readers should avert eyes now.
+  ;; Turn on Font Lock mode if it's supported by the major mode and enabled by
+  ;; the user.
+  (remove-hook 'post-command-hook 'turn-on-font-lock-if-enabled)
+  (while font-lock-buffers
+    (if (buffer-live-p (car font-lock-buffers))
+       (save-excursion
+         (set-buffer (car font-lock-buffers))
+         (if (and (or font-lock-defaults
+                      (assq major-mode font-lock-defaults-alist))
+                  (or (eq font-lock-global-modes t)
+                      (if (eq (car-safe font-lock-global-modes) 'not)
+                          (not (memq major-mode (cdr font-lock-global-modes)))
+                        (memq major-mode font-lock-global-modes))))
+             (let (inhibit-quit)
+               (turn-on-font-lock)))))
+    (setq font-lock-buffers (cdr font-lock-buffers))))
+
+(add-hook 'change-major-mode-hook 'font-lock-change-major-mode)
+
+;; End of Global Font Lock mode.
+\f
+;; Font Lock Support mode.
+;;
+;; This is the code used to interface font-lock.el with any of its add-on
+;; packages, and provide the user interface.  Packages that have their own
+;; local buffer fontification functions (see below) may have to call
+;; `font-lock-after-fontify-buffer' and/or `font-lock-after-unfontify-buffer'
+;; themselves.
+
+;;;###autoload
+(defvar font-lock-support-mode nil
+  "*Support mode for Font Lock mode.
+Support modes speed up Font Lock mode by being choosy about when fontification
+occurs.  Known support modes are Fast Lock mode (symbol `fast-lock-mode') and
+Lazy Lock mode (symbol `lazy-lock-mode').  See those modes for more info.
+If nil, means support for Font Lock mode is never performed.
+If a symbol, use that support mode.
+If a list, each element should be of the form (MAJOR-MODE . SUPPORT-MODE),
+where MAJOR-MODE is a symbol or t (meaning the default).  For example:
+ ((c-mode . fast-lock-mode) (c++-mode . fast-lock-mode) (t . lazy-lock-mode))
+means that Fast Lock mode is used to support Font Lock mode for buffers in C or
+C++ modes, and Lazy Lock mode is used to support Font Lock mode otherwise.
+
+The value of this variable is used when Font Lock mode is turned on.")
+
+(defun font-lock-turn-on-thing-lock ()
+  (let ((thing-mode (font-lock-value-in-major-mode font-lock-support-mode)))
+    (cond ((eq thing-mode 'fast-lock-mode)
+          (fast-lock-mode t))
+         ((eq thing-mode 'lazy-lock-mode)
+          (lazy-lock-mode t)))))
+
+(defvar fast-lock-mode nil)
+(defvar lazy-lock-mode nil)
+
+(defun font-lock-turn-off-thing-lock ()
+  (cond (fast-lock-mode
+        (fast-lock-mode nil))
+       (lazy-lock-mode
+        (lazy-lock-mode nil))))
+
+(defun font-lock-after-fontify-buffer ()
+  (cond (fast-lock-mode
+        (fast-lock-after-fontify-buffer))
+       (lazy-lock-mode
+        (lazy-lock-after-fontify-buffer))))
+
+(defun font-lock-after-unfontify-buffer ()
+  (cond (fast-lock-mode
+        (fast-lock-after-unfontify-buffer))
+       (lazy-lock-mode
+        (lazy-lock-after-unfontify-buffer))))
+
+;; End of Font Lock Support mode.
+\f
+;; Fontification functions.
 
 ;;;###autoload
 (defun font-lock-fontify-buffer ()
   "Fontify the current buffer the way `font-lock-mode' would."
   (interactive)
-  (let ((verbose (and (or font-lock-verbose (interactive-p))
-                     (not (zerop (buffer-size))))))
-    (set (make-local-variable 'font-lock-fontified) nil)
+  (let ((font-lock-verbose (or font-lock-verbose (interactive-p))))
+    (funcall font-lock-fontify-buffer-function)))
+
+(defun font-lock-unfontify-buffer ()
+  (funcall font-lock-unfontify-buffer-function))
+
+(defun font-lock-fontify-region (beg end &optional loudly)
+  (funcall font-lock-fontify-region-function beg end loudly))
+
+(defun font-lock-unfontify-region (beg end)
+  (funcall font-lock-unfontify-region-function beg end))
+
+(defun font-lock-default-fontify-buffer ()
+  (let ((verbose (if (numberp font-lock-verbose)
+                    (> (buffer-size) font-lock-verbose)
+                  font-lock-verbose)))
     (if verbose (message "Fontifying %s..." (buffer-name)))
-    ;; Turn it on to run hooks and get the right `font-lock-keywords' etc.
-    (or font-lock-mode (font-lock-set-defaults))
-    (condition-case nil
-       (save-excursion
-         (save-match-data
-           (font-lock-fontify-region (point-min) (point-max) verbose)
-           (setq font-lock-fontified t)))
-      ;; We don't restore the old fontification, so it's best to unfontify.
-      (quit (font-lock-unfontify-region (point-min) (point-max))))
-    (if verbose (message "Fontifying %s... %s." (buffer-name)
-                        (if font-lock-fontified "done" "aborted")))
-    (font-lock-after-fontify-buffer)))
-\f
-;; Fontification functions.
+    ;; Make sure we have the right `font-lock-keywords' etc.
+    (if (not font-lock-mode) (font-lock-set-defaults))
+    ;; Make sure we fontify etc. in the whole buffer.
+    (save-restriction
+      (widen)
+      (condition-case nil
+         (save-excursion
+           (save-match-data
+             (font-lock-fontify-region (point-min) (point-max) verbose)
+             (font-lock-after-fontify-buffer)
+             (setq font-lock-fontified t)))
+       ;; We don't restore the old fontification, so it's best to unfontify.
+       (quit (font-lock-unfontify-buffer))))
+    (if verbose (message "Fontifying %s...%s" (buffer-name)
+                        (if font-lock-fontified "done" "aborted")))))
+
+(defun font-lock-default-unfontify-buffer ()
+  (save-restriction
+    (widen)
+    (font-lock-unfontify-region (point-min) (point-max))
+    (font-lock-after-unfontify-buffer)
+    (setq font-lock-fontified nil)))
 
 ;; We use this wrapper.  However, `font-lock-fontify-region' used to be the
 ;; name used for `font-lock-fontify-syntactically-region', so a change isn't
 ;; back-compatible.  But you shouldn't be calling these directly, should you?
-(defun font-lock-fontify-region (beg end &optional loudly)
+(defun font-lock-default-fontify-region (beg end loudly)
   (let ((modified (buffer-modified-p))
        (buffer-undo-list t) (inhibit-read-only t)
        (old-syntax-table (syntax-table))
+       before-change-functions after-change-functions
        buffer-file-name buffer-file-truename)
     (unwind-protect
-       (progn
+       (save-restriction
+         (widen)
          ;; Use the fontification syntax table, if any.
          (if font-lock-syntax-table (set-syntax-table font-lock-syntax-table))
          ;; Now do the fontification.
@@ -454,9 +787,10 @@ size, you can use \\[font-lock-fontify-buffer]."
 ;               (or (nth 4 state) (nth 7 state))))
 ;        (font-lock-fontify-keywords-region beg end))
 
-(defun font-lock-unfontify-region (beg end)
+(defun font-lock-default-unfontify-region (beg end)
   (let ((modified (buffer-modified-p))
        (buffer-undo-list t) (inhibit-read-only t)
+       before-change-functions after-change-functions
        buffer-file-name buffer-file-truename)
     (remove-text-properties beg end '(face nil))
     (and (not modified) (buffer-modified-p) (set-buffer-modified-p nil))))
@@ -469,112 +803,146 @@ size, you can use \\[font-lock-fontify-buffer]."
       (font-lock-fontify-region
        (progn (goto-char beg) (beginning-of-line) (point))
        (progn (goto-char end) (forward-line 1) (point))))))
+
+(defun font-lock-fontify-block (&optional arg)
+  "Fontify some lines the way `font-lock-fontify-buffer' would.
+The lines could be a function or paragraph, or a specified number of lines.
+If ARG is given, fontify that many lines before and after point, or 16 lines if
+no ARG is given and `font-lock-mark-block-function' is nil.
+If `font-lock-mark-block-function' non-nil and no ARG is given, it is used to
+delimit the region to fontify."
+  (interactive "P")
+  (let (font-lock-beginning-of-syntax-function deactivate-mark)
+    ;; Make sure we have the right `font-lock-keywords' etc.
+    (if (not font-lock-mode) (font-lock-set-defaults))
+    (save-excursion
+      (save-match-data
+       (condition-case error-data
+           (if (or arg (not font-lock-mark-block-function))
+               (let ((lines (if arg (prefix-numeric-value arg) 16)))
+                 (font-lock-fontify-region
+                  (save-excursion (forward-line (- lines)) (point))
+                  (save-excursion (forward-line lines) (point))))
+             (funcall font-lock-mark-block-function)
+             (font-lock-fontify-region (point) (mark)))
+         ((error quit) (message "Fontifying block...%s" error-data)))))))
+
+(define-key facemenu-keymap "\M-g" 'font-lock-fontify-block)
 \f
 ;; Syntactic fontification functions.
 
+;; These record the parse state at a particular position, always the start of a
+;; line.  Used to make `font-lock-fontify-syntactically-region' faster.
+(defvar font-lock-cache-position nil)
+(defvar font-lock-cache-state nil)
+(make-variable-buffer-local 'font-lock-cache-position)
+(make-variable-buffer-local 'font-lock-cache-state)
+
 (defun font-lock-fontify-syntactically-region (start end &optional loudly)
   "Put proper face on each string and comment between START and END.
 START should be at the beginning of a line."
-  (let ((synstart (if comment-start-skip
-                     (concat "\\s\"\\|" comment-start-skip)
-                   "\\s\""))
-       (comstart (if comment-start-skip
-                     (concat "\\s<\\|" comment-start-skip)
-                   "\\s<"))
+  (let ((synstart (cond (font-lock-comment-start-regexp
+                        (concat "\\s\"\\|" font-lock-comment-start-regexp))
+                       (comment-start-skip
+                        (concat "\\s\"\\|" comment-start-skip))
+                       (t
+                        "\\s\"")))
+       (comstart (cond (font-lock-comment-start-regexp
+                        font-lock-comment-start-regexp)
+                       (comment-start-skip
+                        (concat "\\s<\\|" comment-start-skip))
+                       (t
+                        "\\s<")))
        state prev prevstate)
     (if loudly (message "Fontifying %s... (syntactically...)" (buffer-name)))
-    (save-restriction
-      (widen)
-      (goto-char start)
-      ;;
-      ;; Find the state at the `beginning-of-line' before `start'.
-      (if (eq start font-lock-cache-position)
-         ;; Use the cache for the state of `start'.
-         (setq state font-lock-cache-state)
-       ;; Find the state of `start'.
-       (if (null font-lock-beginning-of-syntax-function)
-           ;; Use the state at the previous cache position, if any, or
-           ;; otherwise calculate from `point-min'.
-           (if (or (null font-lock-cache-position)
-                   (< start font-lock-cache-position))
-               (setq state (parse-partial-sexp (point-min) start))
-             (setq state (parse-partial-sexp font-lock-cache-position start
-                                             nil nil font-lock-cache-state)))
-         ;; Call the function to move outside any syntactic block.
-         (funcall font-lock-beginning-of-syntax-function)
-         (setq state (parse-partial-sexp (point) start)))
-       ;; Cache the state and position of `start'.
-       (setq font-lock-cache-state state
-             font-lock-cache-position start))
-      ;;
-      ;; If the region starts inside a string, show the extent of it.
-      (if (nth 3 state)
-         (let ((beg (point)))
-           (while (and (re-search-forward "\\s\"" end 'move)
-                       (nth 3 (parse-partial-sexp beg (point)
-                                                  nil nil state))))
-           (put-text-property beg (point) 'face font-lock-string-face)
-           (setq state (parse-partial-sexp beg (point) nil nil state))))
-      ;;
-      ;; Likewise for a comment.
-      (if (or (nth 4 state) (nth 7 state))
-         (let ((beg (point)))
-           (save-restriction
-             (narrow-to-region (point-min) end)
-             (condition-case nil
-                 (progn
-                   (re-search-backward comstart (point-min) 'move)
-                   (forward-comment 1)
-                   ;; forward-comment skips all whitespace,
-                   ;; so go back to the real end of the comment.
-                   (skip-chars-backward " \t"))
-               (error (goto-char end))))
-           (put-text-property beg (point) 'face font-lock-comment-face)
-           (setq state (parse-partial-sexp beg (point) nil nil state))))
-      ;;
-      ;; Find each interesting place between here and `end'.
-      (while (and (< (point) end)
-                 (setq prev (point) prevstate state)
-                 (re-search-forward synstart end t)
-                 (progn
-                   ;; Clear out the fonts of what we skip over.
-                   (remove-text-properties prev (point) '(face nil))
-                   ;; Verify the state at that place
-                   ;; so we don't get fooled by \" or \;.
-                   (setq state (parse-partial-sexp prev (point)
-                                                   nil nil state))))
-       (let ((here (point)))
-         (if (or (nth 4 state) (nth 7 state))
+    (goto-char start)
+    ;;
+    ;; Find the state at the `beginning-of-line' before `start'.
+    (if (eq start font-lock-cache-position)
+       ;; Use the cache for the state of `start'.
+       (setq state font-lock-cache-state)
+      ;; Find the state of `start'.
+      (if (null font-lock-beginning-of-syntax-function)
+         ;; Use the state at the previous cache position, if any, or
+         ;; otherwise calculate from `point-min'.
+         (if (or (null font-lock-cache-position)
+                 (< start font-lock-cache-position))
+             (setq state (parse-partial-sexp (point-min) start))
+           (setq state (parse-partial-sexp font-lock-cache-position start
+                                           nil nil font-lock-cache-state)))
+       ;; Call the function to move outside any syntactic block.
+       (funcall font-lock-beginning-of-syntax-function)
+       (setq state (parse-partial-sexp (point) start)))
+      ;; Cache the state and position of `start'.
+      (setq font-lock-cache-state state
+           font-lock-cache-position start))
+    ;;
+    ;; If the region starts inside a string, show the extent of it.
+    (if (nth 3 state)
+       (let ((beg (point)))
+         (while (and (re-search-forward "\\s\"" end 'move)
+                     (nth 3 (parse-partial-sexp beg (point) nil nil state))))
+         (put-text-property beg (point) 'face font-lock-string-face)
+         (setq state (parse-partial-sexp beg (point) nil nil state))))
+    ;;
+    ;; Likewise for a comment.
+    (if (or (nth 4 state) (nth 7 state))
+       (let ((beg (point)))
+         (save-restriction
+           (narrow-to-region (point-min) end)
+           (condition-case nil
+               (progn
+                 (re-search-backward comstart (point-min) 'move)
+                 (forward-comment 1)
+                 ;; forward-comment skips all whitespace,
+                 ;; so go back to the real end of the comment.
+                 (skip-chars-backward " \t"))
+             (error (goto-char end))))
+         (put-text-property beg (point) 'face font-lock-comment-face)
+         (setq state (parse-partial-sexp beg (point) nil nil state))))
+    ;;
+    ;; Find each interesting place between here and `end'.
+    (while (and (< (point) end)
+               (setq prev (point) prevstate state)
+               (re-search-forward synstart end t)
+               (progn
+                 ;; Clear out the fonts of what we skip over.
+                 (remove-text-properties prev (point) '(face nil))
+                 ;; Verify the state at that place
+                 ;; so we don't get fooled by \" or \;.
+                 (setq state (parse-partial-sexp prev (point)
+                                                 nil nil state))))
+      (let ((here (point)))
+       (if (or (nth 4 state) (nth 7 state))
+           ;;
+           ;; We found a real comment start.
+           (let ((beg (or (match-end 1) (match-beginning 0))))
+             (goto-char beg)
+             (save-restriction
+               (narrow-to-region (point-min) end)
+               (condition-case nil
+                   (progn
+                     (forward-comment 1)
+                     ;; forward-comment skips all whitespace,
+                     ;; so go back to the real end of the comment.
+                     (skip-chars-backward " \t"))
+                 (error (goto-char end))))
+             (put-text-property beg (point) 'face font-lock-comment-face)
+             (setq state (parse-partial-sexp here (point) nil nil state)))
+         (if (nth 3 state)
              ;;
-             ;; We found a real comment start.
-             (let ((beg (match-beginning 0)))
-               (goto-char beg)
-               (save-restriction
-                 (narrow-to-region (point-min) end)
-                 (condition-case nil
-                     (progn
-                       (forward-comment 1)
-                       ;; forward-comment skips all whitespace,
-                       ;; so go back to the real end of the comment.
-                       (skip-chars-backward " \t"))
-                   (error (goto-char end))))
-               (put-text-property beg (point) 'face
-                                  font-lock-comment-face)
-               (setq state (parse-partial-sexp here (point) nil nil state)))
-           (if (nth 3 state)
-               ;;
-               ;; We found a real string start.
-               (let ((beg (match-beginning 0)))
-                 (while (and (re-search-forward "\\s\"" end 'move)
-                             (nth 3 (parse-partial-sexp here (point)
-                                                        nil nil state))))
-                 (put-text-property beg (point) 'face font-lock-string-face)
-                 (setq state (parse-partial-sexp here (point)
-                                                 nil nil state))))))
-       ;;
-       ;; Make sure `prev' is non-nil after the loop
-       ;; only if it was set on the very last iteration.
-       (setq prev nil)))
+             ;; We found a real string start.
+             (let ((beg (or (match-end 1) (match-beginning 0))))
+               (while (and (re-search-forward "\\s\"" end 'move)
+                           (nth 3 (parse-partial-sexp here (point)
+                                                      nil nil state))))
+               (put-text-property beg (point) 'face font-lock-string-face)
+               (setq state (parse-partial-sexp here (point)
+                                               nil nil state))))))
+      ;;
+      ;; Make sure `prev' is non-nil after the loop
+      ;; only if it was set on the very last iteration.
+      (setq prev nil))
     ;;
     ;; Clean up.
     (and prev (remove-text-properties prev end '(face nil)))))
@@ -741,108 +1109,122 @@ START should be at the beginning of a line."
 \f
 ;; Various functions.
 
-;; Turn off other related packages if they're on.  I prefer a hook. --sm.
-;; These explicit calls are easier to understand
-;; because people know what they will do.
-;; A hook is a mystery because it might do anything whatever. --rms.
-(defun font-lock-thing-lock-cleanup ()
-  (cond ((and (boundp 'fast-lock-mode) fast-lock-mode)
-        (fast-lock-mode -1))
-       ((and (boundp 'lazy-lock-mode) lazy-lock-mode)
-        (lazy-lock-mode -1))))
-
-;; Do something special for these packages after fontifying.  I prefer a hook.
-(defun font-lock-after-fontify-buffer ()
-  (cond ((and (boundp 'fast-lock-mode) fast-lock-mode)
-        (fast-lock-after-fontify-buffer))
-       ((and (boundp 'lazy-lock-mode) lazy-lock-mode)
-        (lazy-lock-after-fontify-buffer))))
-
-;; If the buffer is about to be reverted, it won't be fontified afterward.
-(defun font-lock-revert-setup ()
-  (setq font-lock-fontified nil))
-
-;; If the buffer has just been reverted, normally that turns off
-;; Font Lock mode.  So turn the mode back on if necessary.
-(defalias 'font-lock-revert-cleanup 'turn-on-font-lock)
-
 (defun font-lock-compile-keywords (&optional keywords)
   ;; Compile `font-lock-keywords' into the form (t KEYWORD ...) where KEYWORD
   ;; is the (MATCHER HIGHLIGHT ...) shown in the variable's doc string.
   (let ((keywords (or keywords font-lock-keywords)))
     (setq font-lock-keywords 
-     (if (eq (car-safe keywords) t)
-        keywords
-       (cons t
-       (mapcar
-        (function (lambda (item)
-           (cond ((nlistp item)
-                  (list item '(0 font-lock-keyword-face)))
-                 ((numberp (cdr item))
-                  (list (car item) (list (cdr item) 'font-lock-keyword-face)))
-                 ((symbolp (cdr item))
-                  (list (car item) (list 0 (cdr item))))
-                 ((nlistp (nth 1 item))
-                  (list (car item) (cdr item)))
-                 (t
-                  item))))
-        keywords))))))
+         (if (eq (car-safe keywords) t)
+             keywords
+           (cons t (mapcar 'font-lock-compile-keyword keywords))))))
+
+(defun font-lock-compile-keyword (keyword)
+  (cond ((nlistp keyword)              ; Just MATCHER
+        (list keyword '(0 font-lock-keyword-face)))
+       ((eq (car keyword) 'eval)       ; Specified (eval . FORM)
+        (font-lock-compile-keyword (eval (cdr keyword))))
+       ((numberp (cdr keyword))        ; Specified (MATCHER . MATCH)
+        (list (car keyword) (list (cdr keyword) 'font-lock-keyword-face)))
+       ((symbolp (cdr keyword))        ; Specified (MATCHER . FACENAME)
+        (list (car keyword) (list 0 (cdr keyword))))
+       ((nlistp (nth 1 keyword))       ; Specified (MATCHER . HIGHLIGHT)
+        (list (car keyword) (cdr keyword)))
+       (t                              ; Hopefully (MATCHER HIGHLIGHT ...)
+        keyword)))
+
+(defun font-lock-value-in-major-mode (alist)
+  ;; Return value in ALIST for `major-mode', or ALIST if it is not an alist.
+  ;; Alist structure is ((MAJOR-MODE . VALUE)) where MAJOR-MODE may be t.
+  (if (consp alist)
+      (cdr (or (assq major-mode alist) (assq t alist)))
+    alist))
 
 (defun font-lock-choose-keywords (keywords level)
   ;; Return LEVELth element of KEYWORDS.  A LEVEL of nil is equal to a
   ;; LEVEL of 0, a LEVEL of t is equal to (1- (length KEYWORDS)).
-  (let ((level (if (not (consp level))
-                  level
-                (cdr (or (assq major-mode level) (assq t level))))))
-    (cond ((symbolp keywords)
-          keywords)
-         ((numberp level)
-          (or (nth level keywords) (car (reverse keywords))))
-         ((eq level t)
-          (car (reverse keywords)))
-         (t
-          (car keywords)))))
+  (cond ((symbolp keywords)
+        keywords)
+       ((numberp level)
+        (or (nth level keywords) (car (reverse keywords))))
+       ((eq level t)
+        (car (reverse keywords)))
+       (t
+        (car keywords))))
 
 (defun font-lock-set-defaults ()
   "Set fontification defaults appropriately for this mode.
-Sets `font-lock-keywords', `font-lock-keywords-only', `font-lock-syntax-table',
-`font-lock-beginning-of-syntax-function' and
-`font-lock-keywords-case-fold-search' using `font-lock-defaults' (or, if nil,
-using `font-lock-defaults-alist') and `font-lock-maximum-decoration'."
+Sets various variables using `font-lock-defaults' (or, if nil, using
+`font-lock-defaults-alist') and `font-lock-maximum-decoration'."
   ;; Set face defaults.
   (font-lock-make-faces)
   ;; Set fontification defaults.
-  (or font-lock-keywords
+  (make-local-variable 'font-lock-fontified)
+  (if (member font-lock-keywords '(nil (t)))
       (let* ((defaults (or font-lock-defaults
                           (cdr (assq major-mode font-lock-defaults-alist))))
-            (keywords (font-lock-choose-keywords
-                       (nth 0 defaults) font-lock-maximum-decoration)))
-       ;; Keywords?
+            (keywords
+             (font-lock-choose-keywords (nth 0 defaults)
+              (font-lock-value-in-major-mode font-lock-maximum-decoration))))
+       ;; Regexp fontification?
        (setq font-lock-keywords (if (fboundp keywords)
                                     (funcall keywords)
                                   (eval keywords)))
-       ;; Syntactic?
+       ;; Syntactic fontification?
        (if (nth 1 defaults)
            (set (make-local-variable 'font-lock-keywords-only) t))
-       ;; Case fold?
+       ;; Case fold during regexp fontification?
        (if (nth 2 defaults)
            (set (make-local-variable 'font-lock-keywords-case-fold-search) t))
-       ;; Syntax table?
+       ;; Syntax table for regexp and syntactic fontification?
        (if (nth 3 defaults)
            (let ((slist (nth 3 defaults)))
              (set (make-local-variable 'font-lock-syntax-table)
                   (copy-syntax-table (syntax-table)))
              (while slist
-               (modify-syntax-entry (car (car slist)) (cdr (car slist))
-                                    font-lock-syntax-table)
-               (setq slist (cdr slist)))))
-       ;; Syntax function?
+               ;; The character to modify may be a single CHAR or a STRING.
+               (let ((chars (if (numberp (car (car slist)))
+                                (list (car (car slist)))
+                              (mapcar 'identity (car (car slist)))))
+                     (syntax (cdr (car slist))))
+                 (while chars
+                   (modify-syntax-entry (car chars) syntax
+                                        font-lock-syntax-table)
+                   (setq chars (cdr chars)))
+                 (setq slist (cdr slist))))))
+       ;; Syntax function for syntactic fontification?
        (if (nth 4 defaults)
            (set (make-local-variable 'font-lock-beginning-of-syntax-function)
-                (nth 4 defaults))))))
+                (nth 4 defaults)))
+       ;; Variable alist?
+       (let ((alist (nthcdr 5 defaults)))
+         (while alist
+           (set (make-local-variable (car (car alist))) (cdr (car alist)))
+           (setq alist (cdr alist)))))))
+
+(defun font-lock-unset-defaults ()
+  "Unset fontification defaults.  See `font-lock-set-defaults'."
+  (setq font-lock-keywords                     nil
+       font-lock-keywords-only                 nil
+       font-lock-keywords-case-fold-search     nil
+       font-lock-syntax-table                  nil
+       font-lock-beginning-of-syntax-function  nil)
+  (let* ((defaults (or font-lock-defaults
+                      (cdr (assq major-mode font-lock-defaults-alist))))
+        (alist (nthcdr 5 defaults)))
+    (while alist
+      (set (car (car alist)) (default-value (car (car alist))))
+      (setq alist (cdr alist)))))
 \f
 ;; Colour etc. support.
 
+;; This section of code is crying out for revision.
+
+;; To begin with, `display-type' and `background-mode' are `frame-parameters'
+;; so we don't have to calculate them here anymore.  But all the face stuff
+;; should be frame-local (and thus display-local) anyway.  Because we're not
+;; sure what support Emacs is going to have for general frame-local face
+;; attributes, we leave this section of code as it is.  For now.  --sm.
+
 (defvar font-lock-display-type nil
   "A symbol indicating the display Emacs is running under.
 The symbol should be one of `color', `grayscale' or `mono'.
@@ -904,9 +1286,14 @@ See `font-lock-make-face' and `list-faces-display'."
                                               "BackgroundMode"))
                  (params (frame-parameters)))
              (cond (bg-resource (intern (downcase bg-resource)))
+                   ((eq system-type 'ms-dos)
+                    (if (string-match "light"
+                                      (cdr (assq 'background-color params)))
+                        'light
+                      'dark))
                    ((< (apply '+ (x-color-values
                                   (cdr (assq 'background-color params))))
-                       (/ (apply '+ (x-color-values "white")) 3))
+                       (* (apply '+ (x-color-values "white")) .6))
                     'dark)
                    (t 'light)))))
   (if (null font-lock-face-attributes)
@@ -1035,14 +1422,14 @@ the face is also set; its value is the face name."
                 "\\)\\)\\>"
                 ;; Any whitespace and declared object.
                 "[ \t'\(]*"
-                "\\([^ \t\n\)]+\\)?")
+                "\\(\\sw+\\)?")
         '(1 font-lock-keyword-face)
         '(8 (cond ((match-beginning 3) font-lock-variable-name-face)
                   ((match-beginning 6) font-lock-type-face)
                   (t font-lock-function-name-face))
             nil t))
    )
- "Subdued level highlighting Lisp modes.")
+ "Subdued level highlighting for Lisp modes.")
 
 (defconst lisp-font-lock-keywords-2
   (append lisp-font-lock-keywords-1
@@ -1050,23 +1437,29 @@ the face is also set; its value is the face name."
     ;;
     ;; Control structures.  ELisp and CLisp combined.
 ;      (make-regexp
-;       '("cond" "if" "while" "let\\*?" "prog[nv12*]?" "catch" "throw"
+;       '("cond" "if" "while" "let\\*?" "prog[nv12*]?" "inline" "catch" "throw"
 ;       "save-restriction" "save-excursion" "save-window-excursion"
 ;       "save-selected-window" "save-match-data" "unwind-protect"
 ;       "condition-case" "track-mouse"
 ;       "eval-after-load" "eval-and-compile" "eval-when-compile"
-;       "when" "unless" "do" "flet" "labels" "return" "return-from"))
+;       "when" "unless" "do" "flet" "labels" "return" "return-from"
+;       "with-output-to-temp-buffer" "with-timeout"))
     (cons
      (concat
       "(\\("
-      "\\(c\\(atch\\|ond\\(\\|ition-case\\)\\)\\|do\\|"
+      "c\\(atch\\|ond\\(\\|ition-case\\)\\)\\|do\\|"
       "eval-\\(a\\(fter-load\\|nd-compile\\)\\|when-compile\\)\\|flet\\|"
-      "if\\|l\\(abels\\|et\\*?\\)\\|prog[nv12*]?\\|return\\(\\|-from\\)\\|"
-      "save-\\(excursion\\|match-data\\|restriction\\|selected-window\\|"
-      "window-excursion\\)\\|t\\(hrow\\|rack-mouse\\)\\|"
-      "un\\(less\\|wind-protect\\)\\|wh\\(en\\|ile\\)\\)"
+      "i\\(f\\|nline\\)\\|l\\(abels\\|et\\*?\\)\\|prog[nv12*]?\\|"
+      "return\\(\\|-from\\)\\|save-\\(excursion\\|match-data\\|restriction\\|"
+      "selected-window\\|window-excursion\\)\\|t\\(hrow\\|rack-mouse\\)\\|"
+      "un\\(less\\|wind-protect\\)\\|"
+      "w\\(h\\(en\\|ile\\)\\|ith-\\(output-to-temp-buffer\\|timeout\\)\\)"
       "\\)\\>") 1)
     ;;
+    ;; Feature symbols as references.
+    '("(\\(featurep\\|provide\\|require\\)\\>[ \t']*\\(\\sw+\\)?"
+      (1 font-lock-keyword-face) (2 font-lock-reference-face nil t))
+    ;;
     ;; Words inside \\[] tend to be for `substitute-command-keys'.
     '("\\\\\\\\\\[\\(\\sw+\\)]" 1 font-lock-reference-face prepend)
     ;;
@@ -1077,7 +1470,7 @@ the face is also set; its value is the face name."
     '("\\<:\\sw+\\>" 0 font-lock-reference-face prepend)
     ;;
     ;; ELisp and CLisp `&' keywords as types.
-    '("\\<\\&\\(optional\\|rest\\|whole\\)\\>" . font-lock-type-face)
+    '("\\<\\&\\sw+\\>" . font-lock-type-face)
     ))
   "Gaudy level highlighting for Lisp modes.")
 
@@ -1159,7 +1552,15 @@ the face is also set; its value is the face name."
   ;; The expect syntax of an item is "word" or "word::word", possibly ending
   ;; with optional whitespace and a "(".  Everything following the item (but
   ;; belonging to it) is expected to by skip-able by `forward-sexp', and items
-  ;; are expected to be separated with a "," or ";".
+  ;; are expected to be separated with a ",".
+  ;;
+  ;; The regexp matches:       word::word (
+  ;;                           ^^^^  ^^^^ ^
+  ;; Match subexps are:                  1     3  4
+  ;;
+  ;; So, the item is delimited by (match-beginning 1) and (match-end 1).
+  ;; If (match-beginning 3) is non-nil, that part of the item follows a ":".
+  ;; If (match-beginning 4) is non-nil, the item is followed by a "(".
   (if (looking-at "[ \t*&]*\\(\\sw+\\)\\(::\\(\\sw+\\)\\)?[ \t]*\\((\\)?")
       (save-match-data
        (condition-case nil
@@ -1168,9 +1569,9 @@ the face is also set; its value is the face name."
              (narrow-to-region (point-min) limit)
              (goto-char (match-end 1))
              ;; Move over any item value, etc., to the next item.
-             (while (not (looking-at "[ \t]*\\([,;]\\|$\\)"))
+             (while (not (looking-at "[ \t]*\\(\\(,\\)\\|;\\|$\\)"))
                (goto-char (or (scan-sexps (point) 1) (point-max))))
-             (goto-char (match-end 0)))
+             (goto-char (match-end 2)))
          (error t)))))
 
 (let ((c-keywords
@@ -1216,15 +1617,15 @@ the face is also set; its value is the face name."
    '("^#[ \t]*include[ \t]+\\(<[^>\"\n]+>\\)" 1 font-lock-string-face)
    ;;
    ;; Fontify function macro names.
-   '("^#[ \t]*define[ \t]+\\(\\(\\sw+\\)(\\)" 2 font-lock-function-name-face)
+   '("^#[ \t]*define[ \t]+\\(\\sw+\\)(" 1 font-lock-function-name-face)
    ;;
-   ;; Fontify symbol names in #if ... defined preprocessor directives.
-   '("^#[ \t]*if\\>"
+   ;; Fontify symbol names in #elif or #if ... defined preprocessor directives.
+   '("^#[ \t]*\\(elif\\|if\\)\\>"
      ("\\<\\(defined\\)\\>[ \t]*(?\\(\\sw+\\)?" nil nil
       (1 font-lock-reference-face) (2 font-lock-variable-name-face nil t)))
    ;;
    ;; Fontify otherwise as symbol names, and the preprocessor directive names.
-   '("^\\(#[ \t]*[a-z]+\\)\\>[ \t]*\\(\\sw+\\)?"
+   '("^#[ \t]*\\(\\sw+\\)\\>[ \t]*\\(\\sw+\\)?"
      (1 font-lock-reference-face) (2 font-lock-variable-name-face nil t))
    ))
 
@@ -1241,7 +1642,7 @@ the face is also set; its value is the face name."
     (cons (concat "\\<\\(" c-keywords "\\)\\>") 'font-lock-keyword-face)
     ;;
     ;; Fontify case/goto keywords and targets, and case default/goto tags.
-    '("\\<\\(case\\|goto\\)\\>[ \t]*\\([^ \t\n:;]+\\)?"
+    '("\\<\\(case\\|goto\\)\\>[ \t]*\\(\\sw+\\)?"
       (1 font-lock-keyword-face) (2 font-lock-reference-face nil t))
     '("^[ \t]*\\(\\sw+\\)[ \t]*:" 1 font-lock-reference-face)
     )))
@@ -1297,7 +1698,7 @@ the face is also set; its value is the face name."
       (1 (if (match-beginning 2)
             font-lock-type-face
           font-lock-function-name-face))
-      (3 (if (match-beginning 2) font-lock-function-name-face) nil t))
+      (3 font-lock-function-name-face nil t))
     )))
 
  (setq c++-font-lock-keywords-2
@@ -1308,11 +1709,11 @@ the face is also set; its value is the face name."
     (cons (concat "\\<\\(" c++-type-types "\\)\\>") 'font-lock-type-face)
     ;;
     ;; Fontify operator function name overloading.
-    '("\\<\\(operator\\)\\>[ \t]*\\([][)(><!=+-][][)(><!=+-]?\\)?"
+    '("\\<\\(operator\\)\\>[ \t]*\\([[(><!=+-][])><=+-]?\\)?"
       (1 font-lock-keyword-face) (2 font-lock-function-name-face nil t))
     ;;
     ;; Fontify case/goto keywords and targets, and case default/goto tags.
-    '("\\<\\(case\\|goto\\)\\>[ \t]*\\([^ \t\n:;]+\\)?"
+    '("\\<\\(case\\|goto\\)\\>[ \t]*\\(\\sw+\\)?"
       (1 font-lock-keyword-face) (2 font-lock-reference-face nil t))
     '("^[ \t]*\\(\\sw+\\)[ \t]*:[^:]" 1 font-lock-reference-face)
     ;;
@@ -1405,8 +1806,8 @@ the face is also set; its value is the face name."
 \f
 ;; Install ourselves:
 
-(or (assq 'font-lock-mode minor-mode-alist)
-    (setq minor-mode-alist (cons '(font-lock-mode " Font") minor-mode-alist)))
+(unless (assq 'font-lock-mode minor-mode-alist)
+  (setq minor-mode-alist (cons '(font-lock-mode " Font") minor-mode-alist)))
 
 ;; Provide ourselves: