no usingnamespace, organize jsc namespace, enable -fincremental (#19122)

Co-authored-by: Dylan Conway <35280289+dylan-conway@users.noreply.github.com>
This commit is contained in:
chloe caruso
2025-04-22 16:34:15 -07:00
committed by GitHub
parent 842fe8664e
commit 3349c995b5
271 changed files with 30131 additions and 30868 deletions

View File

@@ -0,0 +1,325 @@
//! Process information and control APIs (`globalThis.process` / `node:process`)
comptime {
@export(&getTitle, .{ .name = "Bun__Process__getTitle" });
@export(&setTitle, .{ .name = "Bun__Process__setTitle" });
@export(&getArgv, .{ .name = "Bun__Process__getArgv" });
@export(&getCwd, .{ .name = "Bun__Process__getCwd" });
@export(&setCwd, .{ .name = "Bun__Process__setCwd" });
@export(&exit, .{ .name = "Bun__Process__exit" });
@export(&getArgv0, .{ .name = "Bun__Process__getArgv0" });
@export(&getExecPath, .{ .name = "Bun__Process__getExecPath" });
@export(&getExecArgv, .{ .name = "Bun__Process__getExecArgv" });
}
var title_mutex = bun.Mutex{};
pub fn getTitle(_: *JSGlobalObject, title: *ZigString) callconv(.C) void {
title_mutex.lock();
defer title_mutex.unlock();
const str = bun.CLI.Bun__Node__ProcessTitle;
title.* = ZigString.init(str orelse "bun");
}
// TODO: https://github.com/nodejs/node/blob/master/deps/uv/src/unix/darwin-proctitle.c
pub fn setTitle(globalObject: *JSGlobalObject, newvalue: *ZigString) callconv(.C) JSValue {
title_mutex.lock();
defer title_mutex.unlock();
if (bun.CLI.Bun__Node__ProcessTitle) |_| bun.default_allocator.free(bun.CLI.Bun__Node__ProcessTitle.?);
bun.CLI.Bun__Node__ProcessTitle = newvalue.dupe(bun.default_allocator) catch bun.outOfMemory();
return newvalue.toJS(globalObject);
}
pub fn getArgv0(globalObject: *JSC.JSGlobalObject) callconv(.C) JSC.JSValue {
return JSC.ZigString.fromUTF8(bun.argv[0]).toJS(globalObject);
}
pub fn getExecPath(globalObject: *JSC.JSGlobalObject) callconv(.C) JSC.JSValue {
const out = bun.selfExePath() catch {
// if for any reason we are unable to get the executable path, we just return argv[0]
return getArgv0(globalObject);
};
return JSC.ZigString.fromUTF8(out).toJS(globalObject);
}
pub fn getExecArgv(globalObject: *JSC.JSGlobalObject) callconv(.C) JSC.JSValue {
var sfb = std.heap.stackFallback(4096, globalObject.allocator());
const temp_alloc = sfb.get();
const vm = globalObject.bunVM();
if (vm.worker) |worker| {
// was explicitly overridden for the worker?
if (worker.execArgv) |execArgv| {
const array = JSC.JSValue.createEmptyArray(globalObject, execArgv.len);
for (0..execArgv.len) |i| {
array.putIndex(globalObject, @intCast(i), bun.String.init(execArgv[i]).toJS(globalObject));
}
return array;
}
}
var args = std.ArrayList(bun.String).initCapacity(temp_alloc, bun.argv.len - 1) catch bun.outOfMemory();
defer args.deinit();
defer for (args.items) |*arg| arg.deref();
var seen_run = false;
var prev: ?[]const u8 = null;
// we re-parse the process argv to extract execArgv, since this is a very uncommon operation
// it isn't worth doing this as a part of the CLI
for (bun.argv[@min(1, bun.argv.len)..]) |arg| {
defer prev = arg;
if (arg.len >= 1 and arg[0] == '-') {
args.append(bun.String.createUTF8(arg)) catch bun.outOfMemory();
continue;
}
if (!seen_run and bun.strings.eqlComptime(arg, "run")) {
seen_run = true;
continue;
}
// A set of execArgv args consume an extra argument, so we do not want to
// confuse these with script names.
const map = bun.ComptimeStringMap(void, comptime brk: {
const auto_params = bun.CLI.Arguments.auto_params;
const KV = struct { []const u8, void };
var entries: [auto_params.len]KV = undefined;
var i = 0;
for (auto_params) |param| {
if (param.takes_value != .none) {
if (param.names.long) |name| {
entries[i] = .{ "--" ++ name, {} };
i += 1;
}
if (param.names.short) |name| {
entries[i] = .{ &[_]u8{ '-', name }, {} };
i += 1;
}
}
}
var result: [i]KV = undefined;
@memcpy(&result, entries[0..i]);
break :brk result;
});
if (prev) |p| if (map.has(p)) {
args.append(bun.String.createUTF8(arg)) catch @panic("OOM");
continue;
};
// we hit the script name
break;
}
return bun.String.toJSArray(globalObject, args.items);
}
pub fn getArgv(globalObject: *JSC.JSGlobalObject) callconv(.C) JSC.JSValue {
const vm = globalObject.bunVM();
// Allocate up to 32 strings in stack
var stack_fallback_allocator = std.heap.stackFallback(
32 * @sizeOf(JSC.ZigString) + (bun.MAX_PATH_BYTES + 1) + 32,
bun.default_allocator,
);
const allocator = stack_fallback_allocator.get();
var args_count: usize = vm.argv.len;
if (vm.worker) |worker| {
args_count = worker.argv.len;
}
const args = allocator.alloc(
bun.String,
// argv omits "bun" because it could be "bun run" or "bun" and it's kind of ambiguous
// argv also omits the script name
args_count + 2,
) catch bun.outOfMemory();
var args_list: std.ArrayListUnmanaged(bun.String) = .initBuffer(args);
if (vm.standalone_module_graph != null) {
// Don't break user's code because they did process.argv.slice(2)
// Even if they didn't type "bun", we still want to add it as argv[0]
args_list.appendAssumeCapacity(
bun.String.static("bun"),
);
} else {
const exe_path = bun.selfExePath() catch null;
args_list.appendAssumeCapacity(
if (exe_path) |str| bun.String.fromUTF8(str) else bun.String.static("bun"),
);
}
if (vm.main.len > 0 and
!strings.endsWithComptime(vm.main, bun.pathLiteral("/[eval]")) and
!strings.endsWithComptime(vm.main, bun.pathLiteral("/[stdin]")))
{
if (vm.worker != null and vm.worker.?.eval_mode) {
args_list.appendAssumeCapacity(bun.String.static("[worker eval]"));
} else {
args_list.appendAssumeCapacity(bun.String.fromUTF8(vm.main));
}
}
defer allocator.free(args);
if (vm.worker) |worker| {
for (worker.argv) |arg| {
args_list.appendAssumeCapacity(bun.String.init(arg));
}
} else {
for (vm.argv) |arg| {
const str = bun.String.fromUTF8(arg);
// https://github.com/yargs/yargs/blob/adb0d11e02c613af3d9427b3028cc192703a3869/lib/utils/process-argv.ts#L1
args_list.appendAssumeCapacity(str);
}
}
return bun.String.toJSArray(globalObject, args_list.items);
}
pub fn getCwd(globalObject: *JSC.JSGlobalObject) callconv(.C) JSC.JSValue {
return JSC.toJSHostValue(globalObject, getCwd_(globalObject));
}
fn getCwd_(globalObject: *JSC.JSGlobalObject) bun.JSError!JSC.JSValue {
var buf: bun.PathBuffer = undefined;
switch (bun.api.node.path.getCwd(&buf)) {
.result => |r| return JSC.ZigString.init(r).withEncoding().toJS(globalObject),
.err => |e| {
return globalObject.throwValue(e.toJSC(globalObject));
},
}
}
pub fn setCwd(globalObject: *JSC.JSGlobalObject, to: *JSC.ZigString) callconv(.C) JSC.JSValue {
return JSC.toJSHostValue(globalObject, setCwd_(globalObject, to));
}
fn setCwd_(globalObject: *JSC.JSGlobalObject, to: *JSC.ZigString) bun.JSError!JSC.JSValue {
if (to.len == 0) {
return globalObject.throwInvalidArguments("Expected path to be a non-empty string", .{});
}
const vm = globalObject.bunVM();
const fs = vm.transpiler.fs;
var buf: bun.PathBuffer = undefined;
const slice = to.sliceZBuf(&buf) catch return globalObject.throw("Invalid path", .{});
switch (Syscall.chdir(fs.top_level_dir, slice)) {
.result => {
// When we update the cwd from JS, we have to update the bundler's version as well
// However, this might be called many times in a row, so we use a pre-allocated buffer
// that way we don't have to worry about garbage collector
const into_cwd_buf = switch (bun.sys.getcwd(&buf)) {
.result => |r| r,
.err => |err| {
_ = Syscall.chdir(fs.top_level_dir, fs.top_level_dir);
return globalObject.throwValue(err.toJSC(globalObject));
},
};
@memcpy(fs.top_level_dir_buf[0..into_cwd_buf.len], into_cwd_buf);
fs.top_level_dir_buf[into_cwd_buf.len] = 0;
fs.top_level_dir = fs.top_level_dir_buf[0..into_cwd_buf.len :0];
const len = fs.top_level_dir.len;
// Ensure the path ends with a slash
if (fs.top_level_dir_buf[len - 1] != std.fs.path.sep) {
fs.top_level_dir_buf[len] = std.fs.path.sep;
fs.top_level_dir_buf[len + 1] = 0;
fs.top_level_dir = fs.top_level_dir_buf[0 .. len + 1 :0];
}
const withoutTrailingSlash = if (Environment.isWindows) strings.withoutTrailingSlashWindowsPath else strings.withoutTrailingSlash;
var str = bun.String.createUTF8(withoutTrailingSlash(fs.top_level_dir));
return str.transferToJS(globalObject);
},
.err => |e| {
return globalObject.throwValue(e.toJSC(globalObject));
},
}
}
// TODO(@190n) this may need to be noreturn
pub fn exit(globalObject: *JSC.JSGlobalObject, code: u8) callconv(.c) void {
var vm = globalObject.bunVM();
if (vm.worker) |worker| {
vm.exit_handler.exit_code = code;
worker.requestTerminate();
return;
}
vm.exit_handler.exit_code = code;
vm.onExit();
vm.globalExit();
}
// TODO: switch this to using *bun.wtf.String when it is added
pub fn Bun__Process__editWindowsEnvVar(k: bun.String, v: bun.String) callconv(.C) void {
comptime bun.assert(bun.Environment.isWindows);
if (k.tag == .Empty) return;
const wtf1 = k.value.WTFStringImpl;
var fixed_stack_allocator = std.heap.stackFallback(1025, bun.default_allocator);
const allocator = fixed_stack_allocator.get();
var buf1 = allocator.alloc(u16, k.utf16ByteLength() + 1) catch bun.outOfMemory();
defer allocator.free(buf1);
var buf2 = allocator.alloc(u16, v.utf16ByteLength() + 1) catch bun.outOfMemory();
defer allocator.free(buf2);
const len1: usize = switch (wtf1.is8Bit()) {
true => bun.strings.copyLatin1IntoUTF16([]u16, buf1, []const u8, wtf1.latin1Slice()).written,
false => b: {
@memcpy(buf1[0..wtf1.length()], wtf1.utf16Slice());
break :b wtf1.length();
},
};
buf1[len1] = 0;
const str2: ?[*:0]const u16 = if (v.tag != .Dead) str: {
if (v.tag == .Empty) break :str (&[_]u16{0})[0..0 :0];
const wtf2 = v.value.WTFStringImpl;
const len2: usize = switch (wtf2.is8Bit()) {
true => bun.strings.copyLatin1IntoUTF16([]u16, buf2, []const u8, wtf2.latin1Slice()).written,
false => b: {
@memcpy(buf2[0..wtf2.length()], wtf2.utf16Slice());
break :b wtf2.length();
},
};
buf2[len2] = 0;
break :str buf2[0..len2 :0].ptr;
} else null;
_ = bun.c.SetEnvironmentVariableW(buf1[0..len1 :0].ptr, str2);
}
comptime {
if (Environment.export_cpp_apis and Environment.isWindows) {
@export(&Bun__Process__editWindowsEnvVar, .{ .name = "Bun__Process__editWindowsEnvVar" });
}
}
pub export const Bun__version: [*:0]const u8 = "v" ++ bun.Global.package_json_version;
pub export const Bun__version_with_sha: [*:0]const u8 = "v" ++ bun.Global.package_json_version_with_sha;
pub export const Bun__versions_boringssl: [*:0]const u8 = bun.Global.versions.boringssl;
pub export const Bun__versions_libarchive: [*:0]const u8 = bun.Global.versions.libarchive;
pub export const Bun__versions_mimalloc: [*:0]const u8 = bun.Global.versions.mimalloc;
pub export const Bun__versions_picohttpparser: [*:0]const u8 = bun.Global.versions.picohttpparser;
pub export const Bun__versions_uws: [*:0]const u8 = bun.Environment.git_sha;
pub export const Bun__versions_webkit: [*:0]const u8 = bun.Global.versions.webkit;
pub export const Bun__versions_zig: [*:0]const u8 = bun.Global.versions.zig;
pub export const Bun__versions_zlib: [*:0]const u8 = bun.Global.versions.zlib;
pub export const Bun__versions_tinycc: [*:0]const u8 = bun.Global.versions.tinycc;
pub export const Bun__versions_lolhtml: [*:0]const u8 = bun.Global.versions.lolhtml;
pub export const Bun__versions_c_ares: [*:0]const u8 = bun.Global.versions.c_ares;
pub export const Bun__versions_libdeflate: [*:0]const u8 = bun.Global.versions.libdeflate;
pub export const Bun__versions_usockets: [*:0]const u8 = bun.Environment.git_sha;
pub export const Bun__version_sha: [*:0]const u8 = bun.Environment.git_sha;
pub export const Bun__versions_lshpack: [*:0]const u8 = bun.Global.versions.lshpack;
pub export const Bun__versions_zstd: [*:0]const u8 = bun.Global.versions.zstd;
const std = @import("std");
const Environment = bun.Environment;
const bun = @import("bun");
const JSC = bun.JSC;
const JSGlobalObject = JSC.JSGlobalObject;
const JSValue = JSC.JSValue;
const ZigString = JSC.ZigString;
const Syscall = bun.sys;
const strings = bun.strings;