wschar:alias = " " | "\t" | "\n" | "\b" | "\f" | "\r" | "\v"; wsc:alias = wschar | comment; block-comment:alias = "/*" ("*" [^/] | [^*])* "*/"; line-comment:alias = "//" [^\n]*; comment-segment:alias = line-comment | block-comment; ws-no-nl:alias = " " | "\t" | "\b" | "\f" | "\r" | "\v"; comment = comment-segment (ws-no-nl* "\n"? ws-no-nl* comment-segment)*; any-char = "."; // equivalent to [^] // caution: newline is accepted class-not = "^"; class-char = [^\\\[\]\^\-] | "\\" .; char-range = class-char "-" class-char; char-class = "[" class-not? (class-char | char-range)* "]"; // newline is accepted sequence-char = [^\\"] | "\\" .; char-sequence = "\"" sequence-char* "\""; terminal:alias = any-char | char-class | char-sequence; symbol = [^\\ \n\t\b\f\r\v/.\[\]\"{}\^+*?|():=;]+; group:alias = "(" wsc* expression wsc* ")"; number:alias = [0-9]+; count = number; count-quantifier = "{" wsc* count wsc* "}"; range-from = number; range-to = number; range-quantifier = "{" wsc* range-from? wsc* "," wsc* range-to? wsc* "}"; one-or-more = "+"; zero-or-more = "*"; zero-or-one = "?"; quantity:alias = count-quantifier | range-quantifier | one-or-more | zero-or-more | zero-or-one; item = (terminal | symbol | group) quantity?; sequence = item (wsc* item)*; element:alias = terminal | symbol | group | sequence; // DOC: how the order matters choice = element (wsc* "|" wsc* element)+; // DOC: not having 'not' needs some tricks sometimes expression:alias = terminal | symbol | group | sequence | choice; alias = "alias"; ws = "ws"; nows = "nows"; doc = "doc"; root = "root"; flag:alias = alias | ws | nows | doc | root; definition = symbol (":" flag)* wsc* "=" wsc* expression; definitions:alias = definition (wsc* ";" (wsc | ";")* definition)*; syntax:root = (wsc | ";")* definitions? (wsc | ";")*;