pegjs/lib/compiler/passes/report-infinite-recursion.js

65 lines
1.4 KiB
JavaScript
Raw Normal View History

"use strict";
const GrammarError = require( "../../grammar-error" );
2018-01-28 03:00:28 +01:00
const visitor = require( "../../ast" ).visitor;
2016-09-17 16:28:28 +02:00
// Reports left recursion in the grammar, which prevents infinite recursion in
// the generated parser.
//
// Both direct and indirect recursion is detected. The pass also correctly
// reports cases like this:
//
// start = "a"? start
//
// In general, if a rule reference can be reached without consuming any input,
// it can lead to left recursion.
function reportInfiniteRecursion( ast ) {
const visitedRules = [];
const check = visitor.build( {
rule( node ) {
visitedRules.push( node.name );
check( node.expression );
visitedRules.pop( node.name );
},
sequence( node ) {
node.elements.every( element => {
check( element );
2018-01-26 08:49:34 +01:00
return ! ast.alwaysConsumesOnSuccess( element );
} );
},
rule_ref( node ) {
if ( visitedRules.indexOf( node.name ) !== -1 ) {
visitedRules.push( node.name );
const rulePath = visitedRules.join( " -> " );
throw new GrammarError(
`Possible infinite loop when parsing (left recursion: ${ rulePath }).`,
node.location
);
}
2018-01-26 08:49:34 +01:00
check( ast.findRule( node.name ) );
}
} );
check( ast );
}
module.exports = reportInfiniteRecursion;