From fe08386174c0bd2efa68646371de55808c581f66 Mon Sep 17 00:00:00 2001 From: Don Isaac Date: Mon, 20 Jan 2025 19:09:45 -0800 Subject: [PATCH] fix: un-ref'd source map causing memory leak --- src/bun.js/javascript.zig | 20 +++++++++++++++++--- src/sourcemap/CodeCoverage.zig | 1 + src/sourcemap/sourcemap.zig | 1 + 3 files changed, 19 insertions(+), 3 deletions(-) diff --git a/src/bun.js/javascript.zig b/src/bun.js/javascript.zig index 885af89e00bfc2..6054b96f3a3da3 100644 --- a/src/bun.js/javascript.zig +++ b/src/bun.js/javascript.zig @@ -305,6 +305,7 @@ pub const SavedSourceMap = struct { entry.value_ptr.* = value.ptr(); } + /// You must call `.map.deref()` on the returned url or you'll cause a memory leak. fn getWithContent( this: *SavedSourceMap, path: string, @@ -377,10 +378,12 @@ pub const SavedSourceMap = struct { } } + /// You must call `.deref()` the returned source map or you'll cause a leak pub fn get(this: *SavedSourceMap, path: string) ?*ParsedSourceMap { return this.getWithContent(path, .mappings_only).map; } + /// You must call `.source_map.deref()` on the result or you'll introduce a memory leak pub fn resolveMapping( this: *SavedSourceMap, path: []const u8, @@ -395,8 +398,10 @@ pub const SavedSourceMap = struct { const map = parse.map orelse return null; const mapping = parse.mapping orelse - SourceMap.Mapping.find(map.mappings, line, column) orelse - return null; + SourceMap.Mapping.find(map.mappings, line, column) orelse { + map.deref(); + return null; + }; return .{ .mapping = mapping, @@ -1887,7 +1892,12 @@ pub const VirtualMachine = struct { JSC.markBinding(@src()); const allocator = opts.allocator; VMHolder.vm = try allocator.create(VirtualMachine); + errdefer { + if (VMHolder.vm) |vm| allocator.destroy(vm); + VMHolder.vm = null; + } const console = try allocator.create(ConsoleObject); + errdefer allocator.destroy(console); console.* = ConsoleObject.init(Output.errorWriter(), Output.writer()); const log = opts.log.?; const transpiler = try Transpiler.init( @@ -4322,6 +4332,8 @@ pub const VirtualMachine = struct { writer.print("\n", .{}) catch {}; } + /// You must call `.source_map.deref()` on the returned lookup, otherwise + /// you'll cause a memory leak. pub fn resolveSourceMapping( this: *VirtualMachine, path: []const u8, @@ -4339,8 +4351,10 @@ pub const VirtualMachine = struct { this.source_mappings.putValue(path, SavedSourceMap.Value.init(map)) catch bun.outOfMemory(); - const mapping = SourceMap.Mapping.find(map.mappings, line, column) orelse + const mapping = SourceMap.Mapping.find(map.mappings, line, column) orelse { + map.deref(); return null; + }; return .{ .mapping = mapping, diff --git a/src/sourcemap/CodeCoverage.zig b/src/sourcemap/CodeCoverage.zig index eeb34112854d2a..45e916c5ff6bd1 100644 --- a/src/sourcemap/CodeCoverage.zig +++ b/src/sourcemap/CodeCoverage.zig @@ -424,6 +424,7 @@ pub const ByteRangeMapping = struct { var executable_lines: Bitset = Bitset{}; var lines_which_have_executed: Bitset = Bitset{}; const parsed_mappings_ = bun.JSC.VirtualMachine.get().source_mappings.get(source_url.slice()); + defer if (parsed_mappings_) |pm| pm.deref(); var line_hits = LinesHits{}; var functions = std.ArrayListUnmanaged(Block){}; diff --git a/src/sourcemap/sourcemap.zig b/src/sourcemap/sourcemap.zig index 41ddfd67f13d67..a7ed78a051c1ee 100644 --- a/src/sourcemap/sourcemap.zig +++ b/src/sourcemap/sourcemap.zig @@ -618,6 +618,7 @@ pub const ParseResult = union(enum) { success: ParsedSourceMap, }; +/// Reference counted. pub const ParsedSourceMap = struct { input_line_count: usize = 0, mappings: Mapping.List = .{},