]> git.eshelyaron.com Git - emacs.git/commitdiff
* progmodes/grep.el: New file with grep code from compile.el.
authorKim F. Storm <storm@cua.dk>
Sun, 23 Nov 2003 20:56:10 +0000 (20:56 +0000)
committerKim F. Storm <storm@cua.dk>
Sun, 23 Nov 2003 20:56:10 +0000 (20:56 +0000)
(grep): New defcustom group.
(grep-window-height): New defcustom, like compilation-window-height.
(grep-auto-highlight): New defcustom, like compile-auto-highlight.
(grep-scroll-output): New defcustom, like compilation-scroll-output.
(grep-command, grep-use-null-device, grep-find-command)
(grep-tree-files-aliases, grep-tree-ignore-case)
(grep-tree-ignore-CVS-directories): Move to grep custom group.
(grep-setup-hook): New hook variable.
(grep-mode-map): New keymap for grep commands.  Add Grep menu.
(grep-use-compilation-buffer): New defcustom.
(grep-last-buffer): New defvar, override compilation-last-buffer.
(grep): Add optional arg HIGHLIGHT-REGEXP.  Doc fix.
Call compile-internal with args highlight-regexp and grep-mode-map.

lisp/progmodes/grep.el [new file with mode: 0644]

diff --git a/lisp/progmodes/grep.el b/lisp/progmodes/grep.el
new file mode 100644 (file)
index 0000000..2793280
--- /dev/null
@@ -0,0 +1,534 @@
+;;; grep.el --- run compiler as inferior of Emacs, parse error messages
+
+;; Copyright (C) 1985, 86, 87, 93, 94, 95, 96, 97, 98, 1999, 2001, 2002
+;;  Free Software Foundation, Inc.
+
+;; Author: Roland McGrath <roland@gnu.org>
+;; Maintainer: FSF
+;; Keywords: tools, processes
+
+;; 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
+;; 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:
+
+;; This package provides the grep facilities documented in the Emacs
+;; user's manual.
+
+;;; Code:
+
+(require 'compile)
+
+(defgroup grep nil
+  "Run compiler as inferior of Emacs, parse error messages."
+  :group 'tools
+  :group 'processes)
+
+
+;;;###autoload
+(defcustom grep-window-height nil
+  "*Number of lines in a grep window.  If nil, use `compilation-window-height'."
+  :type '(choice (const :tag "Default" nil)
+                integer)
+  :version "21.4"
+  :group 'grep)
+
+(defcustom grep-auto-highlight t
+  "*Specify how many grep matches to highlight (and parse) initially.
+\(Highlighting applies to an grep match when the mouse is over it.)
+If this is a number N, all grep matches in the first N lines
+are highlighted and parsed as soon as they arrive in Emacs.
+If t, highlight and parse the whole grep output as soon as it arrives.
+If nil, don't highlight or parse any of the grep buffer until you try to
+move to the error messages.
+
+Those grep matches which are not parsed and highlighted initially
+will be parsed and highlighted as soon as you try to move to them."
+  :type '(choice (const :tag "All" t)
+                (const :tag "None" nil)
+                (integer :tag "First N lines"))
+  :version "21.4"
+  :group 'grep)
+
+(defcustom grep-scroll-output nil
+  "*Non-nil to scroll the *grep* buffer window as output appears.
+
+Setting it causes the grep commands to put point at the end of their
+output window so that the end of the output is always visible rather
+than the begining."
+  :type 'boolean
+  :version "21.4"
+  :group 'grep)
+
+(defcustom grep-command nil
+  "The default grep command for \\[grep].
+If the grep program used supports an option to always include file names
+in its output (such as the `-H' option to GNU grep), it's a good idea to
+include it when specifying `grep-command'.
+
+The default value of this variable is set up by `grep-compute-defaults';
+call that function before using this variable in your program."
+  :type '(choice string
+                (const :tag "Not Set" nil))
+  :group 'grep)
+
+(defcustom grep-use-null-device 'auto-detect
+  "If t, append the value of `null-device' to `grep' commands.
+This is done to ensure that the output of grep includes the filename of
+any match in the case where only a single file is searched, and is not
+necessary if the grep program used supports the `-H' option.
+
+The default value of this variable is set up by `grep-compute-defaults';
+call that function before using this variable in your program."
+  :type 'boolean
+  :type '(choice (const :tag "Do Not Append Null Device" nil)
+                (const :tag "Append Null Device" t)
+                (other :tag "Not Set" auto-detect))
+  :group 'grep)
+
+(defcustom grep-find-command nil
+  "The default find command for \\[grep-find].
+The default value of this variable is set up by `grep-compute-defaults';
+call that function before using this variable in your program."
+  :type '(choice string
+                (const :tag "Not Set" nil))
+  :group 'grep)
+
+(defcustom grep-tree-command nil
+  "The default find command for \\[grep-tree].
+The default value of this variable is set up by `grep-compute-defaults';
+call that function before using this variable in your program.
+The following place holders should be present in the string:
+ <D> - base directory for find
+ <X> - find options to restrict or expand the directory list
+ <F> - find options to limit the files matched
+ <C> - place to put -i if case insensitive grep
+ <R> - the regular expression searched for."
+  :type '(choice string
+                (const :tag "Not Set" nil))
+  :version "21.4"
+  :group 'grep)
+
+(defcustom grep-tree-files-aliases '(
+       ("ch" . "*.[ch]")
+       ("c" .  "*.c")
+       ("h" .  "*.h")
+       ("m" .  "[Mm]akefile*")
+       ("asm" . "*.[sS]")
+       ("all" . "*")
+       ("el" . "*.el")
+       )
+  "*Alist of aliases for the FILES argument to `grep-tree'."
+  :type 'alist
+  :group 'grep)
+
+(defcustom grep-tree-ignore-case t
+  "*If non-nil, `grep-tree' ignores case in matches."
+  :type 'boolean
+  :group 'grep)
+
+(defcustom grep-tree-ignore-CVS-directories t
+  "*If non-nil, `grep-tree' does no recurse into CVS directories."
+  :type 'boolean
+  :group 'grep)
+
+;;;###autoload
+(defcustom grep-setup-hook nil
+  "List of hook functions run by `grep-process-setup' (see `run-hooks')."
+  :type 'hook
+  :group 'grep)
+
+(defvar grep-mode-map
+  (let ((map (cons 'keymap compilation-minor-mode-map)))
+    (define-key map " " 'scroll-up)
+    (define-key map "\^?" 'scroll-down)
+
+    (define-key map [remap next-line] 'compilation-next-error)
+    (define-key map [remap previous-line] 'compilation-previous-error)
+
+    (define-key map "\r" 'compile-goto-error)  ;; ?
+    (define-key map "n" 'next-error-no-select)
+    (define-key map "p" 'previous-error-no-select)
+    (define-key map "{" 'compilation-previous-file)
+    (define-key map "}" 'compilation-next-file)
+    (define-key map "\t" 'compilation-next-file)
+
+    ;; Set up the menu-bar
+    (define-key map [menu-bar grep]
+      (cons "Grep" (make-sparse-keymap "Grep")))
+
+    (define-key map [menu-bar grep compilation-kill-compilation]
+      '("Kill Grep" . kill-compilation))
+    (define-key map [menu-bar grep compilation-separator2]
+      '("----" . nil))
+    (define-key map [menu-bar grep compilation-compile]
+      '("Compile..." . compile))
+    (define-key map [menu-bar grep compilation-grep]
+      '("Another grep" . grep))
+    (define-key map [menu-bar grep compilation-recompile]
+      '("Repeat grep" . recompile))
+    (define-key map [menu-bar grep compilation-separator2]
+      '("----" . nil))
+    (define-key map [menu-bar grep compilation-first-error]
+      '("First Match" . first-error))
+    (define-key map [menu-bar grep compilation-previous-error]
+      '("Previous Match" . previous-error))
+    (define-key map [menu-bar grep compilation-next-error]
+      '("Next Match" . next-error))
+    map)
+  "Keymap for grep buffers.
+`compilation-minor-mode-map' is a cdr of this.")
+
+;;;; TODO --- refine this!!
+
+(defcustom grep-use-compilation-buffer t
+  "When non-nil, grep specific commands update `compilation-last-buffer'.
+This means that standard compile commands like \\[next-error] and \\[compile-goto-error]
+can be used to navigate between grep matches (the default).
+Otherwise, the grep specific commands like \\[grep-next-match] must
+be used to navigate between grep matches."
+  :type 'boolean
+  :group 'grep)
+
+;; override compilation-last-buffer
+(defvar grep-last-buffer nil
+  "The most recent grep buffer.
+A grep buffer becomes most recent when its process is started
+or when it is used with \\[grep-next-match].
+Notice that using \\[next-error] or \\[compile-goto-error] modifies
+`complation-last-buffer' rather than `grep-last-buffer'.")
+
+;; Note: the character class after the optional drive letter does not
+;; include a space to support file names with blanks.
+(defvar grep-regexp-alist
+  '(("\\([a-zA-Z]?:?[^:(\t\n]+\\)[:( \t]+\\([0-9]+\\)[:) \t]" 1 2))
+  "Regexp used to match grep hits.  See `compilation-error-regexp-alist'.")
+
+(defvar grep-program
+  ;; Currently zgrep has trouble.  It runs egrep instead of grep,
+  ;; and it doesn't pass along long options right.
+  "grep"
+  ;; (if (equal (condition-case nil    ; in case "zgrep" isn't in exec-path
+  ;;            (call-process "zgrep" nil nil nil
+  ;;                          "foo" null-device)
+  ;;          (error nil))
+  ;;        1)
+  ;;     "zgrep"
+  ;;   "grep")
+  "The default grep program for `grep-command' and `grep-find-command'.
+This variable's value takes effect when `grep-compute-defaults' is called.")
+
+(defvar find-program "find"
+  "The default find program for `grep-find-command'.
+This variable's value takes effect when `grep-compute-defaults' is called.")
+
+(defvar grep-find-use-xargs nil
+  "Whether \\[grep-find] uses the `xargs' utility by default.
+
+If nil, it uses `find -exec'; if `gnu', it uses `find -print0' and `xargs -0';
+if not nil and not `gnu', it uses `find -print' and `xargs'.
+
+This variable's value takes effect when `grep-compute-defaults' is called.")
+
+;; History of grep commands.
+(defvar grep-history nil)
+(defvar grep-find-history nil)
+
+(defun grep-process-setup ()
+  "Setup compilation variables and buffer for `grep'.
+Set up `compilation-exit-message-function' and `compilation-window-height'.
+Sets `grep-last-buffer' and runs `grep-setup-hook'."
+  (setq grep-last-buffer (current-buffer))
+  (set (make-local-variable 'compilation-exit-message-function)
+       (lambda (status code msg)
+        (if (eq status 'exit)
+            (cond ((zerop code)
+                   '("finished (matches found)\n" . "matched"))
+                  ((= code 1)
+                   '("finished with no matches found\n" . "no match"))
+                  (t
+                   (cons msg code)))
+          (cons msg code))))
+  (if grep-window-height
+      (set (make-local-variable 'compilation-window-height)
+          grep-window-height))
+  (set (make-local-variable 'compile-auto-highlight)
+       grep-auto-highlight)
+  (set (make-local-variable 'compilation-scroll-output)
+       grep-scroll-output)
+  (run-hooks 'grep-setup-hook))
+
+(defun grep-compute-defaults ()
+  (unless (or (not grep-use-null-device) (eq grep-use-null-device t))
+    (setq grep-use-null-device
+         (with-temp-buffer
+           (let ((hello-file (expand-file-name "HELLO" data-directory)))
+             (not
+              (and (equal (condition-case nil
+                              (if grep-command
+                                  ;; `grep-command' is already set, so
+                                  ;; use that for testing.
+                                  (call-process-shell-command
+                                   grep-command nil t nil
+                                   "^English" hello-file)
+                                ;; otherwise use `grep-program'
+                                (call-process grep-program nil t nil
+                                              "-nH" "^English" hello-file))
+                            (error nil))
+                          0)
+                   (progn
+                     (goto-char (point-min))
+                     (looking-at
+                      (concat (regexp-quote hello-file)
+                              ":[0-9]+:English")))))))))
+  (unless grep-command
+    (setq grep-command
+         (let ((required-options (if grep-use-null-device "-n" "-nH")))
+           (if (equal (condition-case nil ; in case "grep" isn't in exec-path
+                          (call-process grep-program nil nil nil
+                                        "-e" "foo" null-device)
+                        (error nil))
+                      1)
+               (format "%s %s -e " grep-program required-options)
+             (format "%s %s " grep-program required-options)))))
+  (unless grep-find-use-xargs
+    (setq grep-find-use-xargs
+         (if (and
+               (equal (call-process "find" nil nil nil
+                                    null-device "-print0")
+                      0)
+               (equal (call-process "xargs" nil nil nil
+                                    "-0" "-e" "echo")
+                     0))
+             'gnu)))
+  (unless grep-find-command
+    (setq grep-find-command
+         (cond ((eq grep-find-use-xargs 'gnu)
+                (format "%s . -type f -print0 | xargs -0 -e %s"
+                        find-program grep-command))
+               (grep-find-use-xargs
+                (format "%s . -type f -print | xargs %s"
+                         find-program grep-command))
+               (t (cons (format "%s . -type f -exec %s {} %s \\;"
+                                find-program grep-command null-device)
+                        (+ 22 (length grep-command)))))))
+  (unless grep-tree-command
+    (setq grep-tree-command
+         (let* ((glen (length grep-program))
+                (gcmd (concat grep-program " <C>" (substring grep-command glen))))
+           (cond ((eq grep-find-use-xargs 'gnu)
+                  (format "%s <D> <X> -type f <F> -print0 | xargs -0 -e %s <R>"
+                          find-program gcmd))
+                 (grep-find-use-xargs
+                  (format "%s <D> <X> -type f <F> -print | xargs %s <R>"
+                          find-program gcmd))
+                 (t (format "%s <D> <X> -type f <F> -exec %s <R> {} %s \\;"
+                            find-program gcmd null-device)))))))
+
+(defun grep-default-command ()
+  (let ((tag-default
+        (funcall (or find-tag-default-function
+                     (get major-mode 'find-tag-default-function)
+                     ;; We use grep-tag-default instead of
+                     ;; find-tag-default, to avoid loading etags.
+                     'grep-tag-default)))
+       (sh-arg-re "\\(\\(?:\"\\(?:[^\"]\\|\\\\\"\\)+\"\\|'[^']+'\\|[^\"' \t\n]\\)+\\)")
+       (grep-default (or (car grep-history) grep-command)))
+    ;; Replace the thing matching for with that around cursor.
+    (when (or (string-match
+              (concat "[^ ]+\\s +\\(?:-[^ ]+\\s +\\)*"
+                      sh-arg-re "\\(\\s +\\(\\S +\\)\\)?")
+              grep-default)
+             ;; If the string is not yet complete.
+             (string-match "\\(\\)\\'" grep-default))
+      (unless (or (not (stringp buffer-file-name))
+                 (when (match-beginning 2)
+                   (save-match-data
+                     (string-match
+                      (wildcard-to-regexp
+                       (file-name-nondirectory
+                        (match-string 3 grep-default)))
+                      (file-name-nondirectory buffer-file-name)))))
+       (setq grep-default (concat (substring grep-default
+                                             0 (match-beginning 2))
+                                  " *."
+                                  (file-name-extension buffer-file-name))))
+      (replace-match (or tag-default "") t t grep-default 1))))
+
+;;;###autoload
+(defun grep (command-args &optional highlight-regexp)
+  "Run grep, with user-specified args, and collect output in a buffer.
+While grep runs asynchronously, you can use \\[next-error] (M-x next-error),
+or \\<grep-minor-mode-map>\\[compile-goto-error] in the grep \
+output buffer, to go to the lines
+where grep found matches.
+
+This command uses a special history list for its COMMAND-ARGS, so you can
+easily repeat a grep command.
+
+A prefix argument says to default the argument based upon the current
+tag the cursor is over, substituting it into the last grep command
+in the grep command history (or into `grep-command'
+if that history list is empty).
+
+If specified, optional second arg HIGHLIGHT-REGEXP is the regexp to
+temporarily highlight in visited source lines."
+  (interactive
+   (progn
+     (unless (and grep-command
+                 (or (not grep-use-null-device) (eq grep-use-null-device t)))
+       (grep-compute-defaults))
+     (let ((default (grep-default-command)))
+       (list (read-from-minibuffer "Run grep (like this): "
+                                  (if current-prefix-arg
+                                      default grep-command)
+                                  nil nil 'grep-history
+                                  (if current-prefix-arg nil default))))))
+
+  ;; Setting process-setup-function makes exit-message-function work
+  ;; even when async processes aren't supported.
+  (let* ((compilation-process-setup-function 'grep-process-setup)
+        (buf (compile-internal (if (and grep-use-null-device null-device)
+                                   (concat command-args " " null-device)
+                                 command-args)
+                               "No more grep hits" "grep"
+                               ;; Give it a simpler regexp to match.
+                               nil grep-regexp-alist
+                               nil nil nil nil nil nil
+                               highlight-regexp grep-mode-map)))))
+
+;; This is a copy of find-tag-default from etags.el.
+(defun grep-tag-default ()
+  (save-excursion
+    (while (looking-at "\\sw\\|\\s_")
+      (forward-char 1))
+    (when (or (re-search-backward "\\sw\\|\\s_"
+                                 (save-excursion (beginning-of-line) (point))
+                                 t)
+             (re-search-forward "\\(\\sw\\|\\s_\\)+"
+                                (save-excursion (end-of-line) (point))
+                                t))
+      (goto-char (match-end 0))
+      (buffer-substring (point)
+                       (progn (forward-sexp -1)
+                              (while (looking-at "\\s'")
+                                (forward-char 1))
+                              (point))))))
+
+;;;###autoload
+(defun grep-find (command-args)
+  "Run grep via find, with user-specified args COMMAND-ARGS.
+Collect output in a buffer.
+While find runs asynchronously, you can use the \\[next-error] command
+to find the text that grep hits refer to.
+
+This command uses a special history list for its arguments, so you can
+easily repeat a find command."
+  (interactive
+   (progn
+     (unless grep-find-command
+       (grep-compute-defaults))
+     (list (read-from-minibuffer "Run find (like this): "
+                                grep-find-command nil nil
+                                'grep-find-history))))
+  (let ((null-device nil))             ; see grep
+    (grep command-args)))
+
+(defun grep-expand-command-macros (command &optional regexp files dir excl case-fold)
+  "Patch grep COMMAND replacing <D>, etc."
+  (setq command
+       (replace-regexp-in-string "<D>"
+                                 (or dir ".") command t t))
+  (setq command
+       (replace-regexp-in-string "<X>"
+                                 (or excl "") command t t))
+  (setq command
+       (replace-regexp-in-string "<F>"
+                                 (or files "") command t t))
+  (setq command
+       (replace-regexp-in-string "<C>"
+                                 (if case-fold "-i" "") command t t))
+  (setq command
+       (replace-regexp-in-string "<R>"
+                                 (or regexp "") command t t))
+  command)
+
+(defvar grep-tree-last-regexp "")
+(defvar grep-tree-last-files (car (car grep-tree-files-aliases)))
+
+;;;###autoload
+(defun grep-tree (regexp files dir &optional subdirs)
+  "Grep for REGEXP in FILES in directory tree rooted at DIR.
+Collect output in a buffer.
+Interactively, prompt separately for each search parameter.
+With prefix arg, reuse previous REGEXP.
+The search is limited to file names matching shell pattern FILES.
+FILES may use abbreviations defined in `grep-tree-files-aliases', e.g.
+entering `ch' is equivalent to `*.[ch]'.
+
+While find runs asynchronously, you can use the \\[next-error] command
+to find the text that grep hits refer to.
+
+This command uses a special history list for its arguments, so you can
+easily repeat a find command.
+
+When used non-interactively, optional arg SUBDIRS limits the search to
+those sub directories of DIR."
+  (interactive
+   (let* ((regexp
+          (if current-prefix-arg
+              grep-tree-last-regexp
+            (let* ((default (current-word))
+                   (spec (read-string
+                          (concat "Search for"
+                                  (if (and default (> (length default) 0))
+                                      (format " (default %s): " default) ": ")))))
+              (if (equal spec "") default spec))))
+         (files
+          (read-string (concat "Search for \"" regexp "\" in files (default "   grep-tree-last-files  "): ")))
+         (dir
+          (read-directory-name "Base directory: " nil default-directory t)))
+     (list regexp files dir)))
+  (unless grep-tree-command
+    (grep-compute-defaults))
+  (unless (and (stringp files) (> (length files) 0))
+    (setq files grep-tree-last-files))
+  (when files
+    (setq grep-tree-last-files files)
+    (let ((mf (assoc files grep-tree-files-aliases)))
+      (if mf
+         (setq files (cdr mf)))))
+  (let ((command-args (grep-expand-command-macros
+                      grep-tree-command
+                      (setq grep-tree-last-regexp regexp)
+                      (and files (concat "-name '" files "'"))
+                      (if subdirs
+                          (if (stringp subdirs)
+                              subdirs
+                            (mapconcat 'identity subdirs " "))
+                        nil)  ;; we change default-directory to dir
+                      (and grep-tree-ignore-CVS-directories "-path '*/CVS' -prune -o ")
+                      grep-tree-ignore-case))
+       (default-directory dir)
+       (null-device nil))              ; see grep
+    (grep command-args regexp)))
+
+
+(provide 'grep)
+
+;;; grep.el ends here