unofficial mirror of emacs-devel@gnu.org 
 help / color / mirror / code / Atom feed
From: "A. Soare" <alinsoar@voila.fr>
To: "Emacs   Dev  [emacs-devel]" <emacs-devel@gnu.org>
Subject: Backus Naur
Date: Sat,  2 Oct 2010 13:09:41 +0200 (CEST)	[thread overview]
Message-ID: <4437394.2522901286017781295.JavaMail.www@wwinf4636> (raw)

[-- Attachment #1: Type: text/plain, Size: 529 bytes --]

I wrote for myself a backus naur mode, that I use for the grammar of C language.

I attach here the code, and the grammar of C

Emacs has already some major modes for working with extended bnf, which is standardized. This is not the case for my implementation, that I wrote only for my own use.


Alin



____________________________________________________

  Découvrez les nouveaux modèles de voitures présentés au Mondial de l’Automobile à Paris : http://actu.voila.fr/evenementiel/salon-auto-paris-2010/

[-- Attachment #2: test-grammar --]
[-- Type: application/octet-stream, Size: 12131 bytes --]


% * C 99 grammar *
% ****************

decimal-floating-constant:
	fractional-constant [ exponent-part ] [ floating-suffix ]
	| digit-sequence exponent-part [ floating-suffix ]

token:
	keyword [opt-test]
	identifier
	constant
	string-literal
	| punctuator
preprocessing-token:
	 header-name
	| identifier
	| pp-number
	| character-constant
	| string-literal
	| punctuator
%	each non-white-space character that cannot be one of the above

keyword:
	@auto |        @enum |     @restrict | @unsigned
	| @break |     @extern |   @Return |   @void
	| @case |      @float |    @short |    @volatile
	| @char |      @for |      @signed |   @while
	| @const |     @goto |     @sizeof |   @_Bool
	| @continue |  @if |       @static |   @_Complex
	| @default |   @inline |   @struct |   @_Imaginary
	| @do |        @int |      @switch
	| @double |    @long |     @typedef
	| @else |      @register | @union

%  IDENTIFIER: An identifier can denote an object; a function; a tag
%  or a member of a structure, union, or enumeration; a typedef name;
%  a label name; a macro name; or a macro parameter.

identifier:
	identifier-nondigit
	| identifier
	identifier-nondigit
	| identifier digit
identifier-nondigit:
	nondigit
	| universal-character-name
%	other implementation-defined characters
nondigit:
	@_ | @a | @b | @c | @d | @e | @f | @g | @h
	| @i | @j | @k | @l | @m | @n | @o | @p | @q
	| @r | @s | @t | @u | @v | @w | @x | @y | @z
	| @A | @B | @C | @D | @E | @F | @G | @H | @I
	| @J | @K | @L | @M | @N | @O | @P | @Q | @R
	| @S | @T | @U | @V | @W | @X | @Y | @Z
digit:
	@0 | @1 | @2 | @3 | @4 | @5 | @6 | @7 | @8 | @9

universal-character-name:
	@\u hex-quad
	| @\U hex-quad hex-quad
hex-quad:
	hexadecimal-digit hexadecimal-digit
	| hexadecimal-digit hexadecimal-digit

constant:
	integer-constant
	| floating-constant
	| enumeration-constant
	| character-constant

integer-constant:
	decimal-constant [ integer-suffix ]
	| octal-constant [ integer-suffix ]
	| hexadecimal-constant [ integer-suffix ]
decimal-constant:
	nonzero-digit
	| decimal-constant digit
octal-constant:
	@0
	| octal-constant octal-digit
hexadecimal-constant:
	hexadecimal-prefix hexadecimal-digit
	| hexadecimal-constant hexadecimal-digit
hexadecimal-prefix:
	@0x @0X
nonzero-digit:
	@1 @2 @3 @4 @5 @6 @7 @8 @9
octal-digit:
	@0 @1 @2 @3 @4 @5 @6 @7
hexadecimal-digit:
	@0 @1 @2 @3 @4 @5 @6 @7 @8 @9
	| @a @b @c @d @e @f
	| @A @B @C @D @E @F
integer-suffix:
	unsigned-suffix [ long-suffix ]
	| unsigned-suffix long-long-suffix
	| long-suffix [ unsigned-suffix ]
	| long-long-suffix [ unsigned-suffix ]
unsigned-suffix:
	@u @U
long-suffix:
	@l @L
long-long-suffix:
	@ll @LL

floating-constant:
	decimal-floating-constant
	| hexadecimal-floating-constant
decimal-floating-constant:
	fractional-constant [ exponent-part ] [ floating-suffix ]
	| digit-sequence exponent-part [ floating-suffix ]
hexadecimal-floating-constant:
	hexadecimal-prefix hexadecimal-fractional-constant
	| binary-exponent-part [ floating-suffix ]
	| hexadecimal-prefix hexadecimal-digit-sequence
	| binary-exponent-part [ floating-suffix ]
fractional-constant:
	[ digit-sequence ] @. digit-sequence
	| digit-sequence @.
exponent-part:
	@e [ sign ] digit-sequence
	| @E [ sign ] digit-sequence
sign:
	@+ @-
digit-sequence:
	digit
	| digit-sequence digit
hexadecimal-fractional-constant:
	[ hexadecimal-digit-sequence ] @.
	| hexadecimal-digit-sequence
	| hexadecimal-digit-sequence @.
binary-exponent-part:
	@p [ sign ] digit-sequence
	| @P [ sign ] digit-sequence
hexadecimal-digit-sequence:
	hexadecimal-digit
	| hexadecimal-digit-sequence hexadecimal-digit
floating-suffix:
	@f @l @F @L
enumeration-constant:
	identifier
character-constant:
	@@@ c-char-sequence @@
	| @L @@@ c-char-sequence @@
c-char-sequence:
	c-char
	| c-char-sequence c-char
c-char:
%	any member of the source character set except'
%	the single-quote ', backslash \, or new-line character
%	escape-sequence

escape-sequence:
	simple-escape-sequence
	| octal-escape-sequence
	| hexadecimal-escape-sequence
	| universal-character-name
simple-escape-sequence:
	@\@ @\" @\? @\\
	| @\a @\b @\f @\n @\r @\t @\v
octal-escape-sequence:
	@\ octal-digit
	| @\ octal-digit octal-digit
	| @\ octal-digit octal-digit octal-digit
hexadecimal-escape-sequence:
	@\x hexadecimal-digit
	hexadecimal-escape-sequence hexadecimal-digit

string-literal:
s-char-sequence:
	s-char
	| s-char-sequence s-char
s-char:
%	any member of the source character set except
%	the double-quote ", backslash \, or new-line character
%	escape-sequence

punctuator:
	@[ @] @( @) @{ @} @. @->
	| @++ @-- @& @* @+ @- @~ @!
	| @/ @% @<< @>> @< @> @<= @>=
	| @==  @!=  @^   @|  @&& @||
	| @? @: @; @... @= @*= @/= @%=
	| @+= @-= @<<= @>>=  @&=   @^=
	| @|= @, @# @## @<: @:> @<%
	| @%> @%: @%: @%:
header-name:
	@< h-char-sequence @>
	| @" q-char-sequence @"
h-char-sequence:
	h-char
	| h-char-sequence h-char
h-char:
%	any member of the source character set except
%	the new-line character and >
q-char-sequence:
	q-char
	| q-char-sequence q-char
q-char:
%	any member of the source character set except
%	the new-line character and "

pp-number:
	digit
	| @. digit
	| pp-number digit
	| pp-number identifier-nondigit
	| pp-number @e sign
	| pp-number @E sign
	| pp-number @p sign
	| pp-number @P sign
	| pp-number @.
primary-expression:
	identifier
	constant
	string-literal
	@( expression @)

postfix-expression:
	primary-expression
	postfix-expression @[ expression @]
	postfix-expression @( [ argument-expression-list ] @)
	postfix-expression @. identifier
	postfix-expression @-> identifier
	postfix-expression @++
	postfix-expression @--
	@( type-name @) @{ initializer-list @}
	@( type-name @) @{ initializer-list @, @}

argument-expression-list:
	assignment-expression
	argument-expression-list @, assignment-expression

unary-expression:
	postfix-expression
	@++ unary-expression
	@-- unary-expression
	unary-operator cast-expression
	@sizeof unary-expression
	@sizeof @( type-name @)
unary-operator:
	@& @* @+ @- @~ @!

cast-expression:
	unary-expression
	@( type-name @) cast-expression

multiplicative-expression:
	cast-expression
	multiplicative-expression @* cast-expression
	multiplicative-expression @/ cast-expression
	multiplicative-expression @% cast-expression

additive-expression:
	multiplicative-expression
	additive-expression @+ multiplicative-expression
	additive-expression @- multiplicative-expression

shift-expression:
	additive-expression
	shift-expression @<< additive-expression
	shift-expression @>> additive-expression

relational-expression:
	shift-expression
	relational-expression @<  shift-expression
	relational-expression @>  shift-expression
	relational-expression @<= shift-expression
	relational-expression @>= shift-expression

equality-expression:
	relational-expression
	equality-expression @== relational-expression
	equality-expression @!= relational-expression

AND-expression:
	equality-expression
	AND-expression @& equality-expression

exclusive-OR-expression:
	AND-expression
	exclusive-OR-expression @^ AND-expression
inclusive-OR-expression:
	exclusive-OR-expression
	inclusive-OR-expression @| exclusive-OR-expression

logical-AND-expression:
	inclusive-OR-expression
	logical-AND-expression @&& inclusive-OR-expression

logical-OR-expression:
	logical-AND-expression
	logical-OR-expression @|| logical-AND-expression

conditional-expression:
	logical-OR-expression
	logical-OR-expression @? expression @: conditional-expression

assignment-expression:
	conditional-expression
	unary-expression assignment-operator assignment-expression
assignment-operator:
	@= @*= @/= @%= @+= @-= @<<= @>>= @&= @^= @|=
expression:
	assignment-expression
	expression @, assignment-expression

constant-expression:
	conditional-expression

declaration:
	declaration-specifiers [ init-declarator-list ] @;
declaration-specifiers:
	storage-class-specifier [ declaration-specifiers ]
	type-specifier [ declaration-specifiers ]
	type-qualifier [ declaration-specifiers ]
	function-specifier [ declaration-specifiers ]
init-declarator-list:
	init-declarator
	init-declarator-list @, init-declarator
init-declarator:
	declarator
	declarator @= initializer

storage-class-specifier:
	@typedef
	@extern
	@static
	@auto
	@register

type-specifier:
	@void
	@char
	@short
	@int
	@long
	@float
	@double
	@signed
	@unsigned
	@_Bool
	@_Complex
	@_Imaginary
	struct-or-union-specifier
	enum-specifier
	typedef-name

type-qualifier:
	@const
	@restrict
	@volatile

function-specifier:
	@inline

struct-or-union-specifier:
	struct-or-union [ identifier ] @{ struct-declaration-list @}
	struct-or-union identifier
struct-or-union:
	@struct
	@union
struct-declaration-list:
	struct-declaration
	struct-declaration-list struct-declaration
struct-declaration:
	specifier-qualifier-list struct-declarator-list @;
specifier-qualifier-list:
	type-specifier [ specifier-qualifier-list ]
	type-qualifier [ specifier-qualifier-list ]
struct-declarator-list:
	struct-declarator
	struct-declarator-list @, struct-declarator
struct-declarator:
	declarator
	[ declarator ] @: constant-expression

enum-specifier:
	@enum [ identifier ] @{ enumerator-list @}
	@enum [ identifier ] @{ enumerator-list @, @}
	@enum identifier
enumerator-list:
	enumerator
	enumerator-list @, enumerator
enumerator:
	enumeration-constant
	enumeration-constant @= constant-expression

declarator:
	[ pointer ] direct-declarator
direct-declarator:
	identifier
	@( declarator @)
	direct-declarator @[ [ type-qualifier-list ] [ assignment-expression ] @]
	direct-declarator @[ @static
	[ type-qualifier-list ] assignment-expression @]
	direct-declarator @[ type-qualifier-list @static assignment-expression @]
	direct-declarator @[ [ type-qualifier-list ] @* @]
	direct-declarator @( parameter-type-list @)
	direct-declarator @( [ identifier-list ] @)
pointer:
	@* [ type-qualifier-list ]
	@* [ type-qualifier-list ] pointer

	type-qualifier
	type-qualifier-list type-qualifier
parameter-type-list:
	parameter-list
	parameter-list @, @...
parameter-list:
	parameter-declaration
	parameter-list @, parameter-declaration
parameter-declaration:
	declaration-specifiers declarator
	declaration-specifiers [ abstract-declarator ]
identifier-list:
	identifier
	identifier-list @, identifier

type-name:
	specifier-qualifier-list [ abstract-declarator ]
abstract-declarator:
	pointer
	[ pointer ] direct-abstract-declarator
direct-abstract-declarator:
	@( abstract-declarator @)
	[ direct-abstract-declarator ] @[ [ assignment-expression ] @]
	[ direct-abstract-declarator ] @[ @* @]
	[ direct-abstract-declarator ] @( [ parameter-type-list ] @)

typedef-name:
	identifier

initializer:
	assignment-expression
	@{ initializer-list @}
	@{ initializer-list @, @}
initializer-list:
	[ designation ] initializer
	initializer-list @, [ designation ] initializer
designation:
	designator-list @=
designator-list:
	designator
	designator-list designator
designator:
	@[ constant-expression @]
	@. identifier

statement:
	labeled-statement
	compound-statement
	expression-statement
	selection-statement
	iteration-statement
	jump-statement

labeled-statement:
	identifier @: statement
	@case constant-expression @: statement
	@default @: statement

compound-statement:
	@{ [ block-item-list ] @}
block-item-list:
	block-item
	block-item-list block-item
block-item:
	declaration
	statement

expression-statement:
	[ expression ] @;

selection-statement:
	@if @( expression @) statement
	@if @( expression @) statement @else statement
	@switch @( expression @) statement

iteration-statement:
	@while @( expression @) statement
	@do statement @while @( expression @) @;
	@for @( [ expression ] @; [ expression ] @; [ expression ] @) statement
	@for @( declaration [ expression ] @; [ expression ] @) statement

jump-statement:
	@goto identifier @;
	@continue @;
	@break @;
	@return [ expression ] @;

translation-unit:
	external-declaration
	translation-unit external-declaration
external-declaration:
	function-definition
	declaration
function-definition:
	declaration-specifiers declarator [ declaration-list ] compound-statement
declaration-list:
	declaration
	declaration-list declaration


[-- Warning: decoded text below may be mangled, UTF-8 assumed --]
[-- Attachment #3: bn-mode.el --]
[-- Type: text/x-emacs-lisp; name=bn-mode.el, Size: 22590 bytes --]


(defvar bn-left-nonterminal-definition
  "^\\([A-Za-z-]*\\)\\(:\\)[ ]\\{0,3\\}$"
  "The regular expression that marks the left side from the \
definition of a rule" )

(defvar imenu-create-index-function)
(defvar imenu-use-markers)

(defvar bn-left-nonterminal-list nil
  "keeps the list of all the left nonterminals. this could be
defined locally only, inside `bn-lazy-font-lock'. Used for
imenu and completion-list .")

(defvar bn-timer-var nil
  "non nil means that a fontification will follow")

(defvar bn-search-history nil
  "used by search-completion to keep the history of searched
  strings" )

(defvar bn-comment-char "%"
  "a line which starts with this character is commented")

(defvar after-change-functions nil )

(defun bn-mode nil
  "A major mode for well writing context-free grammars
in Backus-Naur notation."
  (interactive)
  'bn-mode
  'fundamental-mode
  '"Backus-Naur"
  (kill-all-local-variables)
  (setq major-mode 'bn-mode
	mode-name "Backus-Naur" )
  (make-local-variable 'bn-comment-char)
  (make-local-variable 'bn-overlays)
  (make-local-variable 'bn-iterator)
  (make-local-variable 'after-change-functions)
  (make-local-variable 'bn-left-nonterminal-list)
  (make-local-variable 'bn-rotate-around-left-nonterminal-match-ring)

  (make-local-variable 'imenu-create-index-function)
  (setq imenu-create-index-function 'bn-imenu)

  (make-local-variable 'imenu-generic-expression)
  (setq imenu-generic-expression t)

  (and t
       (fboundp 'imenu-add-to-menubar)
       (imenu-add-to-menubar "Index"))

  (setq font-lock-defaults
	(list
	 (list
	  (list
	   bn-left-nonterminal-definition
	   '(1 'bn-left-nonterminal-definition-font )
	   '(2 'bn-left-nonterminal-definition-closure ) )
	  '( "\\('\\)\\([[:graph:]]*\\)\\('\\)"
	     (1 'bn-terminal-closure )
	     (2 'bn-terminal-definition )
	     (3 'bn-terminal-closure )
	     )
	  '( "\\(@\\)\\([[:graph:]]+\\)"
	     (1 'bn-terminal-closure )
	     (2 'bn-terminal-definition )
	     )
          (list
	   (concat "[^[:graph:]]"
		   "\\([[]\\)"
		   "\\(.*\\)"
		   "[^[:graph:]]"
		   "\\([]]\\)")
	   '(1 'bn-optional-closure )
	   '(2 'bn-optional-definition )
	   '(3 'bn-optional-closure ))
          (list
	   (concat "^" "\\(" bn-comment-char "\\)"
                   "\\(.*\\)" )
	   '(1 'bn-comment-sign )
	   '(2 'bn-commented-text ) )
	  (list
	   "|"
	   '(0 'bn-or ) ) )
	 t ) )
  (bn-stop-timer)
  (remove-overlays)
  (setq after-change-functions (add-to-list 'after-change-functions 'bn-change) )
  (local-set-key [tab] 'bn-indent)
  (local-set-key [home]
		 (lambda nil (interactive)
		   (if (equal (char-before) ?\n)
		       (bn-beginning-of-defun)
		     (beginning-of-line) ) ) )
  (local-set-key [end]
		 (lambda nil (interactive)
		   (if (equal (char-after) ?\n)
		       (bn-end-of-defun)
		     (end-of-line) ) ) )
  (bn-lazy-font-lock)
  (local-set-key "\M-s" 'bn-search )
  (local-set-key (kbd "C-x SPC")
		 (lambda ()
		   "jump to the definition of the nonterminal under the cursor. "
                   (interactive "")
		   (let* ((type (get-text-property (point) 'type))
			  (nonterminal
			   (and type
				(get-text-property (point) 'nonterminal) ) )
			  (p
			   (and nonterminal
			    (get-text-property (point) 'position) ) ) )
		     (cond
		      ( (eq type 'right)
			(and p
			     (bn-jump p )
			     '(recenter)
			     (message "jump to `%s' %s" nonterminal p ) ) )
		      ((eq type 'left)
			   (error "`%s' is a left nonterminal\n" nonterminal ) ) ) ) ) )

  (make-local-variable 'bn-search-history)

  (local-set-key (kbd "C-;")
                 (lambda nil
                   "replaces isearch"
                   (interactive)
                   (let ((aa
                          (completing-read
                           "search left-nonterminal: "
                           (mapcar 'caar bn-left-nonterminal-list )
                           nil
                           t
                           nil
                           bn-search-history
                           nil
                           nil ) ) )
                     (and aa
                          (bn-jump (cdr (assoc aa (mapcar 'car bn-left-nonterminal-list ) ) ) ) ) ) ) )
  )

(defun bn-jump (p)
  (push-mark)
  (goto-char p) )

(defun bn-stop-timer ()
  "stop the timer for fontification"
  (and bn-timer-var
       (progn
	 (cancel-timer bn-timer-var)
	 (setq bn-timer-var nil) ) ) )

(defface bn-left-nonterminal-definition-font
  '((((class grayscale) (background light)) (:foreground "Gray90" :weight bold))
    (((class grayscale) (background dark)) (:foreground "DimGray" :weight bold))
    (((class color) (min-colors 88) (background light)) (:foreground "red"))
    (((class color) (min-colors 88) (background dark)) (:foreground "orange red"))
    (((class color) (min-colors 16) (background light)) (:foreground "orange"))
    (((class color) (min-colors 16) (background dark)) (:foreground "gold"))
    (((class color) (min-colors 8)) (:foreground "dark orange"))
    (t (:weight bold :underline t)))
  "font used for left nonterminals." )

(defface bn-right-nonterminal-definition-font
  '((((class grayscale) (background light)) (:foreground "AntiqueWhite4" :weight bold))
    (((class grayscale) (background dark)) (:foreground "AntiqueWhite4" :weight bold))
    (((class color) (min-colors 88) (background light)) (:foreground "AntiqueWhite4"))
    (((class color) (min-colors 88) (background dark)) (:foreground "AntiqueWhite4"))
    (((class color) (min-colors 16) (background light)) (:foreground "AntiqueWhite4"))
    (((class color) (min-colors 16) (background dark)) (:foreground "AntiqueWhite4"))
    (((class color) (min-colors 8)) (:foreground "AntiqueWhite4"))
    (t (:weight bold :underline t)))
  "font used for right nonterminals." )

(defface bn-left-nonterminal-definition-closure
  '((((class grayscale) (background light)) (:foreground "Gray90" :weight bold))
    (((class grayscale) (background dark)) (:foreground "DimGray" :weight bold))
    (((class color) (min-colors 88) (background light)) (:foreground "deep sky blue"))
    (((class color) (min-colors 88) (background dark)) (:foreground "spring green"))
    (((class color) (min-colors 16) (background light)) (:foreground "dark sea green"))
    (((class color) (min-colors 16) (background dark)) (:foreground "yellow green"))
    (((class color) (min-colors 8)) (:foreground "olive drab"))
    (t (:weight bold :underline t)))
  "font used for the character `:' at the end of a left nonterminal." )

(defface bn-terminal-definition
  '((((class grayscale) (background light)) (:foreground "DeepSkyBlue1" :weight bold))
    (((class grayscale) (background dark)) (:foreground "DeepSkyBlue1" :weight bold))
    (((class color) (min-colors 88) (background light)) (:foreground "DeepSkyBlue1"))
    (((class color) (min-colors 88) (background dark)) (:foreground "DeepSkyBlue1"))
    (((class color) (min-colors 16) (background light)) (:foreground "DeepSkyBlue1"))
    (((class color) (min-colors 16) (background dark)) (:foreground "DeepSkyBlue1"))
    (((class color) (min-colors 8)) (:foreground "DeepSkyBlue1"))
    (t (:weight bold :underline t)))
  "font used for terminals." )

(defface bn-terminal-closure
  '((((class grayscale) (background light)) (:foreground "dark blue" :weight bold))
    (((class grayscale) (background dark)) (:foreground "dark blue" :weight bold))
    (((class color) (min-colors 88) (background light)) (:foreground "dark blue"))
    (((class color) (min-colors 88) (background dark)) (:foreground "dark blue"))
    (((class color) (min-colors 16) (background light)) (:foreground "dark blue"))
    (((class color) (min-colors 16) (background dark)) (:foreground "dark blue"))
    (((class color) (min-colors 8)) (:foreground "dark blue"))
    (t (:weight bold :underline t)))
  "font used for the characters \' around a terminal." )

(defface bn-optional-closure
  '((((class grayscale) (background light)) (:foreground "RoyalBlue1" :weight bold))
    (((class grayscale) (background dark)) (:foreground "RoyalBlue1" :weight bold))
    (((class color) (min-colors 88) (background light)) (:foreground "RoyalBlue1"))
    (((class color) (min-colors 88) (background dark)) (:foreground "RoyalBlue1"))
    (((class color) (min-colors 16) (background light)) (:foreground "RoyalBlue1"))
    (((class color) (min-colors 16) (background dark)) (:foreground "RoyalBlue1"))
    (((class color) (min-colors 8)) (:foreground "RoyalBlue1"))
    (t (:weight bold :underline t)))
  "font used for the characters [ ] about an optional ." )

(defface bn-optional-definition
  '((((class grayscale) (background light)) (:foreground "dark green" :weight bold))
    (((class grayscale) (background dark)) (:foreground "dark green" :weight bold))
    (((class color) (min-colors 88) (background light)) (:foreground "dark green"))
    (((class color) (min-colors 88) (background dark)) (:foreground "dark green"))
    (((class color) (min-colors 16) (background light)) (:foreground "dark green"))
    (((class color) (min-colors 16) (background dark)) (:foreground "dark green"))
    (((class color) (min-colors 8)) (:foreground "dark green"))
    (t (:weight bold :underline t)))
  "font used for the ." )

(defface bn-search-rotate-face
  '((((class grayscale) (background light)) (:foreground "Gray90" :weight bold))
    (((class grayscale) (background dark)) (:foreground "DimGray" :weight bold))
    (((class color) (min-colors 88) (background light)) (:foreground "chartreuse1"))
    (((class color) (min-colors 88) (background dark)) (:foreground "chartreuse2"))
    (((class color) (min-colors 16) (background light)) (:foreground "chartreuse3"))
    (((class color) (min-colors 16) (background dark)) (:foreground "chartreuse4"))
    (((class color) (min-colors 8)) (:foreground "SpringGreen4"))
    (t (:weight bold :underline t)))
  "font used for matched right nonterminals." )

(defface bn-commented-text
  '((((class grayscale) (background light)) (:foreground "medium sea green" :weight bold))
    (((class grayscale) (background dark)) (:foreground "medium sea green" :weight bold))
    (((class color) (min-colors 88) (background light)) (:foreground "medium sea green"))
    (((class color) (min-colors 88) (background dark)) (:foreground "medium sea green"))
    (((class color) (min-colors 16) (background light)) (:foreground "medium sea green"))
    (((class color) (min-colors 16) (background dark)) (:foreground "medium sea green"))
    (((class color) (min-colors 8)) (:foreground "medium sea green"))
    (t (:weight bold :underline t)))
  "font used for the text on a commented line" )

(defface bn-or
  '((((class grayscale) (background light)) (:foreground "brown" :weight bold))
    (((class grayscale) (background dark)) (:foreground "blue" :weight bold))
    (((class color) (min-colors 88) (background light)) (:foreground "blue"))
    (((class color) (min-colors 88) (background dark)) (:foreground "blue"))
    (((class color) (min-colors 16) (background light)) (:foreground "blue"))
    (((class color) (min-colors 16) (background dark)) (:foreground "blue"))
    (((class color) (min-colors 8)) (:foreground "blue"))
    (t (:weight bold :underline t :foreground "blue" :weight bold)))
  "font used for" )

(defface bn-comment-sign
  '((((class grayscale) (background light)) (:foreground "Gray90" :weight bold))
    (((class grayscale) (background dark)) (:foreground "DimGray" :weight bold))
    (((class color) (min-colors 88) (background light)) (:foreground "indian red"))
    (((class color) (min-colors 88) (background dark)) (:foreground "lightgreen"))
    (((class color) (min-colors 16) (background light)) (:foreground "tomato"))
    (((class color) (min-colors 16) (background dark)) (:foreground "dark violet"))
    (((class color) (min-colors 8)) (:foreground "sienna"))
    (t (:weight bold :underline t)))
  "font used for the character that comments a line" )

(defun bn-indent ()
  " The lines that contain a left nonterminal are indented from
  0. These are the lines that match the regular expression
 `bn-left-nonterminal-definition'.
The blank lines are cleared.
The lines which contain a rule are indented with a tab."
  (interactive)
  (let (
	;; initial point
	(cc (current-column))
	;; end of line
	(eol (prog2
		 (move-end-of-line nil)
		 (current-column) ) )
	;; beginning of text
	(bot (prog2
		 (move-beginning-of-line nil)
		 (skip-chars-forward "[:blank:]")
		 (current-column) ) ) )
    (cond
     ;; empty line
     ((zerop eol)
      t)
     ;; lines with blanks
     ((eq bot eol)
      (move-beginning-of-line nil)
      (just-one-space 0) )
     ;; line containing a left nonterminal
     ((save-excursion
	(move-to-column eol)
	(and (> (current-column) 3)
	     (search-backward ":" (- (point) 1) t ) ) )
      (move-beginning-of-line nil)
      (just-one-space 0) )
     ;; for the rest... add a tab at the beginning of the line
     ( (save-excursion
	 (move-beginning-of-line nil)
	 (not (search-forward-regexp "\t[^[:blank:]]" (+ 2 (point) ) t ) ) )
       (just-one-space 0)
       (insert-char ?\t 1)
       (skip-chars-forward "[:blank:]") ) )
    ;; if the cursor was before the indetation inside the word, keep
    ;; it on the same position; if the initial position of the cursor
    ;; was before the word, move it at the beginning of the word
    (move-to-column (+ (current-column)
		       (max 0 (- cc bot) ) ) ) ) )

(defun bn-commented-line-p ()
  "return t when the point is inside a commented line, and nil
otherwise."
  (equal
   (save-excursion
     (beginning-of-line)
     (following-char) )
   (elt bn-comment-char 0 ) ) )

(defun bn-lazy-font-lock ()
  "called after every buffer change to re-fontify the text."
  (bn-stop-timer)
  (save-excursion
    (let (list-of-left-nonterminals
	  (map-right-nonterminal (make-sparse-keymap) )
	  (map-left-nonterminal (make-sparse-keymap) )
	  ;; font lock properties modify the buffer, so this hook is
	  ;; called recursively. temporarly after-change-functions
	  ;; must be nil.
	  after-change-functions
	  ;; changing the properties, one changes the modified status
	  (buffer-modified-flag (buffer-modified-p) )
	  ;; this is equalent in this case to setting
	  ;; after-change-functions to nil. However, we keep both
	  (inhibit-modification-hooks t)
	  ;; make a temporary copy of the undo
	  ;; list. add-text-properties adds information in the undo
	  ;; list, even if in this case it shouldn't
	  (ul buffer-undo-list) )
      ;; we disable the undo information temporarly, because changing
      ;; the fontification is added by default in the undo list.
      (buffer-disable-undo)
      (define-key map-right-nonterminal [mouse-3]
	(lambda (event) (interactive "e")
	  "jump to a left terminal definition"
	  (let ((p (get-text-property
		    (cadr (cadr event) )
		    'position) ) )
	    (and p
		 (bn-jump p )
		 '(recenter)
		 (message (format "%s" p ) ) ) ) ) )
      (define-key map-left-nonterminal "\M-s" 'bn-search )
      (goto-char (point-min) )
      ;; make the list of left nonterminals out
      (while (search-forward-regexp
              ;; TODO: to add fontification here for optional
              ;; nonterminals inside `[ ... ]'
              (concat bn-left-nonterminal-definition )
              nil t)
	(push (cons (substring-no-properties (match-string 1) )
		    (match-beginning 1) )
	      list-of-left-nonterminals ) )
      ;; clean the previous fontification
      (remove-text-properties
       (point-min) (point-max)
       '(mouse-face highlight font-lock-face keymap position) )
      (setq bn-left-nonterminal-list nil)
      ;; add first the fontification of font-lock
      (font-lock-fontify-buffer)
      ;; scan the buffer to find the positions where every
      ;; left-nonterminal appears in the rules
      (dolist (m list-of-left-nonterminals)
	(let ( (match-positions) )
	  (goto-char (point-min) )
	  (while
	      (and (search-forward-regexp
		    (concat "[\t ]" "\\(" (car m) "\\)" "[\n .\t]") nil t)
		   (goto-char (1- (point) ) ) )
	    (setq make-fontification
		  (not (eq (get-text-property (1- (point)) 'face)
			   'bn-optional-definition)))
	    (and
	     ;; if a right nonterminal is found, add it to the
	     ;; matching list, subject to not being on a commented line
	     ;; or inside a [ comment ]
	     make-fontification
	     (not (bn-commented-line-p) )
	     (add-text-properties
	      (match-beginning 1)
	      (match-end 1)
	      (list
	       'font-lock-face 'bn-right-nonterminal-definition-font)))

	    (and
	     (not (bn-commented-line-p) )
	     (add-text-properties
	      (match-beginning 1)
	      (match-end 1)
	      (list 'mouse-face 'highlight
		    'help-echo
		    (concat (car m)
                            (format " at %d." (cdr m) )
                            (let ((nl (count-lines (point) (cdr m) ) ) )
                              (cond ( (> (point) (cdr m) )
                                      (format " %d line%s backward."
                                              (1- nl)
                                              (if (equal 2 nl)
                                                    ""
                                                "s" ) ) )
                                    ( t
                                      (format " %d line%s forward."
                                              nl
                                              (if (equal 1 nl)
                                                  ""
                                                "s" ) ) )
                                  ) )
                            "\nright click to jump there." )
		    'position (cdr m)
		    'type 'right
		    'nonterminal (car m)
		    'keymap map-right-nonterminal) )
	     (push (match-beginning 1) match-positions) ) )
	  ;; add text properties to the left nonterminal
	  (add-text-properties
	   (cdr m) (+ (length (car m) ) (cdr m) )
	   (list
	    'help-echo (concat "`" (car m) "'" " "
			       (if match-positions
				   (format "at %s" (reverse match-positions) )
				 "is a start symbol" )
			       (format
				".\n %s to rotate around matches."
				(substitute-command-keys "\\[bn-search]") ) )
	    'position (progn
			(let ( ( ring-positions (make-ring (length match-positions) ) ) )
			  (dolist (p (reverse match-positions) )
			    (ring-insert ring-positions p) )
			  ring-positions) )
	    'nonterminal (car m)
	    'type 'left
	    'keymap map-left-nonterminal) )
	  (push (cons m match-positions) bn-left-nonterminal-list ) ) )
      ;; restore the undo, modified status
      (restore-buffer-modified-p buffer-modified-flag)
      (buffer-enable-undo)
      (setq buffer-undo-list ul )
      ) ) )

(defun bn-change (x y z)
  "hook which is called after every buffer change. it starts a
timer that fontify the current buffer, according to the rules
defined in the grammar. if a previous timer is active, stop it
before starting the new timer."
  ;; clear the overlays from the previous search if any
  (bn-stop-timer)
  '(setq overriding-terminal-local-map nil)
  (setq bn-timer-var
	(run-with-idle-timer
	 1 nil 'bn-lazy-font-lock) ) )

(defun bn-beginning-of-defun ()
  "jump at the beginning of a left nonterminal definition."
  (search-backward-regexp bn-left-nonterminal-definition nil t) )

(defun bn-end-of-defun ()
  "jump at the end of a left nonterminal definition."
 (if (search-forward-regexp bn-left-nonterminal-definition nil t)
      (beginning-of-line)
   (bn-jump (point-max) ) )
  (let ((repeat t))
    (while repeat
      (skip-chars-backward " \t\n")
      (if (not (bn-commented-line-p) )
	  (setq repeat nil)
	(beginning-of-line) ) ) )
  (skip-chars-forward " \t")
  (and (not (eobp) )
       (forward-char) ) )

(defun bn-search ()
  "Search for the left nonterminal under the cursor"
  (interactive)
  (let* ((left-nonterminal (get-text-property (point) 'nonterminal ) )
	 (left-nonterminal-positions
	  (and left-nonterminal
	       (get-text-property (point) 'position ) ) ) )
    (cond
     ;; a symbol accessed from nowhere
     ( (and
	(ring-p left-nonterminal-positions )
	(ring-empty-p left-nonterminal-positions ) )
       (message "`%s' is a start symbol" left-nonterminal ) )
     ;; at cursor position there is a left nonterminal
     ( (ring-p left-nonterminal-positions )
       ;; fontify right nonterminals that match
       (dolist (p (ring-elements left-nonterminal-positions ) )
	 (let ((ov (make-overlay p (+ p (length left-nonterminal ) ) ) ) )
	   (overlay-put ov 'face 'bn-search-rotate-face) ) )
       ;; loop of input events
       (let* (key
	      (l t)
	      (pos 0)
	      (number-of-matches (ring-size left-nonterminal-positions ) )
	      (many-matches (> number-of-matches 1 ) )
	      (Mkeys (concat
		      "\n"
		      (if many-matches
			  "`n' jump to the next match. `p' jump to the previous match. "
			"" )
		      "`q' interrupts search." ) )
	      (M (concat
		  (format "`%s'" left-nonterminal )
		  (format " : `%s': " (reverse (ring-elements left-nonterminal-positions ) ) )
		  Mkeys  ) ) )
	 ;; jump to the first match
	 (bn-jump (ring-ref left-nonterminal-positions pos) )
	 '(recenter)
	 (message (concat "Search for " M ) )
	 (while l
	   (setq key (read-key-sequence nil) )
	   (cond
	    ;; quit the search
	    ( (and (stringp key)
		   (string-equal key  "q" ) )
	      (message "quit search." )
	      (setq l nil ) )
	    ;; searck for the next match
	    ( (and (stringp key)
		   many-matches
		   (string-equal key  "n" ) )
	      (setq pos (1- pos) )
	      (goto-char (ring-ref left-nonterminal-positions pos) )
	      (message M ) )
	    ;; search for the previous match
	    ( (and (stringp key)
		   many-matches
		   (string-equal key  "p" ) )
	      (setq pos (1+ pos) )
	      (goto-char (ring-ref left-nonterminal-positions pos ) )
	      (message M ) )
	    (t
	     (message
	      (concat (format "`%s'" (key-description key ) )
		      " does not match. "
		      Mkeys ) ) ) ) ) )
       ;; removing overlays
       (remove-overlays) )
     ((listp left-nonterminal-positions )
      (error "No left-nonterminal defintion at point" ))
     (t
      (error " `%s' : cannot loop for a right symbol. " left-nonterminal ) ) ) ) )

'(defun bn-imenu ()
  (mapcar
   (lambda (l)
     (cons (caar l)
	   (if (cadr l)
	       (copy-marker (cadr l) )
	     0) ) )
   bn-left-nonterminal-list) )

(defun bn-imenu ()
  (mapcar
   (lambda (l)
     (cons (car l)
	   (copy-marker (cdr l) ) ) )
   (mapcar 'car
	   bn-left-nonterminal-list) ) )





             reply	other threads:[~2010-10-02 11:09 UTC|newest]

Thread overview: 3+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2010-10-02 11:09 A. Soare [this message]
2010-10-04  2:03 ` Backus Naur Richard Stallman
  -- strict thread matches above, loose matches on Subject: below --
2010-10-08 19:14 A. Soare

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

  List information: https://www.gnu.org/software/emacs/

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=4437394.2522901286017781295.JavaMail.www@wwinf4636 \
    --to=alinsoar@voila.fr \
    --cc=emacs-devel@gnu.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
Code repositories for project(s) associated with this public inbox

	https://git.savannah.gnu.org/cgit/emacs.git

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).