mirror of
https://github.com/oven-sh/bun
synced 2026-02-09 18:38:55 +00:00
239 lines
7.3 KiB
Zig
239 lines
7.3 KiB
Zig
const BoringSSL = @import("boringssl");
|
|
const std = @import("std");
|
|
|
|
fn NewHasher(comptime digest_size: comptime_int, comptime ContextType: type, comptime Full: anytype, comptime Init: anytype, comptime Update: anytype, comptime Final: anytype) type {
|
|
return struct {
|
|
hasher: ContextType = undefined,
|
|
|
|
pub const Digest = [digest_size]u8;
|
|
pub const digest: comptime_int = digest_size;
|
|
|
|
pub fn init() @This() {
|
|
var this: @This() = .{
|
|
.hasher = undefined,
|
|
};
|
|
|
|
std.debug.assert(Init(&this.hasher) == 1);
|
|
return this;
|
|
}
|
|
|
|
pub fn hash(bytes: []const u8, out: *Digest) void {
|
|
_ = Full(bytes.ptr, bytes.len, out);
|
|
}
|
|
|
|
pub fn update(this: *@This(), data: []const u8) void {
|
|
std.debug.assert(Update(&this.hasher, data.ptr, data.len) == 1);
|
|
}
|
|
|
|
pub fn final(this: *@This(), out: *Digest) void {
|
|
std.debug.assert(Final(out, &this.hasher) == 1);
|
|
}
|
|
};
|
|
}
|
|
|
|
fn NewEVP(
|
|
comptime digest_size: comptime_int,
|
|
comptime MDName: []const u8,
|
|
) type {
|
|
return struct {
|
|
ctx: BoringSSL.EVP_MD_CTX,
|
|
|
|
pub const Digest = [digest_size]u8;
|
|
pub const digest: comptime_int = digest_size;
|
|
|
|
pub fn init() @This() {
|
|
const md = @call(.{}, @field(BoringSSL, MDName), .{});
|
|
var this: @This() = .{
|
|
.ctx = undefined,
|
|
};
|
|
|
|
BoringSSL.EVP_MD_CTX_init(&this.ctx);
|
|
|
|
std.debug.assert(BoringSSL.EVP_DigestInit(&this.ctx, md) == 1);
|
|
|
|
return this;
|
|
}
|
|
|
|
pub fn hash(bytes: []const u8, out: *Digest, engine: *BoringSSL.ENGINE) void {
|
|
const md = @call(.{}, @field(BoringSSL, MDName), .{});
|
|
|
|
std.debug.assert(BoringSSL.EVP_Digest(bytes.ptr, bytes.len, out, null, md, engine) == 1);
|
|
}
|
|
|
|
pub fn update(this: *@This(), data: []const u8) void {
|
|
std.debug.assert(BoringSSL.EVP_DigestUpdate(&this.ctx, data.ptr, data.len) == 1);
|
|
}
|
|
|
|
pub fn final(this: *@This(), out: *Digest) void {
|
|
std.debug.assert(BoringSSL.EVP_DigestFinal(&this.ctx, out, null) == 1);
|
|
}
|
|
};
|
|
}
|
|
pub const EVP = struct {
|
|
pub const SHA1 = NewEVP(std.crypto.hash.Sha1.digest_length, "EVP_sha1");
|
|
pub const MD5 = NewEVP(32, "EVP_md5");
|
|
pub const MD4 = NewEVP(32, "EVP_md4");
|
|
pub const SHA224 = NewEVP(28, "EVP_sha224");
|
|
pub const SHA512 = NewEVP(std.crypto.hash.sha2.Sha512.digest_length, "EVP_sha512");
|
|
pub const SHA384 = NewEVP(std.crypto.hash.sha2.Sha384.digest_length, "EVP_sha384");
|
|
pub const SHA256 = NewEVP(std.crypto.hash.sha2.Sha256.digest_length, "EVP_sha256");
|
|
pub const SHA512_256 = NewEVP(std.crypto.hash.sha2.Sha512256.digest_length, "EVP_sha512_256");
|
|
pub const MD5_SHA1 = NewEVP(std.crypto.hash.Sha1.digest_length, "EVP_md5_sha1");
|
|
};
|
|
|
|
pub const SHA1 = EVP.SHA1;
|
|
pub const MD5 = EVP.MD5;
|
|
pub const MD4 = EVP.MD4;
|
|
pub const SHA224 = EVP.SHA224;
|
|
pub const SHA512 = EVP.SHA512;
|
|
pub const SHA384 = EVP.SHA384;
|
|
pub const SHA256 = EVP.SHA256;
|
|
pub const SHA512_256 = EVP.SHA512_256;
|
|
pub const MD5_SHA1 = EVP.MD5_SHA1;
|
|
|
|
/// API that OpenSSL 3 deprecated
|
|
pub const Hashers = struct {
|
|
pub const SHA1 = NewHasher(
|
|
std.crypto.hash.Sha1.digest_length,
|
|
BoringSSL.SHA_CTX,
|
|
BoringSSL.SHA1,
|
|
BoringSSL.SHA1_Init,
|
|
BoringSSL.SHA1_Update,
|
|
BoringSSL.SHA1_Final,
|
|
);
|
|
|
|
pub const SHA512 = NewHasher(
|
|
std.crypto.hash.sha2.Sha512.digest_length,
|
|
BoringSSL.SHA512_CTX,
|
|
BoringSSL.SHA512,
|
|
BoringSSL.SHA512_Init,
|
|
BoringSSL.SHA512_Update,
|
|
BoringSSL.SHA512_Final,
|
|
);
|
|
|
|
pub const SHA384 = NewHasher(
|
|
std.crypto.hash.sha2.Sha384.digest_length,
|
|
BoringSSL.SHA512_CTX,
|
|
BoringSSL.SHA384,
|
|
BoringSSL.SHA384_Init,
|
|
BoringSSL.SHA384_Update,
|
|
BoringSSL.SHA384_Final,
|
|
);
|
|
|
|
pub const SHA256 = NewHasher(
|
|
std.crypto.hash.sha2.Sha256.digest_length,
|
|
BoringSSL.SHA256_CTX,
|
|
BoringSSL.SHA256,
|
|
BoringSSL.SHA256_Init,
|
|
BoringSSL.SHA256_Update,
|
|
BoringSSL.SHA256_Final,
|
|
);
|
|
|
|
pub const SHA512_256 = NewHasher(
|
|
std.crypto.hash.sha2.Sha512256.digest_length,
|
|
BoringSSL.SHA512_CTX,
|
|
BoringSSL.SHA512_256,
|
|
BoringSSL.SHA512_256_Init,
|
|
BoringSSL.SHA512_256_Update,
|
|
BoringSSL.SHA512_256_Final,
|
|
);
|
|
};
|
|
|
|
const boring = [_]type{
|
|
Hashers.SHA1,
|
|
Hashers.SHA512,
|
|
Hashers.SHA384,
|
|
Hashers.SHA256,
|
|
Hashers.SHA512_256,
|
|
};
|
|
|
|
const zig = [_]type{
|
|
std.crypto.hash.Sha1,
|
|
std.crypto.hash.sha2.Sha512,
|
|
std.crypto.hash.sha2.Sha384,
|
|
std.crypto.hash.sha2.Sha256,
|
|
std.crypto.hash.sha2.Sha512256,
|
|
};
|
|
|
|
const evp = [_]type{
|
|
EVP.SHA1,
|
|
EVP.SHA512,
|
|
EVP.SHA384,
|
|
EVP.SHA256,
|
|
EVP.SHA512_256,
|
|
};
|
|
|
|
const labels = [_][]const u8{
|
|
"SHA1",
|
|
"SHA512",
|
|
"SHA384",
|
|
"SHA256",
|
|
"SHA512_256",
|
|
};
|
|
pub fn main() anyerror!void {
|
|
var file = try std.fs.cwd().openFileZ(std.os.argv[std.os.argv.len - 1], .{});
|
|
var bytes = try file.readToEndAlloc(std.heap.c_allocator, std.math.maxInt(usize));
|
|
|
|
var engine = BoringSSL.ENGINE_new().?;
|
|
|
|
inline for (boring) |BoringHasher, i| {
|
|
const ZigHasher = zig[i];
|
|
std.debug.print(
|
|
comptime labels[i] ++ " - hashing {.3f}:\n",
|
|
.{std.fmt.fmtIntSizeBin(bytes.len)},
|
|
);
|
|
var digest1: BoringHasher.Digest = undefined;
|
|
var digest2: BoringHasher.Digest = undefined;
|
|
var digest3: BoringHasher.Digest = undefined;
|
|
var digest4: BoringHasher.Digest = undefined;
|
|
|
|
var clock1 = try std.time.Timer.start();
|
|
ZigHasher.hash(bytes, &digest1, .{});
|
|
const zig_time = clock1.read();
|
|
|
|
var clock2 = try std.time.Timer.start();
|
|
BoringHasher.hash(bytes, &digest2);
|
|
const boring_time = clock2.read();
|
|
|
|
var clock3 = try std.time.Timer.start();
|
|
evp[i].hash(bytes, &digest3, engine);
|
|
const evp_time = clock3.read();
|
|
|
|
var evp_in = evp[i].init();
|
|
var clock4 = try std.time.Timer.start();
|
|
evp_in.update(bytes);
|
|
evp_in.final(&digest4);
|
|
const evp_in_time = clock4.read();
|
|
|
|
std.debug.print(
|
|
" zig: {}\n",
|
|
.{std.fmt.fmtDuration(zig_time)},
|
|
);
|
|
std.debug.print(
|
|
" boring: {}\n",
|
|
.{std.fmt.fmtDuration(boring_time)},
|
|
);
|
|
std.debug.print(
|
|
" evp: {}\n",
|
|
.{std.fmt.fmtDuration(evp_time)},
|
|
);
|
|
std.debug.print(
|
|
" evp in: {}\n\n",
|
|
.{std.fmt.fmtDuration(evp_in_time)},
|
|
);
|
|
|
|
if (!std.mem.eql(u8, &digest3, &digest2)) {
|
|
@panic("\ndigests don't match! for " ++ labels[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
test "sha256" {
|
|
const value: []const u8 = "hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world! hello, world!";
|
|
var hash: SHA256.Digest = undefined;
|
|
var hash2: SHA256.Digest = undefined;
|
|
SHA256.hash(value, &hash);
|
|
std.crypto.hash.sha2.Sha256.hash(value, &hash2, .{});
|
|
try std.testing.expectEqual(hash, hash2);
|
|
}
|