changeset 4346:3011390123ed

* gud.el (gud-gdb-marker-filter): Do not assume that the position markers from GDB will always be received in one chunk of input; gud-gdb-marker-filter may be called several times, each time providing a little more of the position marker. (gud-gdb-marker-acc): New variable. (gud-gdb-marker-filter): If we have received what could be the beginning of a position marker, hold that text in gud-gdb-marker-acc for the next time we get called, until we have enough information to decide for sure. * gud.el (gud-gdb-marker-filter): Only recognize GDB position markers if they occur at the beginning of the line. They always do, and this reduces the likelihood that the above change will hold back output that isn't really a position marker.
author Jim Blandy <jimb@redhat.com>
date Fri, 30 Jul 1993 01:39:20 +0000
parents 49e68bc65e26
children d6b289b1a6dc
files lisp/gud.el
diffstat 1 files changed, 50 insertions(+), 14 deletions(-) [+]
line wrap: on
line diff
--- a/lisp/gud.el	Thu Jul 29 23:21:30 1993 +0000
+++ b/lisp/gud.el	Fri Jul 30 01:39:20 1993 +0000
@@ -156,21 +156,57 @@
 (defun gud-gdb-massage-args (file args)
   (cons "-fullname" (cons file args)))
 
+;; There's no guarantee that Emacs will hand the filter the entire
+;; marker at once; it could be broken up across several strings.  We
+;; might even receive a big chunk with several markers in it.  If we
+;; receive a chunk of text which looks like it might contain the
+;; beginning of a marker, we save it here between calls to the
+;; filter.
+(defvar gud-gdb-marker-acc "")
+
 (defun gud-gdb-marker-filter (string)
-  (if (string-match  "\032\032\\([^:\n]*\\):\\([0-9]*\\):.*\n" string)
-      (progn
-	(setq gud-last-frame
-	      (cons
-	       (substring string (match-beginning 1) (match-end 1))
-	       (string-to-int
-		(substring string (match-beginning 2) (match-end 2)))))
-	;; this computation means the ^Z^Z-initiated marker in the
-	;; input string is never emitted.
-	(concat
-	 (substring string 0 (match-beginning 0))
-	 (substring string (match-end 0))
-	 ))
-    string))
+  (setq gud-gdb-marker-acc (concat gud-gdb-marker-acc string))
+  (let ((output ""))
+
+    ;; Process all the complete markers in this chunk.
+    (while (string-match "^\032\032\\([^:\n]*\\):\\([0-9]*\\):.*\n"
+			 gud-gdb-marker-acc)
+      (setq
+
+       ;; Extract the frame position from the marker.
+       gud-last-frame
+       (cons (substring gud-gdb-marker-acc (match-beginning 1) (match-end 1))
+	     (string-to-int (substring gud-gdb-marker-acc
+				       (match-beginning 2)
+				       (match-end 2))))
+
+       ;; Append any text before the marker to the output we're going
+       ;; to return - we don't include the marker in this text.
+       output (concat output
+		      (substring gud-gdb-marker-acc 0 (match-beginning 0)))
+
+       ;; Set the accumulator to the remaining text.
+       gud-gdb-marker-acc (substring gud-gdb-marker-acc (match-end 0))))
+
+    ;; Does the remaining text look like it might end with the
+    ;; beginning of another marker?  If it does, then keep it in
+    ;; gud-gdb-marker-acc until we receive the rest of it.  Since we
+    ;; know the full marker regexp above failed, it's pretty simple to
+    ;; test for marker starts.
+    (if (string-match "^\032.*\\'" gud-gdb-marker-acc)
+	(progn
+	  ;; Everything before the potential marker start can be output.
+	  (setq output (concat output (substring gud-gdb-marker-acc
+						 0 (match-beginning 0))))
+
+	  ;; Everything after, we save, to combine with later input.
+	  (setq gud-gdb-marker-acc
+		(substring gud-gdb-marker-acc (match-beginning 0))))
+
+      (setq output gud-gdb-marker-acc
+	    gud-gdb-marker-acc ""))
+
+    output))
 
 (defun gud-gdb-find-file (f)
   (find-file-noselect f))