523 lines
19 KiB
523 lines
19 KiB
;;; cmake-mode.el --- major-mode for editing CMake sources -*- lexical-binding: t; -*-
|
|
|
|
;; Package-Requires: ((emacs "24.1"))
|
|
|
|
; Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
; file Copyright.txt or https://cmake.org/licensing for details.
|
|
|
|
;------------------------------------------------------------------------------
|
|
|
|
;;; Commentary:
|
|
|
|
;; Provides syntax highlighting and indentation for CMakeLists.txt and
|
|
;; *.cmake source files.
|
|
;;
|
|
;; Add this code to your .emacs file to use the mode:
|
|
;;
|
|
;; (setq load-path (cons (expand-file-name "/dir/with/cmake-mode") load-path))
|
|
;; (require 'cmake-mode)
|
|
|
|
;------------------------------------------------------------------------------
|
|
|
|
;;; Code:
|
|
;;
|
|
;; cmake executable variable used to run cmake --help-command
|
|
;; on commands in cmake-mode
|
|
;;
|
|
;; cmake-command-help Written by James Bigler
|
|
;;
|
|
|
|
(require 'rst)
|
|
(require 'rx)
|
|
|
|
(defcustom cmake-mode-cmake-executable "cmake"
|
|
"*The name of the cmake executable.
|
|
|
|
This can be either absolute or looked up in $PATH. You can also
|
|
set the path with these commands:
|
|
(setenv \"PATH\" (concat (getenv \"PATH\") \";C:\\\\Program Files\\\\CMake 2.8\\\\bin\"))
|
|
(setenv \"PATH\" (concat (getenv \"PATH\") \":/usr/local/cmake/bin\"))"
|
|
:type 'file
|
|
:group 'cmake)
|
|
|
|
;; Keywords
|
|
(defconst cmake-keywords-block-open '("BLOCK" "IF" "MACRO" "FOREACH" "ELSE" "ELSEIF" "WHILE" "FUNCTION"))
|
|
(defconst cmake-keywords-block-close '("ENDBLOCK" "ENDIF" "ENDFOREACH" "ENDMACRO" "ELSE" "ELSEIF" "ENDWHILE" "ENDFUNCTION"))
|
|
(defconst cmake-keywords
|
|
(let ((kwds (append cmake-keywords-block-open cmake-keywords-block-close nil)))
|
|
(delete-dups kwds)))
|
|
|
|
;; Regular expressions used by line indentation function.
|
|
;;
|
|
(defconst cmake-regex-blank "^[ \t]*$")
|
|
(defconst cmake-regex-comment "#.*")
|
|
(defconst cmake-regex-paren-left "(")
|
|
(defconst cmake-regex-paren-right ")")
|
|
(defconst cmake-regex-closing-parens-line (concat "^[[:space:]]*\\("
|
|
cmake-regex-paren-right
|
|
"+\\)[[:space:]]*$"))
|
|
(defconst cmake-regex-argument-quoted
|
|
(rx ?\" (* (or (not (any ?\" ?\\)) (and ?\\ anything))) ?\"))
|
|
(defconst cmake-regex-argument-unquoted
|
|
(rx (or (not (any space "()#\"\\\n")) (and ?\\ nonl))
|
|
(* (or (not (any space "()#\\\n")) (and ?\\ nonl)))))
|
|
(defconst cmake-regex-token
|
|
(rx-to-string `(group (or (regexp ,cmake-regex-comment)
|
|
?\( ?\)
|
|
(regexp ,cmake-regex-argument-unquoted)
|
|
(regexp ,cmake-regex-argument-quoted)))))
|
|
(defconst cmake-regex-indented
|
|
(rx-to-string `(and bol (* (group (or (regexp ,cmake-regex-token) (any space ?\n)))))))
|
|
(defconst cmake-regex-block-open
|
|
(rx-to-string `(and symbol-start (or ,@(append cmake-keywords-block-open
|
|
(mapcar 'downcase cmake-keywords-block-open))) symbol-end)))
|
|
(defconst cmake-regex-block-close
|
|
(rx-to-string `(and symbol-start (or ,@(append cmake-keywords-block-close
|
|
(mapcar 'downcase cmake-keywords-block-close))) symbol-end)))
|
|
(defconst cmake-regex-close
|
|
(rx-to-string `(and bol (* space) (regexp ,cmake-regex-block-close)
|
|
(* space) (regexp ,cmake-regex-paren-left))))
|
|
(defconst cmake-regex-token-paren-left (concat "^" cmake-regex-paren-left "$"))
|
|
(defconst cmake-regex-token-paren-right (concat "^" cmake-regex-paren-right "$"))
|
|
|
|
;------------------------------------------------------------------------------
|
|
|
|
;; Line indentation helper functions
|
|
|
|
(defun cmake-line-starts-inside-string ()
|
|
"Determine whether the beginning of the current line is in a string."
|
|
(save-excursion
|
|
(beginning-of-line)
|
|
(let ((parse-end (point)))
|
|
(goto-char (point-min))
|
|
(nth 3 (parse-partial-sexp (point) parse-end))
|
|
)
|
|
)
|
|
)
|
|
|
|
(defun cmake-find-last-indented-line ()
|
|
"Move to the beginning of the last line that has meaningful indentation."
|
|
(let ((point-start (point))
|
|
region)
|
|
(forward-line -1)
|
|
(setq region (buffer-substring-no-properties (point) point-start))
|
|
(while (and (not (bobp))
|
|
(or (looking-at cmake-regex-blank)
|
|
(cmake-line-starts-inside-string)
|
|
(not (and (string-match cmake-regex-indented region)
|
|
(= (length region) (match-end 0))))))
|
|
(forward-line -1)
|
|
(setq region (buffer-substring-no-properties (point) point-start))
|
|
)
|
|
)
|
|
)
|
|
|
|
;------------------------------------------------------------------------------
|
|
|
|
;;
|
|
;; Indentation increment.
|
|
;;
|
|
(defcustom cmake-tab-width 2
|
|
"Number of columns to indent cmake blocks"
|
|
:type 'integer
|
|
:group 'cmake)
|
|
|
|
;;
|
|
;; Line indentation function.
|
|
;;
|
|
(defun cmake-indent ()
|
|
"Indent current line as CMake code."
|
|
(interactive)
|
|
(unless (cmake-line-starts-inside-string)
|
|
(if (bobp)
|
|
(cmake-indent-line-to 0)
|
|
(let (cur-indent)
|
|
(save-excursion
|
|
(beginning-of-line)
|
|
(let ((point-start (point))
|
|
(closing-parens-only (looking-at cmake-regex-closing-parens-line))
|
|
(case-fold-search t) ;; case-insensitive
|
|
token)
|
|
;; Search back for the last indented line.
|
|
(cmake-find-last-indented-line)
|
|
;; Start with the indentation on this line.
|
|
(setq cur-indent (current-indentation))
|
|
(if closing-parens-only
|
|
(let ((open-parens 0))
|
|
(while (re-search-forward cmake-regex-token point-start t)
|
|
(setq token (match-string 0))
|
|
(cond
|
|
((string-match cmake-regex-token-paren-left token)
|
|
(setq open-parens (+ open-parens 1)))
|
|
((string-match cmake-regex-token-paren-right token)
|
|
(setq open-parens (- open-parens 1)))))
|
|
;; Don't outdent if last indented line has open parens
|
|
(unless (> open-parens 0)
|
|
(setq cur-indent (- cur-indent cmake-tab-width))))
|
|
;; Skip detailed analysis if last indented line is a 'closing
|
|
;; parens only line'
|
|
(unless (looking-at cmake-regex-closing-parens-line)
|
|
;; Search forward counting tokens that adjust indentation.
|
|
(while (re-search-forward cmake-regex-token point-start t)
|
|
(setq token (match-string 0))
|
|
(when (or (string-match cmake-regex-token-paren-left token)
|
|
(and (string-match cmake-regex-block-open token)
|
|
(looking-at (concat "[ \t]*" cmake-regex-paren-left))))
|
|
(setq cur-indent (+ cur-indent cmake-tab-width)))
|
|
(when (string-match cmake-regex-token-paren-right token)
|
|
(setq cur-indent (- cur-indent cmake-tab-width)))
|
|
))
|
|
(goto-char point-start)
|
|
;; If next token closes the block, decrease indentation
|
|
(when (looking-at cmake-regex-close)
|
|
(setq cur-indent (- cur-indent cmake-tab-width))
|
|
)
|
|
)
|
|
)
|
|
)
|
|
;; Indent this line by the amount selected.
|
|
(cmake-indent-line-to (max cur-indent 0))
|
|
)
|
|
)
|
|
)
|
|
)
|
|
|
|
(defun cmake-point-in-indendation ()
|
|
(string-match "^[ \\t]*$" (buffer-substring (line-beginning-position) (point))))
|
|
|
|
(defun cmake-indent-line-to (column)
|
|
"Indent the current line to COLUMN.
|
|
If point is within the existing indentation it is moved to the end of
|
|
the indentation. Otherwise it retains the same position on the line"
|
|
(if (cmake-point-in-indendation)
|
|
(indent-line-to column)
|
|
(save-excursion (indent-line-to column))))
|
|
|
|
;------------------------------------------------------------------------------
|
|
|
|
;;
|
|
;; Helper functions for buffer
|
|
;;
|
|
(defun cmake-unscreamify-buffer ()
|
|
"Convert all CMake commands to lowercase in buffer."
|
|
(interactive)
|
|
(save-excursion
|
|
(goto-char (point-min))
|
|
(while (re-search-forward "^\\([ \t]*\\)\\_<\\(\\(?:\\w\\|\\s_\\)+\\)\\_>\\([ \t]*(\\)" nil t)
|
|
(replace-match
|
|
(concat
|
|
(match-string 1)
|
|
(downcase (match-string 2))
|
|
(match-string 3))
|
|
t))
|
|
)
|
|
)
|
|
|
|
|
|
;------------------------------------------------------------------------------
|
|
|
|
;;
|
|
;; Navigation / marking by function or macro
|
|
;;
|
|
|
|
(defconst cmake--regex-defun-start
|
|
(rx line-start
|
|
(zero-or-more space)
|
|
(or "function" "macro")
|
|
(zero-or-more space)
|
|
"("))
|
|
|
|
(defconst cmake--regex-defun-end
|
|
(rx line-start
|
|
(zero-or-more space)
|
|
"end"
|
|
(or "function" "macro")
|
|
(zero-or-more space)
|
|
"(" (zero-or-more (not-char ")")) ")"))
|
|
|
|
(defun cmake-beginning-of-defun ()
|
|
"Move backward to the beginning of a CMake function or macro.
|
|
|
|
Return t unless search stops due to beginning of buffer."
|
|
(interactive)
|
|
(when (not (region-active-p))
|
|
(push-mark))
|
|
(let ((case-fold-search t))
|
|
(when (re-search-backward cmake--regex-defun-start nil 'move)
|
|
t)))
|
|
|
|
(defun cmake-end-of-defun ()
|
|
"Move forward to the end of a CMake function or macro.
|
|
|
|
Return t unless search stops due to end of buffer."
|
|
(interactive)
|
|
(when (not (region-active-p))
|
|
(push-mark))
|
|
(let ((case-fold-search t))
|
|
(when (re-search-forward cmake--regex-defun-end nil 'move)
|
|
(forward-line)
|
|
t)))
|
|
|
|
|
|
;------------------------------------------------------------------------------
|
|
|
|
;;
|
|
;; Keyword highlighting regex-to-face map.
|
|
;;
|
|
(defconst cmake-font-lock-keywords
|
|
`((,(rx-to-string `(and symbol-start
|
|
(or ,@cmake-keywords
|
|
,@(mapcar #'downcase cmake-keywords))
|
|
symbol-end))
|
|
. font-lock-keyword-face)
|
|
(,(rx symbol-start (group (+ (or word (syntax symbol)))) (* blank) ?\()
|
|
1 font-lock-function-name-face)
|
|
(,(rx "${" (group (+(any alnum "-_+/."))) "}")
|
|
1 font-lock-variable-name-face t)
|
|
)
|
|
"Highlighting expressions for CMake mode.")
|
|
|
|
;------------------------------------------------------------------------------
|
|
|
|
(defun cmake--syntax-propertize-until-bracket-close (syntax end)
|
|
;; This function assumes that a previous search has matched the
|
|
;; beginning of a bracket_comment or bracket_argument and that the
|
|
;; second capture group has matched the equal signs between the two
|
|
;; opening brackets
|
|
(let* ((mb (match-beginning 2))
|
|
(me (match-end 2))
|
|
(cb (format "]%s]" (buffer-substring mb me))))
|
|
(save-match-data
|
|
(if (search-forward cb end 'move)
|
|
(progn
|
|
(setq me (match-end 0))
|
|
(put-text-property
|
|
(1- me)
|
|
me
|
|
'syntax-table
|
|
(string-to-syntax syntax)))
|
|
(setq me end)))
|
|
(put-text-property
|
|
(match-beginning 1)
|
|
me
|
|
'syntax-multiline
|
|
t)))
|
|
|
|
(defconst cmake--syntax-propertize-function
|
|
(syntax-propertize-rules
|
|
("\\(#\\)\\[\\(=*\\)\\["
|
|
(1
|
|
(prog1 "!" (cmake--syntax-propertize-until-bracket-close "!" end))))
|
|
("\\(\\[\\)\\(=*\\)\\["
|
|
(1
|
|
(prog1 "|" (cmake--syntax-propertize-until-bracket-close "|" end))))))
|
|
|
|
;; Syntax table for this mode.
|
|
(defvar cmake-mode-syntax-table nil
|
|
"Syntax table for CMake mode.")
|
|
(or cmake-mode-syntax-table
|
|
(setq cmake-mode-syntax-table
|
|
(let ((table (make-syntax-table)))
|
|
(modify-syntax-entry ?\( "()" table)
|
|
(modify-syntax-entry ?\) ")(" table)
|
|
(modify-syntax-entry ?# "<" table)
|
|
(modify-syntax-entry ?\n ">" table)
|
|
(modify-syntax-entry ?$ "'" table)
|
|
table)))
|
|
|
|
;;
|
|
;; User hook entry point.
|
|
;;
|
|
(defvar cmake-mode-hook nil)
|
|
|
|
;;------------------------------------------------------------------------------
|
|
;; Mode definition.
|
|
;;
|
|
;;;###autoload
|
|
(define-derived-mode cmake-mode prog-mode "CMake"
|
|
"Major mode for editing CMake source files."
|
|
|
|
;; Setup jumping to beginning/end of a CMake function/macro.
|
|
(set (make-local-variable 'beginning-of-defun-function) #'cmake-beginning-of-defun)
|
|
(set (make-local-variable 'end-of-defun-function) #'cmake-end-of-defun)
|
|
|
|
; Setup font-lock mode.
|
|
(set (make-local-variable 'font-lock-defaults) '(cmake-font-lock-keywords))
|
|
; Setup indentation function.
|
|
(set (make-local-variable 'indent-line-function) 'cmake-indent)
|
|
; Setup comment syntax.
|
|
(set (make-local-variable 'comment-start) "#")
|
|
;; Setup syntax propertization
|
|
(set (make-local-variable 'syntax-propertize-function) cmake--syntax-propertize-function)
|
|
(add-hook 'syntax-propertize-extend-region-functions #'syntax-propertize-multiline nil t))
|
|
|
|
|
|
; Help mode starts here
|
|
|
|
|
|
;;;###autoload
|
|
(defun cmake-command-run (type &optional topic buffer)
|
|
"Runs the command cmake with the arguments specified. The
|
|
optional argument topic will be appended to the argument list."
|
|
(interactive "s")
|
|
(let* ((bufname (if buffer buffer (concat "*CMake" type (if topic "-") topic "*")))
|
|
(buffer (if (get-buffer bufname) (get-buffer bufname) (generate-new-buffer bufname)))
|
|
(command (concat cmake-mode-cmake-executable " " type " " (if topic (shell-quote-argument topic) topic)))
|
|
;; Turn of resizing of mini-windows for shell-command.
|
|
(resize-mini-windows nil)
|
|
)
|
|
(shell-command command buffer)
|
|
(save-selected-window
|
|
(select-window (display-buffer buffer 'not-this-window))
|
|
(cmake-mode)
|
|
(read-only-mode 1)
|
|
(view-mode 1))
|
|
)
|
|
)
|
|
|
|
;;;###autoload
|
|
(defun cmake-command-run-help (type &optional topic buffer)
|
|
"`cmake-command-run' but rendered in `rst-mode'."
|
|
(interactive "s")
|
|
(let* ((bufname (if buffer buffer (concat "*CMake" type (if topic "-") topic "*")))
|
|
(buffer (if (get-buffer bufname) (get-buffer bufname) (generate-new-buffer bufname)))
|
|
(command (concat cmake-mode-cmake-executable " " type " " (if topic (shell-quote-argument topic) topic)))
|
|
;; Turn of resizing of mini-windows for shell-command.
|
|
(resize-mini-windows nil)
|
|
)
|
|
(shell-command command buffer)
|
|
(save-selected-window
|
|
(select-window (display-buffer buffer 'not-this-window))
|
|
(rst-mode)
|
|
(read-only-mode 1)
|
|
(view-mode 1))
|
|
)
|
|
)
|
|
|
|
;;;###autoload
|
|
(defun cmake-help-list-commands ()
|
|
"Prints out a list of the cmake commands."
|
|
(interactive)
|
|
(cmake-command-run-help "--help-command-list")
|
|
)
|
|
|
|
(defvar cmake-commands '() "List of available topics for --help-command.")
|
|
(defvar cmake-help-command-history nil "Command read history.")
|
|
(defvar cmake-modules '() "List of available topics for --help-module.")
|
|
(defvar cmake-help-module-history nil "Module read history.")
|
|
(defvar cmake-variables '() "List of available topics for --help-variable.")
|
|
(defvar cmake-help-variable-history nil "Variable read history.")
|
|
(defvar cmake-properties '() "List of available topics for --help-property.")
|
|
(defvar cmake-help-property-history nil "Property read history.")
|
|
(defvar cmake-help-complete-history nil "Complete help read history.")
|
|
(defvar cmake-string-to-list-symbol
|
|
'(("command" cmake-commands cmake-help-command-history)
|
|
("module" cmake-modules cmake-help-module-history)
|
|
("variable" cmake-variables cmake-help-variable-history)
|
|
("property" cmake-properties cmake-help-property-history)
|
|
))
|
|
|
|
(defun cmake-get-list (listname)
|
|
"If the value of LISTVAR is nil, run cmake --help-LISTNAME-list
|
|
and store the result as a list in LISTVAR."
|
|
(let ((listvar (car (cdr (assoc listname cmake-string-to-list-symbol)))))
|
|
(if (not (symbol-value listvar))
|
|
(let ((temp-buffer-name "*CMake Temporary*"))
|
|
(save-window-excursion
|
|
(cmake-command-run-help (concat "--help-" listname "-list") nil temp-buffer-name)
|
|
(with-current-buffer temp-buffer-name
|
|
; FIXME: Ignore first line if it is "cmake version ..." from CMake < 3.0.
|
|
(set listvar (split-string (buffer-substring-no-properties (point-min) (point-max)) "\n" t)))))
|
|
(symbol-value listvar)
|
|
))
|
|
)
|
|
|
|
(require 'thingatpt)
|
|
(defun cmake-symbol-at-point ()
|
|
(let ((symbol (symbol-at-point)))
|
|
(and (not (null symbol))
|
|
(symbol-name symbol))))
|
|
|
|
(defun cmake-help-type (type)
|
|
(let* ((default-entry (cmake-symbol-at-point))
|
|
(history (car (cdr (cdr (assoc type cmake-string-to-list-symbol)))))
|
|
(input (completing-read
|
|
(format "CMake %s: " type) ; prompt
|
|
(cmake-get-list type) ; completions
|
|
nil ; predicate
|
|
t ; require-match
|
|
default-entry ; initial-input
|
|
history
|
|
)))
|
|
(if (string= input "")
|
|
(error "No argument given")
|
|
input))
|
|
)
|
|
|
|
;;;###autoload
|
|
(defun cmake-help-command ()
|
|
"Prints out the help message for the command the cursor is on."
|
|
(interactive)
|
|
(cmake-command-run-help "--help-command" (cmake-help-type "command") "*CMake Help*"))
|
|
|
|
;;;###autoload
|
|
(defun cmake-help-module ()
|
|
"Prints out the help message for the module the cursor is on."
|
|
(interactive)
|
|
(cmake-command-run-help "--help-module" (cmake-help-type "module") "*CMake Help*"))
|
|
|
|
;;;###autoload
|
|
(defun cmake-help-variable ()
|
|
"Prints out the help message for the variable the cursor is on."
|
|
(interactive)
|
|
(cmake-command-run-help "--help-variable" (cmake-help-type "variable") "*CMake Help*"))
|
|
|
|
;;;###autoload
|
|
(defun cmake-help-property ()
|
|
"Prints out the help message for the property the cursor is on."
|
|
(interactive)
|
|
(cmake-command-run-help "--help-property" (cmake-help-type "property") "*CMake Help*"))
|
|
|
|
;;;###autoload
|
|
(defun cmake-help ()
|
|
"Queries for any of the four available help topics and prints out the
|
|
appropriate page."
|
|
(interactive)
|
|
(let* ((default-entry (cmake-symbol-at-point))
|
|
(command-list (cmake-get-list "command"))
|
|
(variable-list (cmake-get-list "variable"))
|
|
(module-list (cmake-get-list "module"))
|
|
(property-list (cmake-get-list "property"))
|
|
(all-words (append command-list variable-list module-list property-list))
|
|
(input (completing-read
|
|
"CMake command/module/variable/property: " ; prompt
|
|
all-words ; completions
|
|
nil ; predicate
|
|
t ; require-match
|
|
default-entry ; initial-input
|
|
'cmake-help-complete-history
|
|
)))
|
|
(if (string= input "")
|
|
(error "No argument given")
|
|
(if (member input command-list)
|
|
(cmake-command-run-help "--help-command" input "*CMake Help*")
|
|
(if (member input variable-list)
|
|
(cmake-command-run-help "--help-variable" input "*CMake Help*")
|
|
(if (member input module-list)
|
|
(cmake-command-run-help "--help-module" input "*CMake Help*")
|
|
(if (member input property-list)
|
|
(cmake-command-run-help "--help-property" input "*CMake Help*")
|
|
(error "Not a know help topic.") ; this really should not happen
|
|
))))))
|
|
)
|
|
|
|
;;;###autoload
|
|
(progn
|
|
(add-to-list 'auto-mode-alist '("CMakeLists\\.txt\\'" . cmake-mode))
|
|
(add-to-list 'auto-mode-alist '("\\.cmake\\'" . cmake-mode)))
|
|
|
|
; This file provides cmake-mode.
|
|
(provide 'cmake-mode)
|
|
|
|
;;; cmake-mode.el ends here
|