Enabling Line Number Display in coq-mode

WARNING

This article was translated from Chinese by ChatGPT and may contain some errors.

The display of line numbers (Line Number) should not be a problem. There is even an issue in the vscode repository asking how to conveniently turn off the line number display. For Vim users, a simple set :number can turn on the line number display.

Yes, Emacs is such a special editor. For occasional users like me, the display of line numbers is indeed a problem that troubles me.

What is the problem

I am using a distribution of Emacs, or a mature configuration file, Spacemacs. In the official Spacemacs documentation, to enable line number display, you only need to set a variable in .spacemacs:

1(setq-default dotspacemacs-line-numbers t)

The meaning of this sentence is to set dotspacemacs-line-numbers to t.

At the same time, to support the coq language, I also loaded the coq layer. But if you open a coq file, you may be surprised to find that the coq source code does not have line numbers:

According to normal intuition, setting to t should display line numbers in all cases. So, why are there no line numbers?

Temporary Solution

Solving this problem is actually easier than figuring out the reason.

After a little search, I found that spc t n 1 can turn on line numbers in the current buffer, which can indeed solve the problem. According to the Emacs wiki, this shortcut actually calls the line number command display-line-numbers-mode introduced after Emacs 26.1 version.

Emacs commands can be executed through m-x, or spc spc. In fact, a command is an Emacs Lisp function. According to the documentation of Programming Elisp, as long as there is (interactive) in the function definition of a function, it can be called as a command.

Clearly, Spacemacs also sets line numbers through this function. Although I dont know why Spacemacs didnt turn on line numbers for my .v files, we can turn them on ourselves.

The coq layer actually uses coq-mode. coq-mode is a Major Mode of Emacs, which provides support for the coq language and defines a bunch of shortcuts.

Emacs supports a mechanism called Hooks, which allows us to perform a series of operations when a mode is loaded. Since Spacemacs wont turn on line numbers for coq-mode, we manually execute display-line-numbers-mode through this mechanism.

1(defun dotspacemacs/user-config ()
2  "Configuration for user code:
3This function is called at the very end of Spacemacs startup, after layer
4configuration.
5Put your configuration code here, except for variables that should be set
6before packages are loaded."
7  (add-hook 'coq-mode-hook (lambda () (display-line-numbers-mode 1)))
8)

Here, dotspacemacs/user-config is part of the Spacemacs configuration file, where you can perform some custom operations.

This of course solves the problem. In fact, Spacemacs also mentions an option :enabled-for-modes in the documentation. Adding coq-mode to this option can directly solve the problem. However, I didnt discover this option until I figured out the source of the problem.

1(setq-default dotspacemacs-line-numbers
2  '(:enabled-for-modes coq-mode))

Why

The first thing I noticed was the comment for dotspacemacs-line-numbers:

1;; Control line numbers activation.
2;; If set to `t', `relative' or `visual' then line numbers are enabled in all
3;; `prog-mode' and `text-mode' derivatives. If set to `relative', line
4;; numbers are relative. If set to `visual', line numbers are also relative,
5;; but only visual lines are counted. For example, folded lines will not be
6;; counted and wrapped lines are counted as multiple lines.
7;; This variable can also be set to a property list for finer control:
8;; (default nil)

This comment mentions that if this variable is set to t, it will enable line numbers by default in prog-mode and text-mode. These two modes are known as Basic Major Modes. A natural guess is that coq-mode is not one of them.

Open a coq file, execute m-x describe-mode, and you will see:

1Coq mode defined in ‘coq-mode.el’:
2Major mode for Coq scripts.

Enter the source code of coq-mode.el to take a look, the mystery lies in line 182:

1(defun coq--parent-mode ()
2  (if coq-use-pg (proof-mode) (prog-mode)))
3
4;;;###autoload
5(define-derived-mode coq-mode coq--parent-mode "Coq"
6  "Major mode for Coq scripts.

This coq--parent-mode is obviously the parent mode of coq-mode (parent mode), Emascs mode also has an inheritance tree, a mode inherits another parent mode, the root of this tree is uncertain, there are several situations:

  • Itself
  • Fundamental Mode
  • Basic Major Mode

The value of coq-use-pg in my environment is t, and its parent mode is proof-mode. What is the parent mode of proof-mode?

A question on StackOverflow provides the relevant code:

1(defun derived-modes (mode)
2  "Return a list of the ancestor modes that MODE is derived from."
3  (let ((modes   ())
4        (parent  nil))
5    (while (setq parent (get mode 'derived-mode-parent))
6      (push parent modes)
7      (setq mode parent))
8    (setq modes  (nreverse modes))))

With this function, you can indeed get the inheritance path of some normal modes, such as the proof-mode we just mentioned:

1ELISP> (derived-modes 'proof-mode)
2(prog-mode)

Yes, the parent mode of proof-mode is prog-mode. If you still remember the document just now, prog-mode should also display line numbers when dotspacemacs-line-numbers is set to t.

So, what went wrong? Lets input 'coq-mode into this function:

1ELISP> (derived-modes 'coq-mode)
2(coq--parent-mode)

Its output is (coq--parent-mode), which is just the symbol of the parent mode, not the evaluated parent mode. It can be seen that the expression (get 'coq-mode 'derived-parent-mode) can only get the symbol in syntax, not its real parent mode. Checking the source code of define-derived-mode, you can find that what (get mode 'derived-mode-parent) gets is the parent mode set by this sentence:

1(put ',child 'derived-mode-parent ',parent)

So, is this how Spacemacs gets the parent mode? Spacemacs checks whether line number display is enabled in the current mode in the spacemacs//linum-enabled-for-current-major-mode function in spacemacs/layers/+spacemacs/spacemacs-defaults/funcs.el:

1(defun spacemacs//linum-enabled-for-current-major-mode ()
2  "Return non-nil if line number is enabled for current major-mode."
3  (let* ((disabled-for-modes
4          (spacemacs/mplist-get-values dotspacemacs-line-numbers
5                                       :disabled-for-modes))
6         (user-enabled-for-modes
7          (spacemacs/mplist-get-values dotspacemacs-line-numbers
8                                       :enabled-for-modes))
9         ;; default `enabled-for-modes' to '(prog-mode text-mode), because it is
10         ;; a more sensible default than enabling in all buffers - including
11         ;; Magit buffers, terminal buffers, etc. But don't include prog-mode or
12         ;; text-mode if they're explicitly disabled by user
13         (enabled-for-modes (or user-enabled-for-modes
14                                (seq-difference '(prog-mode text-mode)
15                                                disabled-for-modes
16                                                #'eq)))
17         (enabled-for-parent (or (and (equal enabled-for-modes '(all)) 'all)
18                                 (apply #'derived-mode-p enabled-for-modes)))
19         (disabled-for-parent (apply #'derived-mode-p disabled-for-modes)))
20    (or
21     ;; special case 'all: enable for any mode that isn't specifically disabled
22     (and (eq enabled-for-parent 'all) (not disabled-for-parent))
23     ;; current mode or a parent is in :enabled-for-modes, and there isn't a
24     ;; more specific parent (or the mode itself) in :disabled-for-modes
25     (and enabled-for-parent
26          (or (not disabled-for-parent)
27              ;; handles the case where current major-mode has a parent both in
28              ;; :enabled-for-modes and in :disabled-for-modes. Return non-nil
29              ;; if enabled-for-parent is the more specific parent (IOW derives
30              ;; from disabled-for-parent)
31              (spacemacs/derived-mode-p enabled-for-parent disabled-for-parent)))
32     ;; current mode (or parent) not explicitly disabled
33     (and (null user-enabled-for-modes)
34          enabled-for-parent            ; mode is one of default allowed modes
35          disabled-for-modes
36          (not disabled-for-parent)))))

As you can see, it asks the current buffers major mode if it is a child mode of the preset enabled-for-modes through the derived-mode-p function. dervied-mode-p is a function provided by Emacs, and the documentation claims that if the current buffers main mode is a child mode of modes, then (derived-mode-p modes) will return a non-nil value.

Upon further inspection of the source code of derived-mode-p, we finally found the root of the problem.

1(defun provided-mode-derived-p (mode &rest modes)
2  "Non-nil if MODE is derived from one of MODES or their aliases.
3Uses the `derived-mode-parent' property of the symbol to trace backwards.
4If you just want to check `major-mode', use `derived-mode-p'."
5  (while
6      (and
7       (not (memq mode modes))
8       (let* ((parent (get mode 'derived-mode-parent))
9              (parentfn (symbol-function parent)))
10         (setq mode (if (and parentfn (symbolp parentfn)) parentfn parent)))))
11  mode)
12
13(defun derived-mode-p (&rest modes)
14  "Non-nil if the current major mode is derived from one of MODES.
15Uses the `derived-mode-parent' property of the symbol to trace backwards."
16  (apply #'provided-mode-derived-p major-mode modes))

dervied-mode-p also gets the parent mode of a certain mode through (get mode 'derived-mode-parent), but for coq-mode, it will get coq--parent-mode, and the inheritance chain breaks here.

The final definition of mode in Emacs is indeed a function. However, if 'parent cannot evaluate the correct parent mode, Emacs preset function for mode will not get the expected result. Obviously, coq-mode should not define coq--parent-mode in this way. So, how should coq--parent-mode be defined?

Final Solution

The global variables and global functions of Emacs Lisp are different, which is very different from Scheme. If you directly define coq--parent-mode as a variable, it will make coq-mode unable to load.

1(defvar coq--parent-mode
2  (if coq-use-pg 'proof-mode 'prog-mode))

Fortunately, Emacs Lisp provides a function called defalias, which can define coq--parent-mode as an alias.

1(defalias 'coq--parent-mode
2  (if coq-use-pg 'proof-mode 'prog-mode))

The above modification is enough to make Spacemacs automatically detect that coq-mode is derived from prog-mode, thus turning on line number display. I have submitted a PR and merged it into the main branch.