diff options
Diffstat (limited to 'src')
| -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, | 
