-
Notifications
You must be signed in to change notification settings - Fork 28
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
10% of parse time is tracking line numbers #107
Comments
Some numbers:
So yeah, it would be quite a bit of memory used just for line numbers... |
I was about to say that's a convincing argument for not keeping the files in memory, but I guess if we mmap'd the files the memory use of keeping them around is "free", in that the memory backing of the mmap could be paged out by the OS under memory pressure... I think(?) (I had thought that we only used line numbers in error messages, which would mean we'd only ever need to reread a file once. But looking now we also use them in Just to write it down, the full LLVM trick is: map each input file to an integer (e.g. put them in a Vec), and then pack that integer and the file offset together. So you only need to store a single integer per thing you want to annotate with a file location. This matters more for LLVM because I think they store the file location of every AST node (for compiler diagnostics etc.) In n2's case we only store one of these per build statement. Currently it's a not especially efficient Rc+usize so 16 bytes per Build in memory. For a big Android build that's probably only still costing a few megabytes so probably not worth worrying about too much. |
Note to self, to find all the places we use line numbers, the trick is to comment out the |
After #112, I tried disabling the line counting and wasn't able to observe a noticeable speedup. |
We mark each
build
with its line number for use in error messages, but computing the line number requires keeping track of each newline we encounter.Some experiments:
build
to store the filename (error messages look likefoo.ninja:123: some error
), but that appeared to be ~no impact on perf -- I imagine it's just a trivial increment of a non-atomic refcount for each build statementA trick I learned from LLVM is to instead annotate each
build
with its file byte offset when parsing, which is faster to gather. Then if we encounter an error we can spend the time to re-parse the file for newlines to find the offset. Parsing is slow but it's like milliseconds slow, it's fine to do in an error message codepath.Unfortunately doing this would mean we need to either keep around the build file text at runtime, or re-read the file when generating an error message.
If we kept the file text at runtime:
-there's a couple places where we could avoid allocating a copy of some strings found within the file, though I think those are pretty minor.
If we re-read the file when generating error messages:
The text was updated successfully, but these errors were encountered: