mirror of
https://github.com/ziglang/zig.git
synced 2025-12-06 06:13:07 +00:00
crypto: add AES-CMAC (RFC 4493) (#14545)
* crypto: add AES-CMAC Co-authored-by: Frank Denis <124872+jedisct1@users.noreply.github.com>
This commit is contained in:
parent
7199d7c777
commit
438b71155a
@ -41,6 +41,7 @@ pub const auth = struct {
|
|||||||
pub const Aegis128LMac = @import("crypto/aegis.zig").Aegis128LMac;
|
pub const Aegis128LMac = @import("crypto/aegis.zig").Aegis128LMac;
|
||||||
pub const Aegis256Mac = @import("crypto/aegis.zig").Aegis256Mac;
|
pub const Aegis256Mac = @import("crypto/aegis.zig").Aegis256Mac;
|
||||||
};
|
};
|
||||||
|
pub const cmac = @import("crypto/cmac.zig");
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Core functions, that should rarely be used directly by applications.
|
/// Core functions, that should rarely be used directly by applications.
|
||||||
@ -203,6 +204,7 @@ test {
|
|||||||
_ = aead.salsa_poly.XSalsa20Poly1305;
|
_ = aead.salsa_poly.XSalsa20Poly1305;
|
||||||
|
|
||||||
_ = auth.hmac;
|
_ = auth.hmac;
|
||||||
|
_ = auth.cmac;
|
||||||
_ = auth.siphash;
|
_ = auth.siphash;
|
||||||
|
|
||||||
_ = core.aes;
|
_ = core.aes;
|
||||||
|
|||||||
@ -66,6 +66,7 @@ const macs = [_]Crypto{
|
|||||||
Crypto{ .ty = crypto.auth.siphash.SipHash128(1, 3), .name = "siphash128-1-3" },
|
Crypto{ .ty = crypto.auth.siphash.SipHash128(1, 3), .name = "siphash128-1-3" },
|
||||||
Crypto{ .ty = crypto.auth.aegis.Aegis128LMac, .name = "aegis-128l mac" },
|
Crypto{ .ty = crypto.auth.aegis.Aegis128LMac, .name = "aegis-128l mac" },
|
||||||
Crypto{ .ty = crypto.auth.aegis.Aegis256Mac, .name = "aegis-256 mac" },
|
Crypto{ .ty = crypto.auth.aegis.Aegis256Mac, .name = "aegis-256 mac" },
|
||||||
|
Crypto{ .ty = crypto.auth.cmac.CmacAes128, .name = "aes-cmac" },
|
||||||
};
|
};
|
||||||
|
|
||||||
pub fn benchmarkMac(comptime Mac: anytype, comptime bytes: comptime_int) !u64 {
|
pub fn benchmarkMac(comptime Mac: anytype, comptime bytes: comptime_int) !u64 {
|
||||||
|
|||||||
156
lib/std/crypto/cmac.zig
Normal file
156
lib/std/crypto/cmac.zig
Normal file
@ -0,0 +1,156 @@
|
|||||||
|
const std = @import("std");
|
||||||
|
const crypto = std.crypto;
|
||||||
|
const mem = std.mem;
|
||||||
|
|
||||||
|
/// CMAC with AES-128 - RFC 4493 https://www.rfc-editor.org/rfc/rfc4493
|
||||||
|
pub const CmacAes128 = Cmac(crypto.core.aes.Aes128);
|
||||||
|
|
||||||
|
/// NIST Special Publication 800-38B - The CMAC Mode for Authentication
|
||||||
|
/// https://nvlpubs.nist.gov/nistpubs/SpecialPublications/NIST.SP.800-38b.pdf
|
||||||
|
pub fn Cmac(comptime BlockCipher: type) type {
|
||||||
|
const BlockCipherCtx = @typeInfo(@TypeOf(BlockCipher.initEnc)).Fn.return_type.?;
|
||||||
|
const Block = [BlockCipher.block.block_length]u8;
|
||||||
|
|
||||||
|
return struct {
|
||||||
|
const Self = @This();
|
||||||
|
pub const key_length = BlockCipher.key_bits / 8;
|
||||||
|
pub const block_length = BlockCipher.block.block_length;
|
||||||
|
pub const mac_length = block_length;
|
||||||
|
|
||||||
|
cipher_ctx: BlockCipherCtx,
|
||||||
|
k1: Block,
|
||||||
|
k2: Block,
|
||||||
|
buf: Block = [_]u8{0} ** block_length,
|
||||||
|
pos: usize = 0,
|
||||||
|
|
||||||
|
pub fn create(out: *[mac_length]u8, msg: []const u8, key: *const [key_length]u8) void {
|
||||||
|
var ctx = Self.init(key);
|
||||||
|
ctx.update(msg);
|
||||||
|
ctx.final(out);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn init(key: *const [key_length]u8) Self {
|
||||||
|
const cipher_ctx = BlockCipher.initEnc(key.*);
|
||||||
|
const zeros = [_]u8{0} ** block_length;
|
||||||
|
var k1: Block = undefined;
|
||||||
|
cipher_ctx.encrypt(&k1, &zeros);
|
||||||
|
k1 = double(k1);
|
||||||
|
return Self{
|
||||||
|
.cipher_ctx = cipher_ctx,
|
||||||
|
.k1 = k1,
|
||||||
|
.k2 = double(k1),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn update(self: *Self, msg: []const u8) void {
|
||||||
|
const left = block_length - self.pos;
|
||||||
|
var m = msg;
|
||||||
|
if (m.len > left) {
|
||||||
|
for (self.buf[self.pos..]) |*b, i| b.* ^= m[i];
|
||||||
|
m = m[left..];
|
||||||
|
self.cipher_ctx.encrypt(&self.buf, &self.buf);
|
||||||
|
self.pos = 0;
|
||||||
|
}
|
||||||
|
while (m.len > block_length) {
|
||||||
|
for (self.buf[0..block_length]) |*b, i| b.* ^= m[i];
|
||||||
|
m = m[block_length..];
|
||||||
|
self.cipher_ctx.encrypt(&self.buf, &self.buf);
|
||||||
|
self.pos = 0;
|
||||||
|
}
|
||||||
|
if (m.len > 0) {
|
||||||
|
for (self.buf[self.pos..][0..m.len]) |*b, i| b.* ^= m[i];
|
||||||
|
self.pos += m.len;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn final(self: *Self, out: *[mac_length]u8) void {
|
||||||
|
var mac = self.k1;
|
||||||
|
if (self.pos < block_length) {
|
||||||
|
mac = self.k2;
|
||||||
|
mac[self.pos] ^= 0x80;
|
||||||
|
}
|
||||||
|
for (mac) |*b, i| b.* ^= self.buf[i];
|
||||||
|
self.cipher_ctx.encrypt(out, &mac);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn double(l: Block) Block {
|
||||||
|
const Int = std.meta.Int(.unsigned, block_length * 8);
|
||||||
|
const l_ = mem.readIntBig(Int, &l);
|
||||||
|
const l_2 = switch (block_length) {
|
||||||
|
8 => (l_ << 1) ^ (0x1b & -%(l_ >> 63)), // mod x^64 + x^4 + x^3 + x + 1
|
||||||
|
16 => (l_ << 1) ^ (0x87 & -%(l_ >> 127)), // mod x^128 + x^7 + x^2 + x + 1
|
||||||
|
32 => (l_ << 1) ^ (0x0425 & -%(l_ >> 255)), // mod x^256 + x^10 + x^5 + x^2 + 1
|
||||||
|
64 => (l_ << 1) ^ (0x0125 & -%(l_ >> 511)), // mod x^512 + x^8 + x^5 + x^2 + 1
|
||||||
|
else => @compileError("unsupported block length"),
|
||||||
|
};
|
||||||
|
var l2: Block = undefined;
|
||||||
|
mem.writeIntBig(Int, &l2, l_2);
|
||||||
|
return l2;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const testing = std.testing;
|
||||||
|
|
||||||
|
test "CmacAes128 - Example 1: len = 0" {
|
||||||
|
const key = [_]u8{
|
||||||
|
0x2b, 0x7e, 0x15, 0x16, 0x28, 0xae, 0xd2, 0xa6, 0xab, 0xf7, 0x15, 0x88, 0x09, 0xcf, 0x4f, 0x3c,
|
||||||
|
};
|
||||||
|
var msg: [0]u8 = undefined;
|
||||||
|
const exp = [_]u8{
|
||||||
|
0xbb, 0x1d, 0x69, 0x29, 0xe9, 0x59, 0x37, 0x28, 0x7f, 0xa3, 0x7d, 0x12, 0x9b, 0x75, 0x67, 0x46,
|
||||||
|
};
|
||||||
|
var out: [CmacAes128.mac_length]u8 = undefined;
|
||||||
|
CmacAes128.create(&out, &msg, &key);
|
||||||
|
try testing.expectEqualSlices(u8, &out, &exp);
|
||||||
|
}
|
||||||
|
|
||||||
|
test "CmacAes128 - Example 2: len = 16" {
|
||||||
|
const key = [_]u8{
|
||||||
|
0x2b, 0x7e, 0x15, 0x16, 0x28, 0xae, 0xd2, 0xa6, 0xab, 0xf7, 0x15, 0x88, 0x09, 0xcf, 0x4f, 0x3c,
|
||||||
|
};
|
||||||
|
const msg = [_]u8{
|
||||||
|
0x6b, 0xc1, 0xbe, 0xe2, 0x2e, 0x40, 0x9f, 0x96, 0xe9, 0x3d, 0x7e, 0x11, 0x73, 0x93, 0x17, 0x2a,
|
||||||
|
};
|
||||||
|
const exp = [_]u8{
|
||||||
|
0x07, 0x0a, 0x16, 0xb4, 0x6b, 0x4d, 0x41, 0x44, 0xf7, 0x9b, 0xdd, 0x9d, 0xd0, 0x4a, 0x28, 0x7c,
|
||||||
|
};
|
||||||
|
var out: [CmacAes128.mac_length]u8 = undefined;
|
||||||
|
CmacAes128.create(&out, &msg, &key);
|
||||||
|
try testing.expectEqualSlices(u8, &out, &exp);
|
||||||
|
}
|
||||||
|
|
||||||
|
test "CmacAes128 - Example 3: len = 40" {
|
||||||
|
const key = [_]u8{
|
||||||
|
0x2b, 0x7e, 0x15, 0x16, 0x28, 0xae, 0xd2, 0xa6, 0xab, 0xf7, 0x15, 0x88, 0x09, 0xcf, 0x4f, 0x3c,
|
||||||
|
};
|
||||||
|
const msg = [_]u8{
|
||||||
|
0x6b, 0xc1, 0xbe, 0xe2, 0x2e, 0x40, 0x9f, 0x96, 0xe9, 0x3d, 0x7e, 0x11, 0x73, 0x93, 0x17, 0x2a,
|
||||||
|
0xae, 0x2d, 0x8a, 0x57, 0x1e, 0x03, 0xac, 0x9c, 0x9e, 0xb7, 0x6f, 0xac, 0x45, 0xaf, 0x8e, 0x51,
|
||||||
|
0x30, 0xc8, 0x1c, 0x46, 0xa3, 0x5c, 0xe4, 0x11,
|
||||||
|
};
|
||||||
|
const exp = [_]u8{
|
||||||
|
0xdf, 0xa6, 0x67, 0x47, 0xde, 0x9a, 0xe6, 0x30, 0x30, 0xca, 0x32, 0x61, 0x14, 0x97, 0xc8, 0x27,
|
||||||
|
};
|
||||||
|
var out: [CmacAes128.mac_length]u8 = undefined;
|
||||||
|
CmacAes128.create(&out, &msg, &key);
|
||||||
|
try testing.expectEqualSlices(u8, &out, &exp);
|
||||||
|
}
|
||||||
|
|
||||||
|
test "CmacAes128 - Example 4: len = 64" {
|
||||||
|
const key = [_]u8{
|
||||||
|
0x2b, 0x7e, 0x15, 0x16, 0x28, 0xae, 0xd2, 0xa6, 0xab, 0xf7, 0x15, 0x88, 0x09, 0xcf, 0x4f, 0x3c,
|
||||||
|
};
|
||||||
|
const msg = [_]u8{
|
||||||
|
0x6b, 0xc1, 0xbe, 0xe2, 0x2e, 0x40, 0x9f, 0x96, 0xe9, 0x3d, 0x7e, 0x11, 0x73, 0x93, 0x17, 0x2a,
|
||||||
|
0xae, 0x2d, 0x8a, 0x57, 0x1e, 0x03, 0xac, 0x9c, 0x9e, 0xb7, 0x6f, 0xac, 0x45, 0xaf, 0x8e, 0x51,
|
||||||
|
0x30, 0xc8, 0x1c, 0x46, 0xa3, 0x5c, 0xe4, 0x11, 0xe5, 0xfb, 0xc1, 0x19, 0x1a, 0x0a, 0x52, 0xef,
|
||||||
|
0xf6, 0x9f, 0x24, 0x45, 0xdf, 0x4f, 0x9b, 0x17, 0xad, 0x2b, 0x41, 0x7b, 0xe6, 0x6c, 0x37, 0x10,
|
||||||
|
};
|
||||||
|
const exp = [_]u8{
|
||||||
|
0x51, 0xf0, 0xbe, 0xbf, 0x7e, 0x3b, 0x9d, 0x92, 0xfc, 0x49, 0x74, 0x17, 0x79, 0x36, 0x3c, 0xfe,
|
||||||
|
};
|
||||||
|
var out: [CmacAes128.mac_length]u8 = undefined;
|
||||||
|
CmacAes128.create(&out, &msg, &key);
|
||||||
|
try testing.expectEqualSlices(u8, &out, &exp);
|
||||||
|
}
|
||||||
Loading…
x
Reference in New Issue
Block a user