Another copy of my dotfiles. Because I don't completely trust GitHub.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

913 lines
30 KiB

;;; 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