From: Stefan Monnier Date: Mon, 17 Jul 2006 21:07:23 +0000 (+0000) Subject: (sh-quoted-subshell): Don't match escaped `. Use `cond', push', and `dolist'. X-Git-Tag: emacs-pretest-22.0.90~1397 X-Git-Url: http://git.eshelyaron.com/gitweb/?a=commitdiff_plain;h=69c6ad8391353c627ef9ac52d7914db2b7d1de0a;p=emacs.git (sh-quoted-subshell): Don't match escaped `. Use `cond', push', and `dolist'. --- diff --git a/lisp/ChangeLog b/lisp/ChangeLog index 5dd8940fed7..f08169f3ac5 100644 --- a/lisp/ChangeLog +++ b/lisp/ChangeLog @@ -1,3 +1,8 @@ +2006-07-17 Stefan Monnier + + * progmodes/sh-script.el (sh-quoted-subshell): Don't match escaped `. + Use `cond', push', and `dolist'. + 2006-07-17 Richard Stallman * image-mode.el (tar-superior-buffer, archive-superior-buffer): diff --git a/lisp/progmodes/sh-script.el b/lisp/progmodes/sh-script.el index ef80d28c578..48b34a99311 100644 --- a/lisp/progmodes/sh-script.el +++ b/lisp/progmodes/sh-script.el @@ -982,7 +982,10 @@ Point is at the beginning of the next line." (defun sh-quoted-subshell (limit) "Search for a subshell embedded in a string. Find all the unescaped \" characters within said subshell, remembering that subshells can nest." - (if (re-search-forward "\"\\(?:.\\|\n\\)*?\\(\\$(\\|`\\)" limit t) + ;; FIXME: This can (and often does) match multiple lines, yet it makes no + ;; effort to handle multiline cases correctly, so it ends up being + ;; rather flakey. + (if (re-search-forward "\"\\(?:\\(?:.\\|\n\\)*?[^\\]\\(\\\\\\\\\\)*\\)?\\(\\$(\\|`\\)" limit t) ;; bingo we have a $( or a ` inside a "" (let ((char (char-after (point))) (continue t) @@ -1002,23 +1005,27 @@ Point is at the beginning of the next line." ;; ( [preceeded by $ ] => increase nesting ;; " [nesting <= 0 ] => terminate, we're done. ;; " [nesting > 0 ] => remember this, it's not a proper " - (if (eq ?\\ last) nil - (if (eq ?\` char) (setq nest (+ nest (if bq -1 1)) bq (not bq)) - (if (and (> nest 0) (eq ?\) char)) (setq nest (1- nest)) - (if (and (eq ?$ last) (eq ?\( char)) (setq nest (1+ nest)) - (if (and (> nest 0) (eq ?\( char)) (setq nest (1+ nest)) - (if (eq char ?\") - (if (>= 0 nest) (setq continue nil) - (setq seen (cons pos seen)) ) )))))) + ;; FIXME: don't count parens that appear within quotes. + (cond + ((eq ?\\ last) nil) + ((eq ?\` char) (setq nest (+ nest (if bq -1 1)) bq (not bq))) + ((and (> nest 0) (eq ?\) char)) (setq nest (1- nest))) + ((and (eq ?$ last) (eq ?\( char)) (setq nest (1+ nest))) + ((and (> nest 0) (eq ?\( char)) (setq nest (1+ nest))) + ((eq char ?\") + (if (>= 0 nest) (setq continue nil) (push pos seen)))) ;;(message "POS: %d [%d]" pos nest) (setq last char pos (1+ pos) char (char-after pos)) ) + ;; FIXME: why construct a costly match data to pass to + ;; sh-apply-quoted-subshell rather than apply the highlight + ;; directly here? -- Stef (when seen ;;(message "SEEN: %S" seen) (setq data (list (current-buffer))) - (mapc (lambda (P) - (setq data (cons P (cons (1+ P) data)) ) ) seen) + (dolist(P seen) + (setq data (cons P (cons (1+ P) data)))) (store-match-data data)) data) )) @@ -1554,7 +1561,7 @@ This adds rules for comments and assignments." (regexp-opt (sh-feature sh-builtins) t) "\\>") (2 font-lock-keyword-face nil t) - (6 font-lock-builtin-face)) + (4 font-lock-builtin-face)) ,@(sh-feature sh-font-lock-keywords-var-2))) (,(concat keywords "\\)\\>") 2 font-lock-keyword-face)