105 lines
4.1 KiB
JavaScript
105 lines
4.1 KiB
JavaScript
const commands = [
|
|
"if",
|
|
"elseif",
|
|
"else",
|
|
"endif",
|
|
"foreach",
|
|
"endforeach",
|
|
"while",
|
|
"endwhile",
|
|
"function",
|
|
"endfunction",
|
|
"macro",
|
|
"endmacro",
|
|
"block",
|
|
"endblock",
|
|
];
|
|
|
|
module.exports = grammar({
|
|
name: "cmake",
|
|
|
|
externals: ($) => [$.bracket_argument, $.bracket_comment, $.line_comment],
|
|
extras: (_) => [],
|
|
|
|
rules: {
|
|
source_file: ($) => repeat($._untrimmed_command_invocation),
|
|
|
|
escape_sequence: ($) => choice($._escape_identity, $._escape_encoded, $._escape_semicolon),
|
|
_escape_identity: (_) => /\\[^A-Za-z0-9;]/,
|
|
_escape_encoded: (_) => choice("\\t", "\\r", "\\n"),
|
|
_escape_semicolon: (_) => choice(";", "\\;"),
|
|
|
|
variable: ($) => prec.left(repeat1(choice(/[a-zA-Z0-9/_.+-]/, $.escape_sequence, $.variable_ref))),
|
|
variable_ref: ($) => choice($.normal_var, $.env_var, $.cache_var),
|
|
normal_var: ($) => seq("$", "{", $.variable, "}"),
|
|
env_var: ($) => seq("$", "ENV", "{", $.variable, "}"),
|
|
cache_var: ($) => seq("$", "CACHE", "{", $.variable, "}"),
|
|
|
|
argument: ($) => choice($.bracket_argument, $.quoted_argument, $.unquoted_argument),
|
|
_untrimmed_argument: ($) => choice(/\s/, $.bracket_comment, $.line_comment, $.argument, $._paren_argument),
|
|
_paren_argument: ($) => seq("(", repeat($._untrimmed_argument), ")"),
|
|
|
|
quoted_argument: ($) => seq('"', optional($.quoted_element), '"'),
|
|
quoted_element: ($) => repeat1(choice($.variable_ref, $._quoted_text, $.escape_sequence)),
|
|
_quoted_text: (_) => prec.left(repeat1(choice("$", /[^\\"]/))),
|
|
|
|
unquoted_argument: ($) => prec.right(repeat1(choice($.variable_ref, $._unquoted_text, $.escape_sequence))),
|
|
_unquoted_text: (_) => prec.left(repeat1(choice("$", /[^()#"\\]/))),
|
|
|
|
body: ($) => prec.right(repeat1($._untrimmed_command_invocation)),
|
|
argument_list: ($) => repeat1($._untrimmed_argument),
|
|
|
|
if_command: ($) => command($.if, $.argument_list),
|
|
elseif_command: ($) => command($.elseif, $.argument_list),
|
|
else_command: ($) => command($.else, optional($.argument_list)),
|
|
endif_command: ($) => command($.endif, optional($.argument_list)),
|
|
if_condition: ($) => seq($.if_command, repeat(choice($.body, $.elseif_command, $.else_command)), $.endif_command),
|
|
|
|
foreach_command: ($) => command($.foreach, $.argument_list),
|
|
endforeach_command: ($) => command($.endforeach, optional($.argument)),
|
|
foreach_loop: ($) => seq($.foreach_command, $.body, $.endforeach_command),
|
|
|
|
while_command: ($) => command($.while, $.argument_list),
|
|
endwhile_command: ($) => command($.endwhile, optional(seq(/\s*/, $.argument, /\s*/))),
|
|
while_loop: ($) => seq($.while_command, $.body, $.endwhile_command),
|
|
|
|
function_command: ($) => command($.function, $.argument_list),
|
|
endfunction_command: ($) => command($.endfunction, optional($.argument_list)),
|
|
function_def: ($) => seq($.function_command, $.body, $.endfunction_command),
|
|
|
|
macro_command: ($) => command($.macro, $.argument_list),
|
|
endmacro_command: ($) => command($.endmacro, optional($.argument_list)),
|
|
macro_def: ($) => seq($.macro_command, $.body, $.endmacro_command),
|
|
|
|
block_command: ($) => command($.block, $.argument_list),
|
|
endblock_command: ($) => command($.endblock, optional($.argument_list)),
|
|
block_def: ($) => seq($.block_command, $.body, $.endblock_command),
|
|
|
|
normal_command: ($) => command($.identifier, optional($.argument_list)),
|
|
|
|
_command_invocation: ($) =>
|
|
choice($.normal_command, $.if_condition, $.foreach_loop, $.while_loop, $.function_def, $.macro_def, $.block_def),
|
|
_untrimmed_command_invocation: ($) => choice(/\s/, $.bracket_comment, $.line_comment, $._command_invocation),
|
|
|
|
...commandNames(...commands),
|
|
identifier: (_) => /[A-Za-z_][A-Za-z0-9_]*/,
|
|
integer: (_) => /[+-]*\d+/,
|
|
},
|
|
});
|
|
|
|
function iregex(s) {
|
|
return new RegExp(Array.from(s).reduce((acc, value) => acc + `[${value.toLowerCase()}${value.toUpperCase()}]`, ""));
|
|
}
|
|
|
|
function commandName(name) {
|
|
return { [name]: (_) => iregex(name) };
|
|
}
|
|
|
|
function commandNames(...names) {
|
|
return Object.assign({}, ...names.map(commandName));
|
|
}
|
|
|
|
function command(name_rule, arg_rule) {
|
|
return seq(name_rule, repeat(/[\t ]/), "(", arg_rule, ")");
|
|
}
|