Bug 233206
Summary: | [JSC] indirect eval GC/memory leak | ||
---|---|---|---|
Product: | WebKit | Reporter: | Phillip Mates <pmates> |
Component: | JavaScriptCore | Assignee: | Nobody <webkit-unassigned> |
Status: | NEW | ||
Severity: | Normal | CC: | fpizlo, joseph.j.griego, saam, webkit-bug-importer, ysuzuki |
Priority: | P2 | Keywords: | InRadar |
Version: | WebKit Nightly Build | ||
Hardware: | Unspecified | ||
OS: | Unspecified |
Phillip Mates
While working on the Shadow Realm implementation in JSC I noticed some OOM issues related to indirect eval.
I was able to reproduce them with the following `example.js` snippet:
```
for (i = 0; i<50000; i++) {
// indirect eval: has issues
let f = (0, eval)("{x: " + Math.random() + "}");
// direct eval: is fine
// let f = eval("{x: " + Math.random() + "}");
// filler to prevent optimizing anything away
if (f.x === 0.1) {
print(1);
}
// issue persist even when GC is called after each allocation
fullGC();
}
```
running this snippet in the jsc console with llint disabled results in a out-of-memory crash
```
$ Tools/Scripts/run-jsc --jsc-only --debug --useLLInt=false --logExecutableAllocation=1 --gcMaxHeapSize=1024 --jitMemoryReservationSize=102400 examples.js
...
Allocating 896 bytes of executable memory with 101248 bytes allocated, 102400 bytes reserved, and 102400 committed.
Allocating 896 bytes of executable memory with 102144 bytes allocated, 102400 bytes reserved, and 102400 committed.
Ran out of executable memory while allocating 896 bytes.
[1] 26356 abort (core dumped) Tools/Scripts/run-jsc --jsc-only --debug --useLLInt=false
```
Attachments | ||
---|---|---|
Add attachment proposed patch, testcase, etc. |
Joseph Griego
(poked around a bit and confirmed that adding `--useCodeCache=false` prevents the crash as long as `jitMemoryReservationSize` is large enough to hold at least one instance of the eval'd code, still reading code though)
Phillip Mates
I change `prune` in runtime/CodeCache.h to just be `m_map.clear()` and it stops crashing. Given this, I believe that the pruning/flushing to disk logic that clears the code cache map, and thus allows for releasing otherwise gc'able executable objects, isn't sensitive to GC/memory needs.
Not sure what the best way forward would be. Some random ideas:
- use some sort of weak hashmap that allowed for gc'ing objects even if they appear in the code cache map.
- couple code cache map with the executable allocator and allow the allocator to clear the map.
- tweak the code cache heuristics, such as the workingSetMaxBytes parameter
Radar WebKit Bug Importer
<rdar://problem/85709734>