diff options
Diffstat (limited to '')
-rw-r--r-- | src/bun.js/bindings/bindings.cpp | 14 | ||||
-rw-r--r-- | src/bun.js/event_loop.zig | 237 | ||||
-rw-r--r-- | src/bun.js/javascript.zig | 2 |
3 files changed, 145 insertions, 108 deletions
diff --git a/src/bun.js/bindings/bindings.cpp b/src/bun.js/bindings/bindings.cpp index fe778ce42..d3e718ba7 100644 --- a/src/bun.js/bindings/bindings.cpp +++ b/src/bun.js/bindings/bindings.cpp @@ -950,6 +950,20 @@ void WebCore__DOMURL__pathname_(WebCore__DOMURL* domURL, ZigString* arg1) *arg1 = Zig::toZigString(pathname); } +extern "C" JSC__JSValue ZigString__toJSONObject(const ZigString* strPtr, JSC::JSGlobalObject* globalObject) +{ + auto str = Zig::toString(*strPtr); + auto throwScope = DECLARE_THROW_SCOPE(globalObject->vm()); + auto scope = DECLARE_CATCH_SCOPE(globalObject->vm()); + JSValue result = JSONParseWithException(globalObject, str); + if (auto* exception = scope.exception()) { + scope.clearException(); + RELEASE_AND_RETURN(throwScope, JSC::JSValue::encode(exception->value())); + } + + RELEASE_AND_RETURN(throwScope, JSValue::encode(result)); +} + JSC__JSValue SystemError__toErrorInstance(const SystemError* arg0, JSC__JSGlobalObject* globalObject) { diff --git a/src/bun.js/event_loop.zig b/src/bun.js/event_loop.zig index 30cae84dd..7fe248dad 100644 --- a/src/bun.js/event_loop.zig +++ b/src/bun.js/event_loop.zig @@ -211,15 +211,8 @@ pub const ConcurrentTask = struct { const AsyncIO = @import("bun").AsyncIO; -pub const EventLoop = struct { - tasks: Queue = undefined, - concurrent_tasks: ConcurrentTask.Queue = ConcurrentTask.Queue{}, - global: *JSGlobalObject = undefined, - virtual_machine: *JSC.VirtualMachine = undefined, - waker: ?AsyncIO.Waker = null, - start_server_on_next_tick: bool = false, - defer_count: std.atomic.Atomic(usize) = std.atomic.Atomic(usize).init(0), - +// This type must be unique per JavaScript thread +pub const GarbageCollectionController = struct { gc_timer: *uws.Timer = undefined, gc_last_heap_size: usize = 0, gc_last_heap_size_on_repeating_timer: usize = 0, @@ -229,11 +222,135 @@ pub const EventLoop = struct { gc_timer_interval: i32 = 0, gc_repeating_timer_fast: bool = true, + pub fn init(this: *GarbageCollectionController, vm: *VirtualMachine) void { + var actual = vm.uws_event_loop.?; + this.gc_timer = uws.Timer.createFallthrough(actual, this); + this.gc_repeating_timer = uws.Timer.createFallthrough(actual, this); + + var gc_timer_interval: i32 = 1000; + if (vm.bundler.env.map.get("BUN_GC_TIMER_INTERVAL")) |timer| { + if (std.fmt.parseInt(i32, timer, 10)) |parsed| { + if (parsed > 0) { + gc_timer_interval = parsed; + } + } else |_| {} + } + this.gc_repeating_timer.set(this, onGCRepeatingTimer, gc_timer_interval, gc_timer_interval); + this.gc_timer_interval = gc_timer_interval; + } + + pub fn scheduleGCTimer(this: *GarbageCollectionController) void { + this.gc_timer_state = .scheduled; + this.gc_timer.set(this, onGCTimer, 16, 0); + } + + pub fn bunVM(this: *GarbageCollectionController) *VirtualMachine { + return @fieldParentPtr(VirtualMachine, "gc_controller", this); + } + + pub fn onGCTimer(timer: *uws.Timer) callconv(.C) void { + var this = timer.as(*GarbageCollectionController); + this.gc_timer_state = .run_on_next_tick; + } + + // We want to always run GC once in awhile + // But if you have a long-running instance of Bun, you don't want the + // program constantly using CPU doing GC for no reason + // + // So we have two settings for this GC timer: + // + // - Fast: GC runs every 1 second + // - Slow: GC runs every 30 seconds + // + // When the heap size is increasing, we always switch to fast mode + // When the heap size has been the same or less for 30 seconds, we switch to slow mode + pub fn updateGCRepeatTimer(this: *GarbageCollectionController, comptime setting: @Type(.EnumLiteral)) void { + if (setting == .fast and !this.gc_repeating_timer_fast) { + this.gc_repeating_timer_fast = true; + this.gc_repeating_timer.set(this, onGCRepeatingTimer, this.gc_timer_interval, this.gc_timer_interval); + this.heap_size_didnt_change_for_repeating_timer_ticks_count = 0; + } else if (setting == .slow and this.gc_repeating_timer_fast) { + this.gc_repeating_timer_fast = false; + this.gc_repeating_timer.set(this, onGCRepeatingTimer, 30_000, 30_000); + this.heap_size_didnt_change_for_repeating_timer_ticks_count = 0; + } + } + + pub fn onGCRepeatingTimer(timer: *uws.Timer) callconv(.C) void { + var this = timer.as(*GarbageCollectionController); + const prev_heap_size = this.gc_last_heap_size_on_repeating_timer; + this.performGC(); + this.gc_last_heap_size_on_repeating_timer = this.gc_last_heap_size; + if (prev_heap_size == this.gc_last_heap_size_on_repeating_timer) { + this.heap_size_didnt_change_for_repeating_timer_ticks_count +|= 1; + if (this.heap_size_didnt_change_for_repeating_timer_ticks_count >= 30) { + // make the timer interval longer + this.updateGCRepeatTimer(.slow); + } + } else { + this.heap_size_didnt_change_for_repeating_timer_ticks_count = 0; + this.updateGCRepeatTimer(.fast); + } + } + + pub fn processGCTimer(this: *GarbageCollectionController) void { + var vm = this.bunVM().global.vm(); + const this_heap_size = vm.blockBytesAllocated(); + const prev = this.gc_last_heap_size; + + switch (this.gc_timer_state) { + .run_on_next_tick => { + // When memory usage is not stable, run the GC more. + if (this_heap_size != prev) { + this.scheduleGCTimer(); + this.updateGCRepeatTimer(.fast); + } else { + this.gc_timer_state = .pending; + } + vm.collectAsync(); + this.gc_last_heap_size = this_heap_size; + }, + .pending => { + if (this_heap_size != prev) { + this.updateGCRepeatTimer(.fast); + + if (this_heap_size > prev * 2) { + this.performGC(); + } else { + this.scheduleGCTimer(); + } + } + }, + .scheduled => { + if (this_heap_size > prev * 2) { + this.updateGCRepeatTimer(.fast); + this.performGC(); + } + }, + } + } + + pub fn performGC(this: *GarbageCollectionController) void { + var vm = this.bunVM().global.vm(); + vm.collectAsync(); + this.gc_last_heap_size = vm.blockBytesAllocated(); + } + pub const GCTimerState = enum { pending, scheduled, run_on_next_tick, }; +}; + +pub const EventLoop = struct { + tasks: Queue = undefined, + concurrent_tasks: ConcurrentTask.Queue = ConcurrentTask.Queue{}, + global: *JSGlobalObject = undefined, + virtual_machine: *JSC.VirtualMachine = undefined, + waker: ?AsyncIO.Waker = null, + start_server_on_next_tick: bool = false, + defer_count: std.atomic.Atomic(usize) = std.atomic.Atomic(usize).init(0), pub const Queue = std.fifo.LinearFifo(Task, .Dynamic); @@ -350,46 +467,8 @@ pub const EventLoop = struct { } } - pub fn scheduleGCTimer(this: *EventLoop) void { - this.gc_timer_state = .scheduled; - this.gc_timer.set(this, onGCTimer, 16, 0); - } - pub fn processGCTimer(this: *EventLoop) void { - var vm = this.virtual_machine.global.vm(); - const this_heap_size = vm.blockBytesAllocated(); - const prev = this.gc_last_heap_size; - - switch (this.gc_timer_state) { - .run_on_next_tick => { - // When memory usage is not stable, run the GC more. - if (this_heap_size != prev) { - this.scheduleGCTimer(); - this.updateGCRepeatTimer(.fast); - } else { - this.gc_timer_state = .pending; - } - vm.collectAsync(); - this.gc_last_heap_size = this_heap_size; - }, - .pending => { - if (this_heap_size != prev) { - this.updateGCRepeatTimer(.fast); - - if (this_heap_size > prev * 2) { - this.performGC(); - } else { - this.scheduleGCTimer(); - } - } - }, - .scheduled => { - if (this_heap_size > prev * 2) { - this.updateGCRepeatTimer(.fast); - this.performGC(); - } - }, - } + this.virtual_machine.gc_controller.processGCTimer(); } // TODO: fix this technical debt @@ -481,73 +560,15 @@ pub const EventLoop = struct { if (this.virtual_machine.uws_event_loop == null) { var actual = uws.Loop.get().?; this.virtual_machine.uws_event_loop = actual; - this.gc_timer = uws.Timer.createFallthrough(actual, this); - this.gc_repeating_timer = uws.Timer.createFallthrough(actual, this); - - var gc_timer_interval: i32 = 1000; - if (this.virtual_machine.bundler.env.map.get("BUN_GC_TIMER_INTERVAL")) |timer| { - if (std.fmt.parseInt(i32, timer, 10)) |parsed| { - if (parsed > 0) { - gc_timer_interval = parsed; - } - } else |_| {} - } - this.gc_repeating_timer.set(this, onGCRepeatingTimer, gc_timer_interval, gc_timer_interval); - this.gc_timer_interval = gc_timer_interval; + this.virtual_machine.gc_controller.init(this.virtual_machine); // _ = actual.addPostHandler(*JSC.EventLoop, this, JSC.EventLoop.afterUSocketsTick); // _ = actual.addPreHandler(*JSC.VM, this.virtual_machine.global.vm(), JSC.VM.drainMicrotasks); } } - pub fn onGCTimer(timer: *uws.Timer) callconv(.C) void { - var this = timer.as(*EventLoop); - this.gc_timer_state = .run_on_next_tick; - } - - // We want to always run GC once in awhile - // But if you have a long-running instance of Bun, you don't want the - // program constantly using CPU doing GC for no reason - // - // So we have two settings for this GC timer: - // - // - Fast: GC runs every 1 second - // - Slow: GC runs every 30 seconds - // - // When the heap size is increasing, we always switch to fast mode - // When the heap size has been the same or less for 30 seconds, we switch to slow mode - pub fn updateGCRepeatTimer(this: *EventLoop, comptime setting: @Type(.EnumLiteral)) void { - if (setting == .fast and !this.gc_repeating_timer_fast) { - this.gc_repeating_timer_fast = true; - this.gc_repeating_timer.set(this, onGCRepeatingTimer, this.gc_timer_interval, this.gc_timer_interval); - this.heap_size_didnt_change_for_repeating_timer_ticks_count = 0; - } else if (setting == .slow and this.gc_repeating_timer_fast) { - this.gc_repeating_timer_fast = false; - this.gc_repeating_timer.set(this, onGCRepeatingTimer, 30_000, 30_000); - this.heap_size_didnt_change_for_repeating_timer_ticks_count = 0; - } - } - - pub fn onGCRepeatingTimer(timer: *uws.Timer) callconv(.C) void { - var this = timer.as(*EventLoop); - const prev_heap_size = this.gc_last_heap_size_on_repeating_timer; - this.performGC(); - this.gc_last_heap_size_on_repeating_timer = this.gc_last_heap_size; - if (prev_heap_size == this.gc_last_heap_size_on_repeating_timer) { - this.heap_size_didnt_change_for_repeating_timer_ticks_count +|= 1; - if (this.heap_size_didnt_change_for_repeating_timer_ticks_count >= 30) { - // make the timer interval longer - this.updateGCRepeatTimer(.slow); - } - } else { - this.heap_size_didnt_change_for_repeating_timer_ticks_count = 0; - this.updateGCRepeatTimer(.fast); - } - } - /// Asynchronously run the garbage collector and track how much memory is now allocated pub fn performGC(this: *EventLoop) void { - this.global.vm().collectAsync(); - this.gc_last_heap_size = this.global.vm().blockBytesAllocated(); + this.virtual_machine.gc_controller.performGC(); } pub fn enqueueTaskConcurrent(this: *EventLoop, task: *ConcurrentTask) void { diff --git a/src/bun.js/javascript.zig b/src/bun.js/javascript.zig index b46a69cda..312c052f1 100644 --- a/src/bun.js/javascript.zig +++ b/src/bun.js/javascript.zig @@ -433,6 +433,8 @@ pub const VirtualMachine = struct { modules: ModuleLoader.AsyncModule.Queue = .{}, aggressive_garbage_collection: GCLevel = GCLevel.none, + gc_controller: JSC.GarbageCollectionController = .{}, + pub const GCLevel = enum(u3) { none = 0, mild = 1, |