|
|
- ;;; bison-mode.el --- Major mode for editing bison, yacc and lex files.
-
- ;; Copyright (C) 1998 Eric Beuscher
- ;;
- ;; Author: Eric Beuscher <beuscher@eecs.tulane.edu>
- ;; Created: 2 Feb 1998
- ;; Version: 0.2
- ;; Keywords: bison-mode, yacc-mode
-
- ;; This program is free software; you can redistribute it and/or modify
- ;; it under the terms of the GNU General Public License as published by
- ;; the Free Software Foundation; either version 2 of the License, or
- ;; (at your option) any later version.
-
- ;; This program is distributed in the hope that it will be useful,
- ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
- ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- ;; GNU General Public License for more details.
-
- ;;; Commentary:
-
- ;;;; I wrote this since I saw one mode for yacc files out there roaming the
- ;;;; world. I was daunted by the fact the it was written in 1990, and Emacs
- ;;;; has evolved so much since then (this I assume based on its evolution since
- ;;;; i started using it). So I figured if i wanted one, I should make it
- ;;;; myself. Please excuse idiosyncrasies, as this was my first major mode
- ;;;; of this kind. The indentation code may be a bit weird, I am not sure,
- ;;;; it was my first go at doing Emacs indentation, so I look at how other
- ;;;; modes did it, but then basically did what I thought was right
-
- ;;;; I hope this is useful to other hackers, and happy Bison/Yacc hacking
- ;;;; If you have ideas/suggestions/problems with this code, I can be reached at
- ;;;; beuscher@eecs.tulane.edu
-
- ;;;; Eric --- Sat Mar 7 1:40:20 CDT 1998
-
- ;;;; Bison Sections:
- ;;;; there are five sections to a bison file (if you include the area above the
- ;;;; C declarations section. most everything in this file either does
- ;;;; actions based on which section you are deemed to be in, or based on an
- ;;;; assumption that the function will only be called from certain sections.
- ;;;; the function `bison--section-p' is the section parser
-
- ;;;; Indentation:
- ;;;; indentations are done based on the section of code you are in. there is
- ;;;; a procedure `bison--within-braced-c-expression-p' that checks for being in
- ;;;; C code. if you are within c-code, indentations should occur based on
- ;;;; how you have your C indentation set up. i am pretty sure this is the
- ;;;; case.
- ;;;; there are four variables, which control bison indentation within either
- ;;;; the bison declarations section or the bison grammar section
- ;;;; `bison-rule-separator-column'
- ;;;; `bison-rule-separator-column'
- ;;;; `bison-decl-type-column'
- ;;;; `bison-decl-token-column'
-
- ;;;; flaw: indentation works on a per-line basis, unless within braced C sexp,
- ;;;; i should fix this someday
- ;;;; and to make matters worse, i never took out c-indent-region, so that is
- ;;;; still the state of the `indent-region-function' variable
-
- ;;;; Electricity:
- ;;;; by default, there are electric -colon, -pipe, -open-brace, -close-brace,
- ;;;; -semicolon, -percent, -less-than, -greater-than
- ;;;; the indentation caused by these work closely with the 4 indentation
- ;;;; variables mentioned above.
- ;;;; any of these can be turned off individually by setting the appropriate
- ;;;; `bison-electric-...' variable. or all of them can be turned off by
- ;;;; setting `bison-all-electricity-off'
-
- ;;;; todo: should make available a way to use C-electricity if in C sexps
-
- ;;; Code:
-
- (require 'cc-mode)
-
- ;;;###autoload
- (add-to-list 'auto-mode-alist '("\\.y\\'" . bison-mode))
- ;;;###autoload
- (add-to-list 'auto-mode-alist '("\\.l\\'" . flex-mode))
- ;;;###autoload
- (add-to-list 'auto-mode-alist '("\\.jison\\'" . jison-mode))
-
- ;; *************** internal vars ***************
-
- (defvar bison--declarers '("%union" "%token" "%type"
- "%left" "%right" "%nonassoc")
- "commands which can declare a token or state type")
-
- (defvar bison--word-constituent-re "\\(\\sw\\|_\\)")
- (defvar bison--production-re
- (concat "^" bison--word-constituent-re "+:"))
-
- (defvar bison--pre-c-decls-section 0
- "section before c-declarations-section, if that section exists")
- (defvar bison--c-decls-section 1
- "section denoted by %{ and $} for c-declarations at the top of a bison file")
- (defvar bison--bison-decls-section 2
- "section before the rules section")
- (defvar bison--grammar-rules-section 3
- "section delimited by %%'s where productions and rules are enumerated")
- (defvar bison--c-code-section 4
- "section after the second %% where c-code can be placed")
-
- (defvar bison--c-decls-section-opener "%{")
- (defvar bison--c-decls-section-closer "%}")
- (defvar bison--grammar-rules-section-delimeter "%%")
-
-
- ;; *************** user-definable vars ***************
-
- (defvar bison-rule-separator-column 8
- "column for rule and production separators \"|\" and \";\"")
- (defvar bison-rule-enumeration-column 16
- "column for beginning enumeration of a production's rules")
- (defvar bison-decl-type-column 8
- "columnn in which tokens' and states' types should be when declared")
- (defvar bison-decl-token-column 24
- "column in which tokens and states are listed when declared,
- as with %token, %type, ...")
-
-
- (defvar bison-all-electricity-off nil
- "non-nil means all electric keys will be disabled,
- nil means that a bison-electric-* key will be on or off based on the individual
- key's electric variable")
-
- ;;; i know lisp has the dual name spaces, but i find it more aesthetically
- ;;; pleasing to not take advantage of that
- (defvar bison-electric-colon-v t
- "non-nil means use an electric colon")
- (defvar bison-electric-pipe-v t
- "non-nil means use an electric pipe")
- (defvar bison-electric-open-brace-v t
- "non-nil means use an electric open-brace")
- (defvar bison-electric-close-brace-v t
- "non-nil means use an electric close-brace")
- (defvar bison-electric-semicolon-v t
- "non-nil means use an electric semicolon")
- (defvar bison-electric-percent-v t
- "non-nil means use an electric percent")
- (defvar bison-electric-less-than-v t
- "non-nil means use an electric less-than")
- (defvar bison-electric-greater-than-v t
- "non-nil means use an electric greater-than")
-
-
- (defconst bison-font-lock-keywords
- (append
- (list
- (cons (concat "^\\(" (regexp-opt bison--declarers) "\\)")
- '(1 font-lock-keyword-face))
- )
- c-font-lock-keywords)
- "Default expressions to highlight in Bison mode")
-
- ;; *************** utilities ***************
-
- (defun just-no-space ()
- "Delete all spaces and tabs around point, leaving no spaces."
- (interactive "*")
- (skip-chars-backward " \t")
- (delete-region (point) (progn (skip-chars-forward " \t") (point)))
- t)
-
- (defun previous-white-space-p ()
- "return t if there is whitespace between the beginning of the line and the
- current (point)"
- (save-excursion
- (let ((current-point (point)))
- (beginning-of-line)
- (if (re-search-forward "\\s " current-point t)
- t
- nil))))
-
- (defun previous-non-ws-p ()
- "return t if there are non-whitespace characters between beginning of line
- and \(point\)"
- (save-excursion
- (let ((current-point (point)))
- (beginning-of-line)
- (re-search-forward "[^ \t]" current-point t)
- )))
-
- (defun following-non-ws-p ()
- "return t if there are non-whitespace characters on the line"
- (save-excursion
- (let ((current-point (point)))
- (end-of-line)
- (re-search-backward "[^ \t]+" current-point t)
- )))
-
- (defun line-of-whitespace-p ()
- "return t if the line consists of nothiing but whitespace, nil otherwise"
- (save-excursion
- (let ((eol (progn (end-of-line) (point))))
- (beginning-of-line) ;; should already be there anyway
- (not (re-search-forward "[^ \t\n]" eol t)))))
-
- ;; *************** bison-mode ***************
-
- ;;;###autoload
- (define-derived-mode bison-mode c-mode "Bison"
- "Major mode for editing bison/yacc files."
-
- ;; try to set the indentation correctly
- (setq c-basic-offset 4)
-
- (c-set-offset 'knr-argdecl-intro 0)
-
- ;; remove auto and hungry anything
- (c-toggle-auto-hungry-state -1)
- (c-toggle-auto-newline -1)
- (c-toggle-hungry-state -1)
-
- (use-local-map bison-mode-map)
-
- (define-key bison-mode-map ":" 'bison-electric-colon)
- (define-key bison-mode-map "|" 'bison-electric-pipe)
- (define-key bison-mode-map "{" 'bison-electric-open-brace)
- (define-key bison-mode-map "}" 'bison-electric-close-brace)
- (define-key bison-mode-map ";" 'bison-electric-semicolon)
- (define-key bison-mode-map "%" 'bison-electric-percent)
- (define-key bison-mode-map "<" 'bison-electric-less-than)
- (define-key bison-mode-map ">" 'bison-electric-greater-than)
-
- (define-key bison-mode-map [tab] 'bison-indent-line)
-
- (make-local-variable 'indent-line-function)
- (setq indent-line-function 'bison-indent-new-line)
- (make-local-variable 'comment-start)
- (make-local-variable 'comment-end)
- (setq comment-start "/*"
- comment-end "*/")
- (make-local-variable 'font-lock-keywords)
- (setq font-lock-keywords nil)
- (set (make-local-variable 'font-lock-defaults) '(bison-font-lock-keywords)))
-
-
- ;; *************** section parsers ***************
-
- (defun bison--section-p ()
- "Return the section that user is currently in"
- (save-excursion
- (let ((bound (point)))
- (goto-char (point-min))
- (bison--section-p-helper bound))))
-
- (defun bison--section-p-helper (bound)
- (if (re-search-forward
- (concat "^" bison--c-decls-section-opener)
- bound t)
- (if (re-search-forward
- (concat "^" bison--c-decls-section-closer)
- bound t)
- (if (re-search-forward
- (concat "^" bison--grammar-rules-section-delimeter)
- bound t)
- (if (re-search-forward
- (concat "^" bison--grammar-rules-section-delimeter)
- bound t)
- bison--c-code-section
- bison--grammar-rules-section)
- bison--bison-decls-section)
- bison--c-decls-section)
- (if (re-search-forward
- (concat "^" bison--grammar-rules-section-delimeter)
- bound t)
- (if (re-search-forward
- (concat "^" bison--grammar-rules-section-delimeter)
- bound t)
- bison--c-code-section
- bison--grammar-rules-section)
- (if (re-search-forward
- (concat "^" bison--c-decls-section-opener)
- nil t)
- bison--pre-c-decls-section
- (if (re-search-forward
- (concat "^" bison--grammar-rules-section-delimeter)
- nil t)
- bison--bison-decls-section
- bison--pre-c-decls-section)))))
-
-
- ;; *************** syntax parsers ***************
-
- (defun bison--production-p ()
- "return t if the \(point\) rests immediately after a production"
- (save-excursion
- (let ((current-point (point)))
- (beginning-of-line)
- (let ((position (re-search-forward
- bison--production-re current-point t)))
- (and position
- (not (previous-white-space-p))
- (= position current-point))))))
-
- (defun bison--find-production-opener ()
- "return and goto the point of the nearest production opener above \(point\)"
- (re-search-backward bison--production-re nil t))
-
-
- (defun bison--find-next-production ()
- "return the position of the beginning of the next production,
- or nil if there isnt one"
- (save-excursion
- (if (re-search-forward bison--production-re nil t)
- (progn
- (beginning-of-line)
- (point))
- nil)))
-
- (defun bison--find-grammar-end ()
- "return the position of the end of the grammar rules (assuming we are within
- the grammar rules section), or nil if there isnt one"
- (save-excursion
- (if (re-search-forward
- (concat "^" bison--grammar-rules-section-delimeter)
- nil t)
- (progn
- (beginning-of-line)
- (point))
- nil)))
-
- (defun bison--find-grammar-begin ()
- "return the position of the beginning of the grammar rules (assuming we are
- within the grammar rules section), or nil if there isnt one"
- (save-excursion
- (if (re-search-backward
- (concat "^" bison--grammar-rules-section-delimeter)
- nil t)
- (point)
- nil)))
-
- (defun bison--within-started-production-p ()
- "is used by bison-electric-* functions to determine actions
- return t if within a production, nil if not
-
- a point is within a production if there is some non whitespace text before
- either the beginnings of another production or the end of the grammar rules"
- (save-excursion
- (let ((bound (cond ((bison--find-next-production))
- ((bison--find-grammar-end))
- (t nil))))
- (if bound
- (let ((sval (re-search-forward
- (concat "\\(\\s \\|" ;; whitespace or
- ;; comments
- (regexp-quote comment-start)
- "\\(.\\|\n\\)*" ;; comment body
- (regexp-quote comment-end)
- "\\)+") ;; end or
- bound t)))
- (if sval
- (not (= sval bound))
- nil))
- nil))))
-
- (defun bison--within-some-sexp-p (starter ender)
- "return t if the \(point\) is within the sexp marked by the re's STARTER and
- ENDER"
- (save-excursion
- (let ((current-point (point)))
- (if (re-search-backward starter nil t) ;; find nearest starter
- ;; look for ender, if found, then not within sexp
- (progn
- (goto-char (match-end 0))
- (not (re-search-forward ender current-point t)))))))
-
- (defun bison--within-c-comment-p ()
- "return t if the point is within a c comment delimited by \"/*\" \"*/\""
- (bison--within-some-sexp-p (regexp-quote comment-start)
- (regexp-quote comment-end)))
-
-
- (defun bison--within-string-p (&optional point)
- "
- start from the beginning of the buffer and toggle state as un-escaped \"'s are
- found."
- (let ((point (or point (point)))
- (in-p nil))
- (save-excursion
- (goto-char (point-min))
-
- (while (re-search-forward "[^\\]\"" point t)
- (setq in-p (not in-p)))
-
- in-p)))
-
- ;;; bison--within-braced-c-expression-p
- ;;; new and improved, no more recursion, does not break when literal strings
- ;;; contain un-matched braces
- (defun bison--within-braced-c-expression-p (section)
- "return t if the point is within an sexp delimited by braces \({,}\)
- "
- (save-excursion
- (bison--within-braced-c-expression-p-h section (point))))
-
- (defun bison--within-braced-c-expression-p-h (section low-pt)
- "
- Notes:
- save excursion is done higher up, so i dont concern myself here.
- "
- (cond ((= section bison--pre-c-decls-section) nil)
- ((= section bison--c-decls-section)
- (let ((opener (save-excursion (search-backward "%{"))))
- (bison--within-braced-c-expression-p-h-h opener low-pt)))
- ((= section bison--bison-decls-section)
- (let ((opener (save-excursion
- (or (search-backward "%}" nil t)
- (point-min)))))
- (bison--within-braced-c-expression-p-h-h opener low-pt)))
- ((= section bison--grammar-rules-section)
- (let ((opener (save-excursion (bison--find-production-opener))))
- (if opener
- (bison--within-braced-c-expression-p-h-h opener low-pt)
- nil)))
- ((= section bison--c-code-section)
- t)))
-
- (defun bison--within-braced-c-expression-p-h-h (high-pt low-pt)
- "
- Notes:
- HIGH-PT goes toward (point-min), LOW-PT goes toward (point-max)
- save excursion is done higher up, so i dont concern myself here.
- "
- (let ((pt (point)))
- (let ((success nil) (count 1) (done nil))
- ;; loop until open brace found, that is not in comment or string literal
- (while (and (not done)
- (re-search-backward "[^%]{" high-pt t count)) ;find nearest
- ;starter
- (goto-char (match-end 0))
- (if (or (bison--within-c-comment-p)
- (bison--within-string-p))
-
- (setq count (+ count 1))
- (progn
- (setq success t)
- (setq done t))))
-
- (if success
- (let ((end-pt
- (condition-case nil
- (progn
- (backward-char)
- (forward-sexp)
- (point))
- (error nil))))
- (if end-pt
- (if (> end-pt low-pt)
- t ; then in braced-c-exp
- nil)
- t)) ; if no sexp close brace, then w/in
- nil))))
-
-
- (defun bison--bison-decl-opener-p (bol eol)
- "return t if the current line is a bison declaration starter
- \(i.e. has a %type, %token, %right, ...\)"
- (save-excursion
- (goto-char bol)
- (re-search-forward
- (concat "^" (regexp-opt (copy-sequence bison--declarers))) eol t)))
-
- (defun bison--production-opener-p (bol eol)
- "return t if the current line is a line that introduces a new production"
- (save-excursion
- (goto-char bol)
- (re-search-forward bison--production-re eol t)))
-
- (defun bison--find-bison-semicolon ()
- "return the position of next semicolon not within braces, nil otherwise"
- (save-excursion
- (if (search-forward ";" nil t)
- (if (not (bison--within-braced-c-expression-p (bison--section-p)))
- (point)
- (bison--find-bison-semicolon))
- nil)))
-
- (defun bison--within-production-body-p (section)
- "return t if the \(point\) is within the body of a production
-
- this procedure will fail if it is in a production header"
- (save-excursion
- (if (= section bison--grammar-rules-section)
- (let ((current-point (point)))
- (if (re-search-backward bison--production-re nil t)
- t
- nil))
- nil)))
-
- (defun bison--production-alternative-p (bol eol section)
- "return t if the current line contains a \"|\" used to designate a rule
- alternative"
- (save-excursion
- (goto-char bol)
- (if (search-forward "|" eol t)
- (not (bison--within-braced-c-expression-p section))
- nil)))
-
-
- ;; *************** indent functions ***************
-
- (defun bison--handle-indent-c-sexp (section indent-column bol)
- (let* ((o-brace (re-search-backward "[^%]{" bol t))
- )
- (if o-brace
- (if (save-excursion
- (goto-char o-brace)
- (bison--within-braced-c-expression-p section))
- (c-indent-line)
- (if (= (current-indentation) o-brace) ;; if o-brace is first char
- (if (not (= o-brace indent-column)) ;; but not in right spot
- (progn
- (back-to-indentation)
- (just-no-space)
- (indent-to-column indent-column))
- ;; else all is good
- )
- ;; else, non-ws before o-brace, leave it alone
- ))
- (c-indent-line))))
-
- (defun bison-indent-new-line (&optional c-sexp)
- "Indent a fresh line of bison code
-
- assumes indenting a new line, i.e. at column 0
- "
- (interactive)
-
- (let* ((section (bison--section-p))
- (c-sexp (or c-sexp (bison--within-braced-c-expression-p section)))
- )
- (cond
- (c-sexp
- (cond
- ((= section bison--grammar-rules-section)
- (c-indent-line
- (save-excursion
- (forward-line -1)
- (let ((bol (save-excursion (beginning-of-line) (point)))
- (eol (save-excursion (end-of-line) (point))))
- (if (bison--production-opener-p bol eol)
- (list
- (cons 'defun-block-intro
- (progn
- (re-search-forward bison--production-re) ; SIGERR
- (- (re-search-forward "[^ \t]") ; SIGERR
- 1))))
- nil)))))
- (t (c-indent-line))))
- ((= section bison--pre-c-decls-section)
- (c-indent-line))
- ((= section bison--bison-decls-section)
- (indent-to-column bison-decl-token-column))
- ((= section bison--grammar-rules-section)
- (indent-to-column
- (save-excursion
- (let* ((bound (or (save-excursion (bison--find-production-opener))
- (bison--find-grammar-begin)))
- (prev-semi (search-backward ";" bound t))
- )
- (if prev-semi
- (if (bison--within-braced-c-expression-p section) ; CRACK
- bison-rule-enumeration-column
- 0)
- (if (save-excursion (bison--find-production-opener))
- bison-rule-enumeration-column
- 0))))))
- ((= section bison--c-code-section)) ;;leave-alone
- )))
-
- (defun bison-indent-line ()
- "Indent a line of bison code."
- (interactive)
-
- (let* ((pos (- (point-max) (point)))
- (reset-pt (function (lambda ()
- (if (> (- (point-max) pos) (point))
- (goto-char (- (point-max) pos))))))
- (bol (save-excursion (beginning-of-line) (point)))
- (eol (save-excursion (end-of-line) (point)))
- )
- (let* ((section (bison--section-p))
- (c-sexp (bison--within-braced-c-expression-p section))
- (ws-line (line-of-whitespace-p))
- )
- (cond
- ;; if you are a line of whitespace, let indent-new-line take care of it
- (ws-line
- (bison-indent-new-line c-sexp))
-
- ((= section bison--pre-c-decls-section)
- ;; leave things alone
- )
-
- ((= section bison--c-decls-section)
- (if c-sexp
- (bison--handle-indent-c-sexp section 0 bol)
- (if (not (= (current-indentation) 0))
- (progn
- (back-to-indentation)
- (just-no-space)
- (funcall reset-pt)))))
-
- ((= section bison--bison-decls-section)
- (let ((opener (bison--bison-decl-opener-p bol eol)))
- (cond
- (opener
- (goto-char opener)
- (skip-chars-forward " \t" eol)
- (if (looking-at "{")
- (save-excursion
- (if (following-non-ws-p)
- (progn
- (forward-char 1)
- (just-no-space)
- (newline)
- (bison-indent-new-line t))))
- (let ((complete-type t))
- (if (looking-at "<")
- (progn
- (setq complete-type nil)
- (if (not (= (current-column) bison-decl-type-column))
- (progn
- (just-no-space)
- (indent-to-column bison-decl-type-column))
- (and (re-search-forward
- (concat "<" bison--word-constituent-re "+>")
- eol t)
- (setq complete-type t)))))
- (and complete-type
- (skip-chars-forward " \t" eol)
- (looking-at
- (concat "\\(" bison--word-constituent-re "\\|'\\)"))
- (if (not (= (current-column) bison-decl-token-column))
- (progn
- (just-no-space)
- (indent-to-column bison-decl-token-column))))))
- (funcall reset-pt))
- (c-sexp
- (bison--handle-indent-c-sexp section 0 bol))
- (t
- (back-to-indentation)
- ;; only tab in names, leave comments alone
- (cond (;; put word-constiuents in bison-decl-token-column
- (looking-at bison--word-constituent-re)
- (if (not (= (current-column) bison-decl-token-column))
- (progn
- (just-no-space)
- (indent-to-column bison-decl-token-column))))
- ;; put/keep close-brace in the 0 column
- ((looking-at "}")
- (if (not (= (current-column) 0))
- (just-no-space)))
- ;; leave comments alone
- ((looking-at (regexp-quote comment-start)) nil)
- ;; else do nothing
- )
- (funcall reset-pt)))))
- ((= section bison--grammar-rules-section)
- (cond
- ((bison--production-opener-p bol eol)
- (beginning-of-line)
- (re-search-forward bison--production-re);; SIGERR
- (if (following-non-ws-p)
- (if (> (current-column) bison-rule-enumeration-column)
- (progn
- (just-no-space)
- (newline)
- (indent-to-column bison-rule-enumeration-column))
- (save-excursion
- (re-search-forward bison--word-constituent-re);; SIGERR
- (let ((col (current-column)))
- (cond ((> col (+ 1 bison-rule-enumeration-column))
- (forward-char -1)
- (just-no-space)
- (indent-to-column bison-rule-enumeration-column))
- ((< col (+ 1 bison-rule-enumeration-column))
- (forward-char -1)
- (indent-to-column
- bison-rule-enumeration-column)))))))
- (funcall reset-pt))
- ((bison--production-alternative-p bol eol section)
- (back-to-indentation);; should put point on "|"
- (if (not (= (current-column) bison-rule-separator-column))
- (progn
- (just-no-space)
- (indent-to-column bison-rule-separator-column)))
- (forward-char 1)
- (if (following-non-ws-p)
- (save-excursion
- (re-search-forward bison--word-constituent-re);; SIGERR
- (let ((col (current-column)))
- (cond ((> col (+ 1 bison-rule-enumeration-column))
- (forward-char -1)
- (just-no-space)
- (indent-to-column bison-rule-enumeration-column))
- ((< col (+ 1 bison-rule-enumeration-column))
- (forward-char -1)
- (indent-to-column
- bison-rule-enumeration-column))))))
- (funcall reset-pt))
- (c-sexp
- (bison--handle-indent-c-sexp
- section bison-rule-enumeration-column bol)
- (funcall reset-pt))
- ((bison--within-production-body-p section)
- (back-to-indentation)
- (if (not (= (current-column) bison-rule-enumeration-column))
- (progn
- (just-no-space)
- (indent-to-column
- bison-rule-enumeration-column)))
- (funcall reset-pt))
- (t
- (let ((cur-ind (current-indentation)))
- (if (eq (save-excursion (search-backward "}" bol t))
- cur-ind)
- (if (not (= cur-ind bison-rule-enumeration-column))
- (progn
- (back-to-indentation)
- (just-no-space)
- (indent-to-column bison-rule-enumeration-column)
- (funcall reset-pt)))
- ;; else leave alone
- )))))
- ((= section bison--c-code-section)
- (c-indent-line))
- ))))
-
- ;; *************** electric-functions ***************
-
- (defun bison-electric-colon (arg)
- "If the colon <:> delineates a production,
- then insert a semicolon on the next line in the BISON-RULE-SEPARATOR-COLUMN,
- put the cursor in the BISON-RULE-ENUMERATION-COLUMN for the beginning
- of the rule
- else just run self-insert-command
- A colon delineates a production by the fact that it is immediately preceded by
- a word(alphanumerics or '_''s), and there is no previous white space.
- "
- (interactive "P")
-
- (self-insert-command (prefix-numeric-value arg))
- (if (and bison-electric-colon-v
- (not bison-all-electricity-off))
- (if (and (= bison--grammar-rules-section (bison--section-p))
- (bison--production-p)
- (not (bison--within-started-production-p)))
- (progn
- (save-excursion ; put in a closing semicolon
- (newline)
- (indent-to-column bison-rule-separator-column)
- (insert ";"))
- (save-excursion ; remove opening whitespace
- (if (re-search-backward
- "\\s "
- (save-excursion (beginning-of-line) (point))
- t)
- (just-no-space)))
- (if (not (< (current-column) bison-rule-enumeration-column))
- (newline))
- (indent-to-column bison-rule-enumeration-column)))))
-
- (defun bison-electric-pipe (arg)
- "If the pipe <|> is used as a rule separator within a production,
- then move it into BISON-RULE-SEPARATOR-COLUMN
- indent to BISON-RULE-ENUMERATION-COLUMN on the same line
- else just run self-insert-command
- "
- (interactive "P")
-
- (if (and bison-electric-pipe-v
- (not bison-all-electricity-off)
- (= bison--grammar-rules-section (bison--section-p))
- (line-of-whitespace-p)
- )
- (progn
- (beginning-of-line)
- (just-no-space)
- (indent-to-column bison-rule-separator-column)
- (self-insert-command (prefix-numeric-value arg))
- (indent-to-column bison-rule-enumeration-column)
- )
- (self-insert-command (prefix-numeric-value arg))))
-
- (defun bison-electric-open-brace (arg)
- "used for the opening brace of a C action definition for production rules,
- if there is only whitespace before \(point\), then put open-brace in
- bison-rule-enumeration-column"
- (interactive "P")
-
- (if (and bison-electric-open-brace-v
- (not bison-all-electricity-off))
- (let ((section (bison--section-p)))
- (cond ((and (= section bison--grammar-rules-section)
- (not (bison--within-braced-c-expression-p section))
- (not (previous-non-ws-p)))
- (if (not (= (current-column) bison-rule-enumeration-column))
- (progn
- (just-no-space)
- (indent-to-column bison-rule-enumeration-column))))
- ((and (= section bison--bison-decls-section)
- (not (bison--within-braced-c-expression-p section))
- (not (previous-non-ws-p)))
- (if (not (= (current-column) 0))
- (progn
- (just-no-space)
- (indent-to-column 0)))))))
-
- (self-insert-command (prefix-numeric-value arg)))
-
-
- (defun bison-electric-close-brace (arg)
- "If the close-brace \"}\" is used as the c-declarations section closer
- in \"%}\", then make sure the \"%}\" indents to the beginning of the line"
- (interactive "P")
-
- (self-insert-command (prefix-numeric-value arg))
-
- (if (and bison-electric-close-brace-v
- (not bison-all-electricity-off))
- (cond ((search-backward "%}" (- (point) 2) t)
- (if (= (bison--section-p) bison--c-decls-section)
- (progn
- (just-no-space)
- (forward-char 2)) ; for "%}"
- (forward-char 1)))
- )))
-
- (defun bison-electric-semicolon (arg)
- "if the semicolon is used to end a production, then place it in
- bison-rule-separator-column
-
- a semicolon is deemed to be used for ending a production if it is not found
- within braces
-
- this is just self-insert-command as i have yet to write the actual
- bison-electric-semicolon function yet
- "
- (interactive "P")
-
- (self-insert-command (prefix-numeric-value arg)))
-
- (defun bison-electric-percent (arg)
- "If the percent is a declarer in the bison declaration's section,
- then put it in the 0 column."
- (interactive "P")
-
- (if (and bison-electric-percent-v
- (not bison-all-electricity-off))
- (let ((section (bison--section-p)))
- (if (and (= section bison--bison-decls-section)
- (not (bison--within-braced-c-expression-p section))
- (not (previous-non-ws-p))
- (not (= (current-column) 0)))
- (just-no-space))))
-
- (self-insert-command (prefix-numeric-value arg)))
-
- (defun bison-electric-less-than (arg)
- "If the less-than is a type declarer opener for tokens in the bison
- declaration section, then put it in the bison-decl-type-column column."
- (interactive "P")
-
- (if (and bison-electric-less-than-v
- (not bison-all-electricity-off))
- (if (and (= (bison--section-p) bison--bison-decls-section)
- (bison--bison-decl-opener-p
- (save-excursion (beginning-of-line) (point))
- (point)))
- (progn
- (just-no-space)
- (indent-to-column bison-decl-type-column))))
-
- (self-insert-command (prefix-numeric-value arg)))
-
- (defun bison-electric-greater-than (arg)
- "If the greater-than is a type declarer closer for tokens in the bison
- declaration section, then indent to bison-decl-token-column."
- (interactive "P")
-
- (self-insert-command (prefix-numeric-value arg))
-
- (if (and bison-electric-greater-than-v
- (not bison-all-electricity-off))
- (let ((current-pt (point))
- (bol (save-excursion (beginning-of-line) (point))))
- (if (and (= (bison--section-p) bison--bison-decls-section)
- (bison--bison-decl-opener-p bol (point)))
- (if (search-backward "<" bol t)
- (if (re-search-forward
- (concat "<" bison--word-constituent-re "+>")
- current-pt t)
- (if (not (following-non-ws-p))
- (progn
- (just-no-space)
- (indent-to-column bison-decl-token-column)))))))))
-
- ;;;###autoload
- (define-derived-mode jison-mode bison-mode "jison"
- "Major mode for editing jison files.")
- ;;;###autoload
- (define-derived-mode flex-mode bison-mode "flex"
- "Major mode for editing flex files. (bison-mode by any other name)")
-
- (provide 'bison-mode)
- (provide 'jison-mode)
- (provide 'flex-mode)
-
- ;;; bison-mode.el ends here
|