cl-generic: Fix advertised-calling-convention declarations

* lisp/emacs-lisp/cl-generic.el (cl-generic-define-method):
Preserve the `advertised-calling-convention`, if any (bug#58563).

* lisp/subr.el (declare): Warn when we hit this.

* lisp/emacs-lisp/byte-run.el (get-advertised-calling-convention): New fun.
* lisp/progmodes/elisp-mode.el (elisp-get-fnsym-args-string):
* lisp/help-fns.el (help-fns--signature):
* lisp/emacs-lisp/bytecomp.el (byte-compile-fdefinition): Use it.

* test/lisp/emacs-lisp/cl-generic-tests.el (cl-generic-tests--acc): New fun.
(cl-generic-tests--advertised-calling-convention-bug58563): New test.
This commit is contained in:
Stefan Monnier 2022-10-16 12:01:47 -04:00
parent 0e5fc2345d
commit 13d6e8fa54
7 changed files with 51 additions and 12 deletions

View file

@ -481,6 +481,11 @@ convention was modified."
(puthash (indirect-function function) signature (puthash (indirect-function function) signature
advertised-signature-table)) advertised-signature-table))
(defun get-advertised-calling-convention (function)
"Get the advertised SIGNATURE of FUNCTION.
Return t if there isn't any."
(gethash function advertised-signature-table t))
(defun make-obsolete (obsolete-name current-name when) (defun make-obsolete (obsolete-name current-name when)
"Make the byte-compiler warn that function OBSOLETE-NAME is obsolete. "Make the byte-compiler warn that function OBSOLETE-NAME is obsolete.
OBSOLETE-NAME should be a function name or macro name (a symbol). OBSOLETE-NAME should be a function name or macro name (a symbol).

View file

@ -1405,11 +1405,11 @@ when printing the error message."
(and (not macro-p) (and (not macro-p)
(compiled-function-p (symbol-function fn))))) (compiled-function-p (symbol-function fn)))))
(setq fn (symbol-function fn))) (setq fn (symbol-function fn)))
(let ((advertised (gethash (if (and (symbolp fn) (fboundp fn)) (let ((advertised (get-advertised-calling-convention
;; Could be a subr. (if (and (symbolp fn) (fboundp fn))
(symbol-function fn) ;; Could be a subr.
fn) (symbol-function fn)
advertised-signature-table t))) fn))))
(cond (cond
((listp advertised) ((listp advertised)
(if macro-p (if macro-p

View file

@ -650,13 +650,17 @@ The set of acceptable TYPEs (also called \"specializers\") is defined
(cl--generic-name generic) (cl--generic-name generic)
qualifiers specializers)) qualifiers specializers))
current-load-list :test #'equal) current-load-list :test #'equal)
(let (;; Prevent `defalias' from recording this as the definition site of (let ((old-adv-cc (get-advertised-calling-convention
(symbol-function sym)))
;; Prevent `defalias' from recording this as the definition site of
;; the generic function. ;; the generic function.
current-load-list current-load-list
;; BEWARE! Don't purify this function definition, since that leads ;; BEWARE! Don't purify this function definition, since that leads
;; to memory corruption if the hash-tables it holds are modified ;; to memory corruption if the hash-tables it holds are modified
;; (the GC doesn't trace those pointers). ;; (the GC doesn't trace those pointers).
(purify-flag nil)) (purify-flag nil))
(when (listp old-adv-cc)
(set-advertised-calling-convention gfun old-adv-cc nil))
;; But do use `defalias', so that it interacts properly with nadvice, ;; But do use `defalias', so that it interacts properly with nadvice,
;; e.g. for tracing/debug-on-entry. ;; e.g. for tracing/debug-on-entry.
(defalias sym gfun))))) (defalias sym gfun)))))

View file

@ -669,7 +669,7 @@ the C sources, too."
"Insert usage at point and return docstring. With highlighting." "Insert usage at point and return docstring. With highlighting."
(if (keymapp function) (if (keymapp function)
doc ; If definition is a keymap, skip arglist note. doc ; If definition is a keymap, skip arglist note.
(let* ((advertised (gethash real-def advertised-signature-table t)) (let* ((advertised (get-advertised-calling-convention real-def))
(arglist (if (listp advertised) (arglist (if (listp advertised)
advertised (help-function-arglist real-def))) advertised (help-function-arglist real-def)))
(usage (help-split-fundoc doc function))) (usage (help-split-fundoc doc function)))

View file

@ -1826,8 +1826,8 @@ or elsewhere, return a 1-line docstring."
(eq 'function (aref elisp--eldoc-last-data 2))) (eq 'function (aref elisp--eldoc-last-data 2)))
(aref elisp--eldoc-last-data 1)) (aref elisp--eldoc-last-data 1))
(t (t
(let* ((advertised (gethash (indirect-function sym) (let* ((advertised (get-advertised-calling-convention
advertised-signature-table t)) (indirect-function sym)))
doc doc
(args (args
(cond (cond

View file

@ -344,7 +344,7 @@ in compilation warnings about unused variables.
;; FIXME: This let often leads to "unused var" warnings. ;; FIXME: This let often leads to "unused var" warnings.
`((let ((,var ,counter)) ,@(cddr spec))))))) `((let ((,var ,counter)) ,@(cddr spec)))))))
(defmacro declare (&rest _specs) (defmacro declare (&rest specs)
"Do not evaluate any arguments, and return nil. "Do not evaluate any arguments, and return nil.
If a `declare' form appears as the first form in the body of a If a `declare' form appears as the first form in the body of a
`defun' or `defmacro' form, SPECS specifies various additional `defun' or `defmacro' form, SPECS specifies various additional
@ -355,8 +355,16 @@ The possible values of SPECS are specified by
`defun-declarations-alist' and `macro-declarations-alist'. `defun-declarations-alist' and `macro-declarations-alist'.
For more information, see info node `(elisp)Declare Form'." For more information, see info node `(elisp)Declare Form'."
;; FIXME: edebug spec should pay attention to defun-declarations-alist. ;; `declare' is handled directly by `defun/defmacro' rather than here.
nil) ;; If we get here, it's because there's a `declare' somewhere not attached
;; to a `defun/defmacro', i.e. a `declare' which doesn't do what it's
;; intended to do.
(let ((form `(declare . ,specs))) ;; FIXME: WIBNI we had &whole?
(macroexp-warn-and-return
(format-message "Stray `declare' form: %S" form)
;; Make a "unique" harmless form to circumvent
;; the cache in `macroexp-warn-and-return'.
`(progn ',form nil) nil 'compile-only)))
(defmacro ignore-errors (&rest body) (defmacro ignore-errors (&rest body)
"Execute BODY; if an error occurs, return nil. "Execute BODY; if an error occurs, return nil.

View file

@ -297,5 +297,27 @@ Edebug symbols (Bug#42672)."
(intern "cl-defgeneric/edebug/method/2 (number)") (intern "cl-defgeneric/edebug/method/2 (number)")
'cl-defgeneric/edebug/method/2)))))) 'cl-defgeneric/edebug/method/2))))))
(cl-defgeneric cl-generic-tests--acc (x &optional y)
(declare (advertised-calling-convention (x) "671.2")))
(cl-defmethod cl-generic-tests--acc ((x float)) (+ x 5.0))
(ert-deftest cl-generic-tests--advertised-calling-convention-bug58563 ()
(should (equal (get-advertised-calling-convention
(indirect-function 'cl-generic-tests--acc))
'(x)))
(should
(condition-case err
(let ((lexical-binding t)
(byte-compile-debug t)
(byte-compile-error-on-warn t))
(byte-compile '(cl-defmethod cl-generic-tests--acc ((x list))
(declare (advertised-calling-convention (y) "1.1"))
(cons x '(5 5 5 5 5))))
nil)
(error
(and (eq 'error (car err))
(string-match "Stray.*declare" (cadr err)))))))
(provide 'cl-generic-tests) (provide 'cl-generic-tests)
;;; cl-generic-tests.el ends here ;;; cl-generic-tests.el ends here