mirror of
https://github.com/ziglang/zig.git
synced 2024-11-27 15:42:49 +00:00
d29871977f
We already have a LICENSE file that covers the Zig Standard Library. We no longer need to remind everyone that the license is MIT in every single file. Previously this was introduced to clarify the situation for a fork of Zig that made Zig's LICENSE file harder to find, and replaced it with their own license that required annual payments to their company. However that fork now appears to be dead. So there is no need to reinforce the copyright notice in every single file.
129 lines
3.3 KiB
Zig
129 lines
3.3 KiB
Zig
// Adler32 checksum.
|
|
//
|
|
// https://tools.ietf.org/html/rfc1950#section-9
|
|
// https://github.com/madler/zlib/blob/master/adler32.c
|
|
|
|
const std = @import("../std.zig");
|
|
const testing = std.testing;
|
|
|
|
pub const Adler32 = struct {
|
|
const base = 65521;
|
|
const nmax = 5552;
|
|
|
|
adler: u32,
|
|
|
|
pub fn init() Adler32 {
|
|
return Adler32{ .adler = 1 };
|
|
}
|
|
|
|
// This fast variant is taken from zlib. It reduces the required modulos and unrolls longer
|
|
// buffer inputs and should be much quicker.
|
|
pub fn update(self: *Adler32, input: []const u8) void {
|
|
var s1 = self.adler & 0xffff;
|
|
var s2 = (self.adler >> 16) & 0xffff;
|
|
|
|
if (input.len == 1) {
|
|
s1 +%= input[0];
|
|
if (s1 >= base) {
|
|
s1 -= base;
|
|
}
|
|
s2 +%= s1;
|
|
if (s2 >= base) {
|
|
s2 -= base;
|
|
}
|
|
} else if (input.len < 16) {
|
|
for (input) |b| {
|
|
s1 +%= b;
|
|
s2 +%= s1;
|
|
}
|
|
if (s1 >= base) {
|
|
s1 -= base;
|
|
}
|
|
|
|
s2 %= base;
|
|
} else {
|
|
const n = nmax / 16; // note: 16 | nmax
|
|
|
|
var i: usize = 0;
|
|
|
|
while (i + nmax <= input.len) {
|
|
var rounds: usize = 0;
|
|
while (rounds < n) : (rounds += 1) {
|
|
comptime var j: usize = 0;
|
|
inline while (j < 16) : (j += 1) {
|
|
s1 +%= input[i + j];
|
|
s2 +%= s1;
|
|
}
|
|
i += 16;
|
|
}
|
|
|
|
s1 %= base;
|
|
s2 %= base;
|
|
}
|
|
|
|
if (i < input.len) {
|
|
while (i + 16 <= input.len) : (i += 16) {
|
|
comptime var j: usize = 0;
|
|
inline while (j < 16) : (j += 1) {
|
|
s1 +%= input[i + j];
|
|
s2 +%= s1;
|
|
}
|
|
}
|
|
while (i < input.len) : (i += 1) {
|
|
s1 +%= input[i];
|
|
s2 +%= s1;
|
|
}
|
|
|
|
s1 %= base;
|
|
s2 %= base;
|
|
}
|
|
}
|
|
|
|
self.adler = s1 | (s2 << 16);
|
|
}
|
|
|
|
pub fn final(self: *Adler32) u32 {
|
|
return self.adler;
|
|
}
|
|
|
|
pub fn hash(input: []const u8) u32 {
|
|
var c = Adler32.init();
|
|
c.update(input);
|
|
return c.final();
|
|
}
|
|
};
|
|
|
|
test "adler32 sanity" {
|
|
try testing.expectEqual(@as(u32, 0x620062), Adler32.hash("a"));
|
|
try testing.expectEqual(@as(u32, 0xbc002ed), Adler32.hash("example"));
|
|
}
|
|
|
|
test "adler32 long" {
|
|
const long1 = [_]u8{1} ** 1024;
|
|
try testing.expectEqual(@as(u32, 0x06780401), Adler32.hash(long1[0..]));
|
|
|
|
const long2 = [_]u8{1} ** 1025;
|
|
try testing.expectEqual(@as(u32, 0x0a7a0402), Adler32.hash(long2[0..]));
|
|
}
|
|
|
|
test "adler32 very long" {
|
|
const long = [_]u8{1} ** 5553;
|
|
try testing.expectEqual(@as(u32, 0x707f15b2), Adler32.hash(long[0..]));
|
|
}
|
|
|
|
test "adler32 very long with variation" {
|
|
const long = comptime blk: {
|
|
@setEvalBranchQuota(7000);
|
|
var result: [6000]u8 = undefined;
|
|
|
|
var i: usize = 0;
|
|
while (i < result.len) : (i += 1) {
|
|
result[i] = @truncate(u8, i);
|
|
}
|
|
|
|
break :blk result;
|
|
};
|
|
|
|
try testing.expectEqual(@as(u32, 0x5af38d6e), std.hash.Adler32.hash(long[0..]));
|
|
}
|