diff options
| author | 2022-09-14 18:23:22 -0700 | |
|---|---|---|
| committer | 2022-09-14 18:23:22 -0700 | |
| commit | a291c1676f9351a03ae1092993d762c3147cff49 (patch) | |
| tree | f9f32abc2145d2896ad2bb4595110609663badf0 /src/bun.js | |
| parent | 7bfa302b75c2450a872dc6b5de0002a9c7959ea9 (diff) | |
| download | bun-a291c1676f9351a03ae1092993d762c3147cff49.tar.gz bun-a291c1676f9351a03ae1092993d762c3147cff49.tar.zst bun-a291c1676f9351a03ae1092993d762c3147cff49.zip | |
5x faster crypto.randomValues()
Diffstat (limited to 'src/bun.js')
| -rw-r--r-- | src/bun.js/rare_data.zig | 60 | ||||
| -rw-r--r-- | src/bun.js/uuid.zig | 24 | ||||
| -rw-r--r-- | src/bun.js/webcore.zig | 52 | 
3 files changed, 114 insertions, 22 deletions
| diff --git a/src/bun.js/rare_data.zig b/src/bun.js/rare_data.zig index cfca61ce0..8237d99a9 100644 --- a/src/bun.js/rare_data.zig +++ b/src/bun.js/rare_data.zig @@ -7,6 +7,7 @@ const Syscall = @import("./node/syscall.zig");  const JSC = @import("javascript_core");  const std = @import("std");  const BoringSSL = @import("boringssl"); +const bun = @import("../global.zig");  const WebSocketClientMask = @import("../http/websocket_http_client.zig").Mask;  boring_ssl_engine: ?*BoringSSL.ENGINE = null, @@ -16,11 +17,70 @@ stdin_store: ?*Blob.Store = null,  stdout_store: ?*Blob.Store = null,  websocket_mask: WebSocketClientMask = WebSocketClientMask{}, +uuid_entropy_cache: ?*EntropyCache = null, +  // TODO: make this per JSGlobalObject instead of global  // This does not handle ShadowRealm correctly!  tail_cleanup_hook: ?*CleanupHook = null,  cleanup_hook: ?*CleanupHook = null, +pub fn nextUUID(this: *RareData) [16]u8 { +    if (this.uuid_entropy_cache == null) { +        this.uuid_entropy_cache = default_allocator.create(EntropyCache) catch unreachable; +        this.uuid_entropy_cache.?.init(); +    } + +    return this.uuid_entropy_cache.?.get(); +} + +pub fn entropySlice(this: *RareData, len: usize) []u8 { +    if (this.uuid_entropy_cache == null) { +        this.uuid_entropy_cache = default_allocator.create(EntropyCache) catch unreachable; +        this.uuid_entropy_cache.?.init(); +    } + +    return this.uuid_entropy_cache.?.slice(len); +} + +pub const EntropyCache = struct { +    pub const buffered_uuids_count = 16; +    pub const size = buffered_uuids_count * 128; + +    cache: [size]u8 = undefined, +    index: usize = 0, + +    pub fn init(instance: *EntropyCache) void { +        instance.fill(); +    } + +    pub fn fill(this: *EntropyCache) void { +        bun.rand(&this.cache); +        this.index = 0; +    } + +    pub fn slice(this: *EntropyCache, len: usize) []u8 { +        if (len > this.cache.len) { +            return &[_]u8{}; +        } + +        if (this.index + len > this.cache.len) { +            this.fill(); +        } +        const result = this.cache[this.index..][0..len]; +        this.index += len; +        return result; +    } + +    pub fn get(this: *EntropyCache) [16]u8 { +        if (this.index + 16 > this.cache.len) { +            this.fill(); +        } +        const result = this.cache[this.index..][0..16].*; +        this.index += 16; +        return result; +    } +}; +  pub const CleanupHook = struct {      next: ?*CleanupHook = null,      ctx: ?*anyopaque, diff --git a/src/bun.js/uuid.zig b/src/bun.js/uuid.zig index 740309396..fa59520c2 100644 --- a/src/bun.js/uuid.zig +++ b/src/bun.js/uuid.zig @@ -3,16 +3,17 @@ const std = @import("std");  const crypto = std.crypto;  const fmt = std.fmt;  const testing = std.testing; +const bun = @import("../global.zig");  pub const Error = error{InvalidUUID};  const UUID = @This(); -bytes: [16]u8, +bytes: [16]u8 = undefined,  pub fn init() UUID {      var uuid = UUID{ .bytes = undefined }; -    crypto.random.bytes(&uuid.bytes); +    bun.rand(&uuid.bytes);      // Version 4      uuid.bytes[6] = (uuid.bytes[6] & 0x0f) | 0x40;      // Variant 1 @@ -67,22 +68,29 @@ pub fn format(  ) !void {      _ = options; // currently unused -    if (layout.len != 0 and layout[0] != 's') +    if (comptime layout.len != 0 and layout[0] != 's')          @compileError("Unsupported format specifier for UUID type: '" ++ layout ++ "'."); -      var buf: [36]u8 = undefined; +    self.print(&buf); + +    try fmt.format(writer, "{s}", .{buf}); +} + +pub fn print( +    self: UUID, +    buf: *[36]u8, +) void {      const hex = "0123456789abcdef"; +    const bytes = self.bytes;      buf[8] = '-';      buf[13] = '-';      buf[18] = '-';      buf[23] = '-';      inline for (encoded_pos) |i, j| { -        buf[i + 0] = hex[self.bytes[j] >> 4]; -        buf[i + 1] = hex[self.bytes[j] & 0x0f]; +        buf[comptime i + 0] = hex[bytes[j] >> 4]; +        buf[comptime i + 1] = hex[bytes[j] & 0x0f];      } - -    try fmt.format(writer, "{s}", .{buf});  }  pub fn parse(buf: []const u8) Error!UUID { diff --git a/src/bun.js/webcore.zig b/src/bun.js/webcore.zig index ac5238dac..4d06003b5 100644 --- a/src/bun.js/webcore.zig +++ b/src/bun.js/webcore.zig @@ -358,7 +358,7 @@ pub const Prompt = struct {  pub const Crypto = struct {      const UUID = @import("./uuid.zig"); - +    const BoringSSL = @import("boringssl");      pub const Class = JSC.NewClass(void, .{ .name = "crypto" }, .{          .getRandomValues = JSC.DOMCall("Crypto", @This(), "getRandomValues", JSC.JSValue, JSC.DOMEffect.top),          .randomUUID = JSC.DOMCall("Crypto", @This(), "randomUUID", *JSC.JSString, JSC.DOMEffect.top), @@ -389,20 +389,40 @@ pub const Crypto = struct {              return JSC.JSValue.jsUndefined();          };          var slice = array_buffer.byteSlice(); -        if (slice.len > 0) -            std.crypto.random.bytes(slice); + +        switch (slice.len) { +            0 => {}, +            1...JSC.RareData.EntropyCache.size / 8 => { +                if (arguments.len > 1) { +                    bun.rand(slice); +                } else { +                    std.mem.copy(u8, slice, globalThis.bunVM().rareData().entropySlice(slice.len)); +                } +            }, +            else => { +                bun.rand(slice); +            }, +        }          return arguments[0];      }      pub fn getRandomValuesWithoutTypeChecks( -        _: *JSC.JSGlobalObject, +        globalThis: *JSC.JSGlobalObject,          _: *anyopaque,          array: *JSC.JSUint8Array,      ) callconv(.C) JSC.JSValue {          var slice = array.slice(); -        if (slice.len > 0) -            std.crypto.random.bytes(slice); +        switch (slice.len) { +            0 => {}, +            // 512 bytes or less we reuse from the same cache as UUID generation. +            1...JSC.RareData.EntropyCache.size / 8 => { +                std.mem.copy(u8, slice, globalThis.bunVM().rareData().entropySlice(slice.len)); +            }, +            else => { +                bun.rand(slice); +            }, +        }          return @intToEnum(JSC.JSValue, @bitCast(i64, @ptrToInt(array)));      } @@ -412,20 +432,24 @@ pub const Crypto = struct {          _: JSC.JSValue,          _: []const JSC.JSValue,      ) JSC.JSValue { -        var uuid = UUID.init(); -        var out: [128]u8 = undefined; -        var str = std.fmt.bufPrint(&out, "{s}", .{uuid}) catch unreachable; -        return JSC.ZigString.init(str).toValueGC(globalThis); +        var out: [36]u8 = undefined; +        const uuid: UUID = .{ +            .bytes = globalThis.bunVM().rareData().nextUUID(), +        }; +        uuid.print(&out); +        return JSC.ZigString.init(&out).toValueGC(globalThis);      }      pub fn randomUUIDWithoutTypeChecks(          globalThis: *JSC.JSGlobalObject,          _: *anyopaque,      ) callconv(.C) JSC.JSValue { -        var uuid = UUID.init(); -        var out: [128]u8 = undefined; -        var str = std.fmt.bufPrint(&out, "{s}", .{uuid}) catch unreachable; -        return JSC.ZigString.init(str).toValueGC(globalThis); +        var out: [36]u8 = undefined; +        const uuid: UUID = .{ +            .bytes = globalThis.bunVM().rareData().nextUUID(), +        }; +        uuid.print(&out); +        return JSC.ZigString.init(&out).toValueGC(globalThis);      }      pub fn call( | 
