pigpigyyy/YueScript

Adding '{' makes code run exponentially slow.

nm17 opened this issue · 8 comments

nm17 commented

Just an interesting thing that I found. It seems that continuously adding '{' doubles execution time with each '{' and also consumes 100% CPU while running. Not sure if this should be treated as a bug or not.

Long backtrace of running {{{{{{{{{{{{{{{{{{ (thats 18 '{'):

https://gist.github.com/nm17/ed64b33f838a97c1285aa5ef1229eab0

Confirmed this performance issue with continuously nesting table literals.
Found that it's caused by not arranging the orders of syntax parsing rules properly.
Commit 0cc452c should have fixed this.
Before this commit I got

-- file test.yue
{{{{{{{{{{{{{{{{{{}}}}}}}}}}}}}}}}}}
> yue -b test.yue
Parse time:     975.43 ms
Compile time:   0.039875 ms

After this commit I got

> yue -b test.yue
Parse time:     0.25387 ms
Compile time:   0.048875 ms

There may still be cases the parser will go in a needless pretty deep parsing stack. You may keep opening issues when you find more.

nm17 commented

@pigpigyyy While the commit did fix the issue of parsing closed blocks of {{{}}}, it still takes a lot of time parsing expressions like {{{ and (((((. Please make a check for this if you can, it will help me find bugs quicker.

Found it was a redundant PEG syntax back trace issue which happened in rule ChainValue. When the parser failed to check rule Callable >> ChainItems, it fell back to check rule Callable again, the parsing stack for Callable became doubled.
Tried to fix it in commit aed06d6.

nm17 commented

@pigpigyyy here are cases of hanging I found while fuzzing (This is is on the latest main commit 89b606d):

hangs.zip

Some of them are possibly false positives.

The nested syntax parser tree is leaving lots of fallback patterns to try, that causes the unclosed {{{{{{{{ expression taking a pretty long match to fail. Added a stoping rule for the error case syntax to prevent parser hanging. Sorry for forgetting testing the invalid codes you mentioned earlier.

nm17 commented

Now stuff like [f[f[f[f[f[[f[f[f[f[f[f[f[f[[f[f[f[f[f[f[f[f[f[f[ is making yue hang

nm17 commented

@pigpigyyy Here are some more hangs and crashes that I've been able to find using a fuzzer on the latest commit 32f2a57.

hangsandcrashes.zip

Tried fixing all similar issues by limiting nested expressions parsing depths to 100 levels.
The parser is written in a parser combinator which is spawning lots of recursive function call stacks.
Maybe a better way to fix is rewriting the recursive parser functions into a common loops and expand temporary memory usage into heap memory.