# HG changeset patch # User Jim Blandy # Date 743996360 0 # Node ID 3011390123edb17999061283bb0754c45cb8a646 # Parent 49e68bc65e26fdb1adcf9217f5a84d2398663eed * 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. diff -r 49e68bc65e26 -r 3011390123ed lisp/gud.el --- 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))