;;; bison-mode.el --- Major mode for editing bison, yacc and lex files. ;; Copyright (C) 1998 Eric Beuscher ;; ;; Author: Eric Beuscher ;; 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