]> git.eshelyaron.com Git - emacs.git/commitdiff
CC Mode: On removal of "typedef", remove pertinent types from c-found-types
authorAlan Mackenzie <acm@muc.de>
Tue, 17 Jan 2023 18:15:45 +0000 (18:15 +0000)
committerAlan Mackenzie <acm@muc.de>
Tue, 17 Jan 2023 18:15:45 +0000 (18:15 +0000)
For this purpose, record the type names declared by typedef in a text
property, c-typedef, on the typedef.  On any change to that "typedef" or a
type, remove the old identifier(s) from c-found-types.

This should fix bug #59671.

* lisp/progmodes/cc-defs.el (c-search-forward-non-nil-char-property): New
macro.

* lisp/progmodes/cc-engine.el (c-forward-decl-or-cast-1): Move the scope of
identifier-start from the "inner" let form to the outer one.  Amend the
return value such that the middle element of the second element is now the
position of the "typedef", not merely non-nil.

* lisp/progmodes/cc-fonts.el (c-font-lock-declarators): Disregard the LIMIT
parameter when fontifying the declarators of a typedef construct.  Also in
this case, set the c-typedef text property on the "typedef" to the list of
declared types.  Amend this list when these declared types change.
(c-font-lock-single-decl): Massage the `types' argument given to
c-font-lock-declarators.
(c-font-lock-cut-off-declarators): Amend to work when the starting point of
the fontification is inside a brace block.

* lisp/progmodes/cc-mode.el (c-before-change-de-typedef)
(c-after-change-de-typedef): New functions.
(c-update-new-id): Replace the erroneous c-end-of-current-token with a clause
containing c-forward-token-2.
(c-before-change): Call c-before-change-de-typedef.
(c-after-change): Call c-after-change-de-typedef.

lisp/progmodes/cc-defs.el
lisp/progmodes/cc-engine.el
lisp/progmodes/cc-fonts.el
lisp/progmodes/cc-mode.el

index 493035d38b4d70cc40629cfc95329b35df9f49a9..bdbc03e7c94e3911a495a55e77bed847251d3919 100644 (file)
@@ -1361,6 +1361,28 @@ nil; point is then left undefined."
        (search-forward-regexp "\\(\n\\|.\\)")  ; to set the match-data.
        (point))))
 
+(defmacro c-search-forward-non-nil-char-property (property &optional limit)
+  "Search forward for a text-property PROPERTY value non-nil.
+LIMIT bounds the search.
+
+Leave point just after the character.  The match data remain
+unchanged.  Return the value of PROPERTY.  If a non-nil value
+isn't found, return nil; point is then left undefined."
+  (declare (debug t))
+  `(let* ((-limit- (or ,limit (point-max)))
+         (value (c-get-char-property (point) ,property)))
+     (cond
+      ((>= (point) -limit-)
+       nil)
+      (value
+       (forward-char)
+       value)
+      (t (let ((place (c-next-single-property-change
+                      (point) ,property nil -limit-)))
+          (when place
+            (goto-char (1+ place))
+            (c-get-char-property place ,property)))))))
+
 (defmacro c-search-backward-char-property (property value &optional limit)
   "Search backward for a text-property PROPERTY having value VALUE.
 LIMIT bounds the search.  The comparison is done with `equal'.
index 45d90ea243141a5ce045ec3d9f267ac49efdf3d4..3fa407dd33854af99bf6a581dd021128f5a80d77 100644 (file)
 ;;       Put on the brace which introduces a brace list and on the commas
 ;;       which separate the elements within it.
 ;;
+;; 'c-typedef This property is applied to the first character of a
+;;   "typedef" keyword.  It's value is a list of the identifiers that
+;;   the "typedef" declares as types.
+;;
 ;; 'c-<>-c-types-set
 ;;   This property is set on an opening angle bracket, and indicates that
 ;;   any "," separators within the template/generic expression have been
@@ -10024,10 +10028,10 @@ This function might do hidden buffer changes."
   ;; an identifier instead.
   (declare (debug nil))
   `(progn
+     (setq identifier-start type-start)
      ,(unless short
        ;; These identifiers are bound only in the inner let.
        '(setq identifier-type at-type
-              identifier-start type-start
               got-parens nil
               got-identifier t
               got-suffix t
@@ -10102,10 +10106,11 @@ This function might do hidden buffer changes."
   ;;   The second element of the return value is non-nil when something
   ;;   indicating the identifier is a type occurs in the declaration.
   ;;   Specifically it is nil, or a three element list (A B C) where C is t
-  ;;   when context is '<> and the "identifier" is a found type, B is t when a
-  ;;   `c-typedef-kwds' ("typedef") is present, and A is t when some other
-  ;;   `c-typedef-decl-kwds' (e.g. class, struct, enum) specifier is present.
-  ;;   I.e., (some of) the declared identifier(s) are types.
+  ;;   when context is '<> and the "identifier" is a found type, B is the
+  ;;   position of the `c-typedef-kwds' keyword ("typedef") when such is
+  ;;   present, and A is t when some other `c-typedef-decl-kwds' (e.g. class,
+  ;;   struct, enum) specifier is present.  I.e., (some of) the declared
+  ;;   identifier(s) are types.
   ;;
   ;;   The third element of the return value is non-nil when the declaration
   ;;   parsed might be an expression.  The fourth element is the position of
@@ -10173,6 +10178,9 @@ This function might do hidden buffer changes."
        ;; `c-decl-hangon-kwds' and their associated clauses that
        ;; occurs after the type.
        id-start
+       ;; The earlier value of `type-start' if we've shifted the type
+       ;; backwards.
+       identifier-start
        ;; These store `at-type', `type-start' and `id-start' of the
        ;; identifier before the one in those variables.  The previous
        ;; identifier might turn out to be the real type in a
@@ -10183,7 +10191,8 @@ This function might do hidden buffer changes."
        ;; Set if we've found a specifier (apart from "typedef") that makes
        ;; the defined identifier(s) types.
        at-type-decl
-       ;; Set if we've a "typedef" keyword.
+       ;; If we've a "typedef" keyword (?or similar), the buffer position of
+       ;; its first character.
        at-typedef
        ;; Set if `context' is '<> and the identifier is definitely a type, or
        ;; has already been recorded as a found type.
@@ -10266,7 +10275,7 @@ This function might do hidden buffer changes."
                 (looking-at "@[A-Za-z0-9]+")))
            (save-match-data
              (if (looking-at c-typedef-key)
-                 (setq at-typedef t)))
+                 (setq at-typedef (point))))
            (setq kwd-sym (c-keyword-sym (match-string 1)))
            (save-excursion
              (c-forward-keyword-clause 1)
@@ -10486,9 +10495,9 @@ This function might do hidden buffer changes."
          ;; True if we've parsed the type decl to a token that is
          ;; known to end declarations in this context.
          at-decl-end
-         ;; The earlier values of `at-type' and `type-start' if we've
-         ;; shifted the type backwards.
-         identifier-type identifier-start
+         ;; The earlier value of `at-type' if we've shifted the type
+         ;; backwards.
+         identifier-type
          ;; If `c-parse-and-markup-<>-arglists' is set we need to
          ;; turn it off during the name skipping below to avoid
          ;; getting `c-type' properties that might be bogus.  That
@@ -10530,6 +10539,10 @@ This function might do hidden buffer changes."
                              (progn (setq got-identifier nil) t)
                            ;; It turned out to be the real identifier,
                            ;; so stop.
+                           (save-excursion
+                             (c-backward-syntactic-ws)
+                             (c-simple-skip-symbol-backward)
+                             (setq identifier-start (point)))
                            nil))
                      t))
 
@@ -10555,6 +10568,10 @@ This function might do hidden buffer changes."
          (and (looking-at c-identifier-start)
               (setq pos (point))
               (setq got-identifier (c-forward-name))
+              (save-excursion
+                (c-backward-syntactic-ws)
+                (c-simple-skip-symbol-backward)
+                (setq identifier-start (point)))
               (setq name-start pos))
          (when (looking-at "[0-9]")
            (setq got-number t)) ; We probably have an arithmetic expression.
@@ -10573,7 +10590,8 @@ This function might do hidden buffer changes."
               (setq at-type nil
                     name-start type-start
                     id-start type-start
-                    got-identifier t)))
+                    got-identifier t)
+              (setq identifier-start type-start)))
 
       ;; Skip over type decl suffix operators and trailing noise macros.
       (while
index 4dcc3e0ade9a710afdacfccf6c44c50ab7d77231..405ffb9f1f3076b7276ae7ff67b2e909b266d6e8 100644 (file)
@@ -1109,10 +1109,12 @@ casts and declarations are fontified.  Used on level 2 and higher."
   ;; additionally, mark the commas with c-type property 'c-decl-id-start or
   ;; 'c-decl-type-start (according to TYPES).  Stop at LIMIT.
   ;;
-  ;; If TYPES is t, fontify all identifiers as types, if it is nil fontify as
-  ;; either variables or functions, otherwise TYPES is a face to use.  If
-  ;; NOT-TOP is non-nil, we are not at the top-level ("top-level" includes
-  ;; being directly inside a class or namespace, etc.).
+  ;; If TYPES is t, fontify all identifiers as types; if it is a number, a
+  ;; buffer position, additionally set the `c-deftype' text property on the
+  ;; keyword at that position; if it is nil fontify as either variables or
+  ;; functions, otherwise TYPES is a face to use.  If NOT-TOP is non-nil, we
+  ;; are not at the top-level ("top-level" includes being directly inside a
+  ;; class or namespace, etc.).
   ;;
   ;; TEMPLATE-CLASS is non-nil when the declaration is in template delimiters
   ;; and was introduced by, e.g. "typename" or "class", such that if there is
@@ -1129,17 +1131,28 @@ casts and declarations are fontified.  Used on level 2 and higher."
   ;;(message "c-font-lock-declarators from %s to %s" (point) limit)
   (c-fontify-types-and-refs
       ()
+    ;; If we're altering the declarators in a typedef, we need to scan ALL of
+    ;; them because of the way we check for changes.
+    (let ((c-do-decl-limit (if (numberp types) (point-max) limit))
+         decl-ids)
     (c-do-declarators
-     limit list not-top
-     (cond ((eq types t) 'c-decl-type-start)
+     c-do-decl-limit
+     list not-top
+     (cond ((or (numberp types)
+               (eq types t))
+           'c-decl-type-start)
           ((null types) 'c-decl-id-start))
      (lambda (id-start id-end end-pos _not-top is-function init-char)
-       (if (eq types t)
+       (if (or (numberp types)
+              (eq types t))
           (when id-start
             ;; Register and fontify the identifier as a type.
             (let ((c-promote-possible-types t))
               (goto-char id-start)
-              (c-forward-type)))
+              (c-forward-type))
+            (when (numberp types)
+              (push (buffer-substring-no-properties id-start id-end)
+                    decl-ids)))
         (when id-start
           (goto-char id-start)
           (when c-opt-identifier-prefix-key
@@ -1147,7 +1160,7 @@ casts and declarations are fontified.  Used on level 2 and higher."
                          (eq (match-end 1) id-end))
               (while (and (< (point) id-end)
                           (re-search-forward c-opt-identifier-prefix-key id-end t))
-                (c-forward-syntactic-ws limit))))
+                (c-forward-syntactic-ws c-do-decl-limit))))
           ;; Only apply the face when the text doesn't have one yet.
           ;; Exception: The "" in C++'s operator"" will already wrongly have
           ;; string face.
@@ -1164,7 +1177,7 @@ casts and declarations are fontified.  Used on level 2 and higher."
                  (equal (buffer-substring-no-properties id-start id-end)
                         "\"\""))
             (goto-char id-end)
-            (c-forward-syntactic-ws limit)
+            (c-forward-syntactic-ws c-do-decl-limit)
             (when (c-on-identifier)
               (c-put-font-lock-face
                (point)
@@ -1174,10 +1187,21 @@ casts and declarations are fontified.  Used on level 2 and higher."
            (eq init-char ?=)           ; C++ "<class X = Y>"?
            (progn
              (goto-char end-pos)
-             (c-forward-token-2 1 nil limit) ; Over "="
+             (c-forward-token-2 1 nil c-do-decl-limit) ; Over "="
              (let ((c-promote-possible-types t))
                (c-forward-type t)))))
      accept-anon)                      ; Last argument to c-do-declarators.
+    ;; If we've changed types declared by a "typedef", update the `c-typedef'
+    ;; text property.
+    (when (numberp types)
+      (let* ((old-decl-ids (c-get-char-property types 'c-typedef))
+            (old-types (c--set-difference old-decl-ids decl-ids :test #'equal))
+            (new-types (c--set-difference decl-ids old-decl-ids :test #'equal)))
+       (dolist (type old-types)
+         (c-unfind-type type))
+       ;; The new types have already been added to `c-found-types', as needed.
+       (when (or old-types new-types)
+         (c-put-char-property types 'c-typedef decl-ids)))))
     nil))
 
 (defun c-get-fontification-context (match-pos not-front-decl &optional toplev)
@@ -1433,7 +1457,10 @@ casts and declarations are fontified.  Used on level 2 and higher."
       (c-font-lock-declarators
        (min limit (point-max))
        decl-list
-       (not (null (cadr decl-or-cast)))
+       (cond ((null (cadr decl-or-cast))
+             nil)
+            ((cadr (cadr decl-or-cast)))
+            (t t))
        (not toplev)
        template-class
        (memq context '(decl <>))))
@@ -1749,12 +1776,21 @@ casts and declarations are fontified.  Used on level 2 and higher."
                                        ; speeds up lisp.h tremendously.
       (save-excursion
        (when (not (c-back-over-member-initializers decl-search-lim))
+         (setq paren-state (c-parse-state))
          (unless (or (eobp)
                      (looking-at "\\s(\\|\\s)"))
            (forward-char))
          (c-syntactic-skip-backward "^;{}" decl-search-lim t)
-         (when (eq (char-before) ?})
-           (c-go-list-backward)        ; brace block of struct, etc.?
+         ;; Do we have the brace block of a struct, etc.?
+         (when (cond
+                ((and (consp (car paren-state))
+                      (eq (char-before) ?}))
+                 (goto-char (caar paren-state))
+                 t)
+                ((and (numberp (car paren-state))
+                      (eq (char-after (car paren-state)) ?{))
+                 (goto-char (car paren-state))
+                 t))
            (c-syntactic-skip-backward "^;{}" decl-search-lim t))
          (when (or (bobp)
                    (memq (char-before) '(?\; ?{ ?})))
index b04ed7584c41abd1a00646a7f13db3640df225c3..330202bb5f994b412851cd6455ada759d0a7883a 100644 (file)
@@ -2077,6 +2077,37 @@ with // and /*, not more generic line and block comments."
                 (not (eobp))))
        (forward-char))))))
 
+(defun c-before-change-de-typedef (beg end)
+  ;; For each "typedef" starting in (BEG END), remove the defined types from
+  ;; c-found-types
+  (let (prop)
+    (save-excursion
+      (goto-char beg)
+      (while (and (< (point) end)
+                 (setq prop (c-search-forward-non-nil-char-property
+                             'c-typedef)))
+       (dolist (type prop)
+         (c-unfind-type type))))))
+
+(defun c-after-change-de-typedef (beg end _old-len)
+  ;; For each former "typedef" in (BEG END), remove the defined types from
+  ;; those which are no longer typedefs.
+  (let (prop)
+    (save-excursion
+      (goto-char beg)
+      (c-backward-token-2
+       1 nil (- (point) 20))
+      (while (and (< (point) end)
+                 (setq prop (c-search-forward-non-nil-char-property
+                             'c-typedef end)))
+       (backward-char)
+       (when (or (not (looking-at c-typedef-key))
+                 (<= (match-end 1) beg))
+         (dolist (type prop)
+           (c-unfind-type type))
+         (c-clear-char-property (point) 'c-typedef))
+       (forward-char)))))
+
 (defun c-update-new-id (end)
   ;; Note the bounds of any identifier that END is in or just after, in
   ;; `c-new-id-start' and `c-new-id-end'.  Otherwise set these variables to
@@ -2086,7 +2117,9 @@ with // and /*, not more generic line and block comments."
     (let ((id-beg (c-on-identifier)))
       (setq c-new-id-start id-beg
            c-new-id-end (and id-beg
-                             (progn (c-end-of-current-token) (point)))
+                             (progn (goto-char id-beg)
+                                    (c-forward-token-2)
+                                    (point)))
            c-new-id-is-type nil))))
 
 (defun c-post-command ()
@@ -2215,6 +2248,10 @@ with // and /*, not more generic line and block comments."
                                                               term-pos)
                               (buffer-substring-no-properties beg end)))))))
 
+          ;; If we're about to delete "typedef"s, clear the identifiers from
+          ;; `c-found-types'.
+          (c-before-change-de-typedef beg end)
+
           (if c-get-state-before-change-functions
               (mapc (lambda (fn)
                       (funcall fn beg end))
@@ -2306,6 +2343,7 @@ with // and /*, not more generic line and block comments."
           (c-update-new-id end)
           (c-trim-found-types beg end old-len) ; maybe we don't
                                        ; need all of these.
+          (c-after-change-de-typedef beg end old-len)
           (c-invalidate-sws-region-after beg end old-len)
           ;; (c-invalidate-state-cache beg) ; moved to
           ;; `c-before-change'.