网易首页 > 网易号 > 正文 申请入驻

密码学系列之:Argon2加密算法详解

0
分享至

简介

Argon2是一个密钥推导函数,在2015年7月被选为密码哈希大赛的冠军,它由卢森堡大学的Alex Biryukov、Daniel Dinu和Dmitry Khovratovich设计,Argon2的实现通常是以Creative Commons CC0许可(即公共领域)或Apache License 2.0发布,并提供了三个相关版本,分别是Argon2d,Argon2i和Argon2id。

本文将会讨论一下Argon2的原理和使用。

密钥推导函数key derivation function

在密码学中,密钥推导函数(KDF)是一种密码学哈希函数,它使用伪随机函数从一个秘密值(如主密钥、密码或口令)中推导出一个或多个密钥。KDF可用于将密钥拉伸成更长的密钥,或获得所需格式的密钥,例如将Diffie-Hellman密钥交换的结果转换为用于AES的对称密钥。

Password Hashing Competition

密码学虽然是研究密码的,但是其加密算法是越公开越好,只有公开才能去检视该算法的好坏,只有经过大家的彻底研究,才能够让该算法得以在业界使用和传播。

最出名的密码算法大赛肯定是由NIST在2001年为了指定标准的AES算法举办的大赛,该大赛的目的寻找最新的加密算法来替代老的DES算法。在这次大赛中,涌现了许多优秀的算法,包括CAST-256, CRYPTON, DEAL, DFC, E2, FROG, HPC, LOKI97, MAGENTA, MARS, RC6, Rijndael, SAFER+, Serpent, 和 Twofish等。最终Rijndael算法被选为最终的AES算法实现。

同样的PHC也是一个这样的算法比赛,和NIST举办的算法比赛不同的是,这是一个非官方的,由密码学家们组织的比赛。它是在由Jean-Philippe Aumasson于2012年秋季发起。

2013年第一季度,发布了征集意见书的通知,到2014年3月31日截止日期,共收到24份意见书。2014年12月,确定了9个入围名单。2015年7月,宣布Argon2为优胜者。

Argon2算法

Argon2 的设计很简单,旨在实现最高的内存填充率和对多个计算单元的有效利用,同时还能提供对 tradeoff attacks 的防御(通过利用处理器的缓存和内存)。

Argon2有三个变种。Argon2i、Argon2d和Argon2id。Argon2d速度更快,并且使用数据依赖的内存访问方式,这使得它对GPU破解攻击有很强的抵抗力,适合没有side-channel timing attacks威胁的应用(例如加密货币)。

Argon2i则使用数据无关的内存访问,这对于密码哈希和基于密码的密钥推导算法来说是首选,其特点是速度较慢,因为它在内存上运行了更多的处理逻辑,以防止 tradeoff attacks 。

Argon2id是Argon2i和Argon2d的混合体,采用数据依赖型和数据独立型内存访问相结合的方式,从而可以同时抵御side-channel timing attacks和GPU破解攻击的能力。

Argon2的输入参数

Argon2有两类输入参数,分别是primary inputs和secondary inputs。

primary inputs包括要加密的消息P和nonce S,分别代表password和salt。

P的长度是0到232-1字节,S的长度是8到232-1字节(如果是做密码hash,推荐16字节)。

之所以叫做primary inputs,是因为这两个参数是必须输入的。

剩下的参数叫做secondary inputs,他们包括:

  • 并行程度p,表示同时可以有多少独立的计算链同时运行,取值是1到224-1。

  • Tag长度 τ, 长度从4到232-1字节。‘

  • 内存大小 m, 单位是兆,值取 8p到232-1。

  • 迭代器的个数t,提升运行速度。取值1到232-1。

  • 版本号v,一个字节,取值0x13。

  • 安全值 K , 长度是0到232-1字节。

  • 附加数据 X,长度是0到232-1字节。

  • Argon2的类型,0代表Argon2d,1代表Argon2i,2代表Argon2id。

这些输入可以用下面的代码来表示:

Inputs:
password (P): Bytes (0..232-1) Password (or message) to be hashed
salt (S): Bytes (8..232-1) Salt (16 bytes recommended for password hashing)
parallelism (p): Number (1..224-1) Degree of parallelism (i.e. number of threads)
tagLength (T): Number (4..232-1) Desired number of returned bytes
memorySizeKB (m): Number (8p..232-1) Amount of memory (in kibibytes) to use
iterations (t): Number (1..232-1) Number of iterations to perform
version (v): Number (0x13) The current version is 0x13 (19 decimal)
key (K): Bytes (0..232-1) Optional key (Errata: PDF says 0..32 bytes, RFC says 0..232 bytes)
associatedData (X): Bytes (0..232-1) Optional arbitrary extra data
hashType (y): Number (0=Argon2d, 1=Argon2i, 2=Argon2id)
Output:
tag: Bytes (tagLength) The resulting generated bytes, tagLength bytes long
处理流程

我们先来看一下非并行的Argon2的算法流程:

非并行的Argon2是最简单的。

上图中G表示的是一个压缩函数,接收两个1024byte的输入,输出一个1024byte。

i表示的是执行的步数,上面的φ(i) 就是输入,取自内存空间。

作为一个memory-hard的算法,一个很重要的工作就是构建初始内存。接下来,我们看一下如何构建初始内存空间。

首先,我们需要构建 H0 ,这是一个 64-byte 的block值,通过H0,可以去构建更多的block。计算H0的公式如下:

H0 = H(p,τ,m,t,v,y,⟨P⟩,P,⟨S⟩,S,⟨K⟩,K,⟨X⟩,X)

它是前面我们提到的输入参数的H函数。H0的大小是64byte。

看下H0的代码生成:

Generate initial 64-byte block H0.
All the input parameters are concatenated and input as a source of additional entropy.
Errata: RFC says H0 is 64-bits; PDF says H0 is 64-bytes.
Errata: RFC says the Hash is H^, the PDF says it's ℋ (but doesn't document what ℋ is). It's actually Blake2b.
Variable length items are prepended with their length as 32-bit little-endian integers.
buffer ← parallelism ∥ tagLength ∥ memorySizeKB ∥ iterations ∥ version ∥ hashType
∥ Length(password) ∥ Password
∥ Length(salt) ∥ salt
∥ Length(key) ∥ key
∥ Length(associatedData) ∥ associatedData
H0 ← Blake2b(buffer, 64) //default hash size of Blake2b is 64-bytes

对于输入参数并行程度p来说,需要将内存分成一个内存矩阵B[i][j], 它是一个 p 行的矩阵。

计算矩阵B的值:

其中H′ 是一个基于H的变长hash算法。

我们给一下这个算法的实现:

Function Hash(message, digestSize)
Inputs:
message: Bytes (0..232-1) Message to be hashed
digestSize: Integer (1..232) Desired number of bytes to be returned
Output:
digest: Bytes (digestSize) The resulting generated bytes, digestSize bytes long

Hash is a variable-length hash function, built using Blake2b, capable of generating
digests up to 232 bytes.

If the requested digestSize is 64-bytes or lower, then we use Blake2b directly
if (digestSize <= 64) then
return Blake2b(digestSize ∥ message, digestSize) //concatenate 32-bit little endian digestSize with the message bytes

For desired hashes over 64-bytes (e.g. 1024 bytes for Argon2 blocks),
we use Blake2b to generate twice the number of needed 64-byte blocks,
and then only use 32-bytes from each block

Calculate the number of whole blocks (knowing we're only going to use 32-bytes from each)
r ← Ceil(digestSize/32)-1;

Generate r whole blocks.
Initial block is generated from message
V1 ← Blake2b(digestSize ∥ message, 64);
Subsequent blocks are generated from previous blocks
for i ← 2 to r do
Vi ← Blake2b(Vi-1, 64)
Generate the final (possibly partial) block
partialBytesNeeded ← digestSize – 32*r;
Vr+1 ← Blake2b(Vr, partialBytesNeeded)

Concatenate the first 32-bytes of each block Vi
(except the possibly partial last block, which we take the whole thing)
Let Ai represent the lower 32-bytes of block Vi
return A1 ∥ A2 ∥ ... ∥ Ar ∥ Vr+1

如果我们的迭代次数多于一次,也就是说t > 1, 我们这样计算下一次迭代的 B :

最终遍历T次之后,我们得到最终的B :

B_{\text {final }}=B^{T}[0][q-1] \oplus B^{T}[1][q-1] \oplus \cdots \oplus B^{T}[p-1][q-1]

最后得到输出:

\mathrm{Tag} \leftarrow H^{\prime}\left(B_{\text {final }}\right)

这段逻辑也可以用代码来表示:

Calculate number of 1 KB blocks by rounding down memorySizeKB to the nearest multiple of 4*parallelism kibibytes
blockCount ← Floor(memorySizeKB, 4*parallelism)

Allocate two-dimensional array of 1 KiB blocks (parallelism rows x columnCount columns)
columnCount ← blockCount / parallelism; //In the RFC, columnCount is referred to as q

Compute the first and second block (i.e. column zero and one ) of each lane (i.e. row)
for i ← 0 to parallelism-1 do for each row
Bi[0] ← Hash(H0 ∥ 0 ∥ i, 1024) //Generate a 1024-byte digest
Bi[1] ← Hash(H0 ∥ 1 ∥ i, 1024) //Generate a 1024-byte digest

Compute remaining columns of each lane
for i ← 0 to parallelism-1 do //for each row
for j ← 2 to columnCount-1 do //for each subsequent column
//i' and j' indexes depend if it's Argon2i, Argon2d, or Argon2id (See section 3.4)
i′, j′ ← GetBlockIndexes(i, j) //the GetBlockIndexes function is not defined
Bi[j] = G(Bi[j-1], Bi′[j′]) //the G hash function is not defined

Further passes when iterations > 1
for nIteration ← 2 to iterations do
for i ← 0 to parallelism-1 do for each row
for j ← 0 to columnCount-1 do //for each subsequent column
//i' and j' indexes depend if it's Argon2i, Argon2d, or Argon2id (See section 3.4)
i′, j′ ← GetBlockIndexes(i, j)
if j == 0 then
Bi[0] = Bi[0] xor G(Bi[columnCount-1], Bi′[j′])
else
Bi[j] = Bi[j] xor G(Bi[j-1], Bi′[j′])

Compute final block C as the XOR of the last column of each row
C ← B0[columnCount-1]
for i ← 1 to parallelism-1 do
C ← C xor Bi[columnCount-1]

Compute output tag
return Hash(C, tagLength)

本文已收录于 http://www.flydean.com/40-argon2/

特别声明:以上内容(如有图片或视频亦包括在内)为自媒体平台“网易号”用户上传并发布,本平台仅提供信息存储服务。

Notice: The content above (including the pictures and videos if any) is uploaded and posted by a user of NetEase Hao, which is a social media platform and only provides information storage services.

相关推荐
热点推荐
阿曼达·塞弗里德:我错过了漫威最赚钱的"烂片"

阿曼达·塞弗里德:我错过了漫威最赚钱的"烂片"

影视情报室
2026-04-18 10:09:58
“新疆棉”事件5年后,那个丑态百出的“反华妖女”,如今怎样了

“新疆棉”事件5年后,那个丑态百出的“反华妖女”,如今怎样了

博览历史
2025-09-10 20:25:07
放弃纳格尔斯曼!曼联 5 年前错过的他才是天选,比卡里克更强

放弃纳格尔斯曼!曼联 5 年前错过的他才是天选,比卡里克更强

奶盖熊本熊
2026-04-19 01:03:53
高志凯:拜登是美国最卑鄙、最无耻的总统!在位四年煽动多国战争

高志凯:拜登是美国最卑鄙、最无耻的总统!在位四年煽动多国战争

扶苏聊历史
2025-12-26 09:53:52
形势已然大变!西方媒体集体改口:中国,已无需再向世界证明什么

形势已然大变!西方媒体集体改口:中国,已无需再向世界证明什么

乐享人生风雨
2026-04-11 02:05:32
巴基斯坦空军进驻沙特,真实目的曝光,不是防伊朗,是怕有人搞鬼

巴基斯坦空军进驻沙特,真实目的曝光,不是防伊朗,是怕有人搞鬼

爱吃醋的猫咪
2026-04-15 21:20:06
上海市民傻眼:古董樟木箱里的银元金条金项链,都没了!立案后,一个人主动投案

上海市民傻眼:古董樟木箱里的银元金条金项链,都没了!立案后,一个人主动投案

新民晚报
2026-04-18 16:56:38
全国景区为啥都卖臭豆腐、打铁花?圈内人不说的潜规则!

全国景区为啥都卖臭豆腐、打铁花?圈内人不说的潜规则!

沈理职谈
2026-04-06 16:27:46
沙特带头,阿联酋紧跟,卡塔尔随后,中东开始大变局

沙特带头,阿联酋紧跟,卡塔尔随后,中东开始大变局

明天见灌装冰块
2026-04-18 13:24:18
18岁儿子撒娇要零花钱拽掉爸爸胳膊

18岁儿子撒娇要零花钱拽掉爸爸胳膊

观威海
2026-04-18 15:28:37
北京人均GDP,是全国平均值的2.4倍,也是世界平均值的2.4倍

北京人均GDP,是全国平均值的2.4倍,也是世界平均值的2.4倍

安安小小姐姐说城市
2026-04-18 06:40:07
立陶宛总统:中国若是还想跟立陶宛和好,必须对我们展现充分诚意

立陶宛总统:中国若是还想跟立陶宛和好,必须对我们展现充分诚意

老鹈爱说事
2026-04-19 01:46:04
男子抓住跳楼的女友,5分钟后力竭女友坠楼身亡,法院:男子疏于照看存在过错,已尽力施救,承担10%责任

男子抓住跳楼的女友,5分钟后力竭女友坠楼身亡,法院:男子疏于照看存在过错,已尽力施救,承担10%责任

鲁中晨报
2026-04-18 07:28:17
78岁连路都走不稳还开演唱会,全网骂声一片,她却扬言回馈粉丝

78岁连路都走不稳还开演唱会,全网骂声一片,她却扬言回馈粉丝

LULU生活家
2026-04-14 18:43:54
戴维斯:若特鲁姆普愿意他可以住我家,赵心童要能卫冕将令人惊讶

戴维斯:若特鲁姆普愿意他可以住我家,赵心童要能卫冕将令人惊讶

世界体坛观察家
2026-04-18 09:11:52
同曦告别季后赛,山东五连败濒临崩盘,队内矛盾爆发

同曦告别季后赛,山东五连败濒临崩盘,队内矛盾爆发

梦忆之浅
2026-04-19 01:19:38
3比1击败辽宁铁人,两位新外援有了化学反应,申花队攻击线让人眼前一亮

3比1击败辽宁铁人,两位新外援有了化学反应,申花队攻击线让人眼前一亮

文汇报
2026-04-19 04:08:06
一位71岁大爷哭诉:我退休金5800,过年连一套新衣服都没钱买

一位71岁大爷哭诉:我退休金5800,过年连一套新衣服都没钱买

烙任情感
2026-04-16 17:48:57
查获超15吨!东莞这些企业被立案调查!

查获超15吨!东莞这些企业被立案调查!

东莞纪实
2026-04-18 20:38:47
中纪委放了话:宁可掉层皮,也要抓出群众满意成效!

中纪委放了话:宁可掉层皮,也要抓出群众满意成效!

林子说事
2026-04-19 02:00:33
2026-04-19 05:31:00
flydean程序那些事
flydean程序那些事
最通俗的解读,最深刻的干货!
356文章数 438关注度
往期回顾 全部

科技要闻

传Meta下月拟裁8000 大举清退人力为AI腾位

头条要闻

伊朗革命卫队向油轮开火 伊朗最高领袖发声

头条要闻

伊朗革命卫队向油轮开火 伊朗最高领袖发声

体育要闻

时隔25年重返英超!没有人再嘲笑他了

娱乐要闻

刘德华回应潘宏彬去世,拒谈丧礼细节

财经要闻

"影子万科"2.0:管理层如何吸血万物云?

汽车要闻

奇瑞威麟R08 PRO正式上市 售价14.48万元起

态度原创

数码
亲子
游戏
本地
公开课

数码要闻

华为版的科技春晚来了!Pura 90/Pura X Max下周发:阵容豪华

亲子要闻

退烧药怎么用?90%家长都搞错了

让老粥批直呼“计划有变”的岁兽代理人,到底是什么东西?

本地新闻

12吨巧克力有难,全网化身超级侦探添乱

公开课

李玫瑾:为什么性格比能力更重要?

无障碍浏览 进入关怀版