Attack on RSA with Low Public Exponent
作者简介:Xor0v0,硕士在读,零知识证明小白,目前在做一些circom开发和zk审计,密码学爱好者,打过一些web2/3 CTF,最近对zkHACK产生兴趣。欢迎各位大佬一起交流学习。
RSA 公钥加密系统自1977年被提出至今,目前仍然在很多领域被广泛使用,许多研究人员致力于找到其安全漏洞来攻克它。
在此期间,很多高明的攻击方法被提出。但至少在目前(2024初),在保证选取足够安全的加密参数情况下,RSA的加密强度仍然是现代电脑无法攻破的。
本小节,我们介绍其中一种高明的攻击方法,它源于 Hastad ,后由 Coppersmith 改良,当 RSA 使用一个小公共指数 时,攻击容易被实现。攻击思路基于找到低次多项式的一个小的根的算法,这个算法又需要使用 LLL 算法。这种寻找根的算法本身非常有意思,也用于其他攻击 RSA 系统的算法中。
1. RSA Review
让我们回顾一下 RSA 最简单的加密版本,同时也最能体现密码学和数学的高明之处的加密算法。该算法基于大数分解难题假设。
令 , 其中 是同等规模的大质数。令 是满足 ,其中 是乘法群 的阶。
我们称 为 RSA 模数, 为公共指数。 对是公钥,用于加密信息。 对是私钥,只有拥有私钥的人才能解密信息。给定要加密的信息为整数 ,为了对 加密,需要计算 。为了对 解密,需要计算 。算法的正确性可由欧拉定理证明。
2. Low Public Exponent RSA
在许多实际应用场景中,加密过程是在算力受限设备上进行的,因此将 计算到高次是非常消耗电量、时间的。因此有人尝试简化加密过程,把公共指数 设置为很小的数,比如 。如此一来,加密过程简化很多,只需要计算 ,使用两次乘法即可。
乍一看,在不知道 的因数分解的情况下,好像没办法恢复出密文。然而,正如本文所要介绍的,存在一些高明的攻击手段。
首先我们回顾一下中国剩余定理(CRT):给定 个等式 ,其中 互素。则存在唯一的 ,并且可以高效的找到这个 。
回到低指数加密场景中,假设 A 现在要发送 B、C、D 同样的信息 ,用各自的公钥分别加密,为: 。那么这种情况下,可以很容易恢复明文 。
不失一般性,我们假设这三个模数都是互素的(如果不互素,则可以找到公因子,然后直接恢复明文),攻击者可以由中国剩余定理计算一个值 。由于 成立,则 成立。那么我们计算的 ,最后我们对 开三次方即可。
上述攻击成功的前提是若干次加密都是使用同样的明文,那么如果给每个人发送的消息都不一样呢?考虑如下解决方案:每个人除了公私钥,还有一个唯一的 ID 号,现在 A 对 B、C、D 发送的加密消息变更为: 。其中 是消息 的比特长度。这种方式下,攻击者就不能通过上述中国剩余定理进行攻击了。
3. Coppersmith’ Method
事实上,对于上述修改方案,我们可以采用一个更一般的攻击方法,这需要使用到 Coppersmith’ Method。
Coppersmith’ Method的作用:设存在一个模多项式 。如果该模多项式的根为 ,即 ,且根足够小。那么就可以用 Coppersmith’ Method 去找这个小根。
首先,如果知道 的因式分解,那么这个问题是容易解决的,只需要分别在素因子的子群下解模多项式,然后用一个中国剩余定理即可。另外,如果我们能够找到一个解满足 ,并且这个 不等于 ,那么可以用欧几里得算法将模数分解。因此,我们不寄希望于有一个有效算法对于所有这样的同余式都能找到解,否则也就意味着大数分解难题可以破解了。
既然不能对所有的模多项式都能找到解,那么找到解的条件是什么呢?结论是:对于次数(degree)为 的多项式 ,如果 满足 且 ,那么这个解可以在多项式时间内找到。
First step
不失一般性,设 模 多项式为首一多项式: (得到首一多项式很简单,只需要对多项式乘以 。而如果 不存在,则我们找到了 的一个因子,这个同余式可以拆成同余式组,使用中国剩余定理即可)。假设存在一个整数 满足 且 ,我们的任务就是找到这个 。
我们想:如果存在另一个多项式 的根也是 ,且它的系数很小,那我们就可以通过求根公式或者牛顿迭代法将 求出。而Coppersmith’ Method 算法核心思路就是把 通过一系列变换规约成 。【注意 不是模多项式】
Example 1
设 ,我们想找到一个小根满足 。【这里 ,但在整数域下 】
我们可以找到 满足 ,这个解可以用求根公式得到。
ok,这就是 Coppersmith’ Method 的核心思想。
接下来是讨论 的界的问题(多小的根算小根?)以及提高界的手法。
我们定义 为这个 的上界,然后我们把 用向量的形式表示:
Howgrave-Gramham 定理
给定模多项式 ,模数为 ,根的上界为 , 的向量表示为 ,满足 。那么,当 时,有 。
Proof:
根据柯西不等式有: ,当 时,柯西不等式变形为: 。
我们首先把 表示为 ,可得到不等式: 。
把 的上界代入有: 。
根据之前柯西不等式的变形有:
因此当 时,有 。
于是有: 。又因为: ,因此 。
这个定理(简称 HG 定理)对于估计根的界非常重要!!
之前 example 1 中 是直接给出的,下面介绍一下 G(x) 到底该怎么找?首先考虑 个多项式 ,还有 。显然它们均有解 ,因此我们对其进行线性组合之后它们仍然有解 。
谈到线性组合,那么就很容易联想到矩阵,我们讲这些式子的系数向量组合写成矩阵:
其中 是 取值的上界。
由于是下三角矩阵,则矩阵的行列式为: 。
我们对这个矩阵利用 LLL 算法进行格基规约,设规约后的第一行行向量为: 。根据 LLL 算法第一个性质有: 满足 。因此 。
为了满足 HG 定理,使得规约之后的向量( 的系数)“足够小”,使得我们可以很快的求出根,故要求 ,移项之后有: 。如果 ,则 ;如果 ,则 。
至此我们大致有了 的取值范围,但还没达到前面给出的结论 的程度。因此格子还有继续优化的空间。
Example 2
设 ,多项式 。【这里根 ,因此满足 】
这里我们初步构想 ,则构造格子:
利用 LLL 格基规约之后,我们得到第一行向量为: ,消去 我们得到最终的系数 ,在对这个多项式采用牛顿迭代法求根即可。
from sage.rings.polynomial.refine_root import refine_root
M = 10001
X = 10
L = matrix(ZZ, 4, 4)
for i in range(3):
L[i, i] = M * X ^ i
L[3, 0] = -222
L[3, 1] = 50000
L[3, 2] = 1000
L[3, 3] = 1000
v = L.LLL()[0]
# print(v)
p = 0
x = polygen(ZZ)
for i, coef in enumerate(v):
p += (coef / X ^ i) * x ^ i
ans = p.roots()
# [(4, 1)]
最终得到的结果就是我们预想的 。
Full Coppersmith Method
回顾一下 Example 2,即使以 来计算边界,那么应该在 左右,那么为什么我们取 也能计算出正确结果?而且,如果把 代入 ,那么 X 的边界值应该在 左右。所以为什么我们能得到正确结果呢?
因为其实这个边界值也并不是很严格,在推导得出这个值的时候本身就用了很多次不等式,再者,我们利用的LLL中的那个性质,我们取的是 LLL 算法规约出来的最坏的情况,而大多数情况得到的结果要比这值小许多。
回到不等式: ,再往前还原是: ,其中 是格的维度。
观察这个不等式,我们发现,要增大 ,有两种方案:1. 增大 ;2. 增大 。
针对第一种方案,我们称往格里增加的格的维度,而不增加 的多项式为 x-shift polynomial。它们是 。显然这些多项式的解都为 。
第二种方案,可以增加 的幂次来增加 。由于 ,则有 。
在 Example 2 中,我们的格子的维度为 ,我们带入不等式 ,得到 。现在我们往格子里添加 x-shift polynomials ,新的格子为:
现在格子的维度为 ,再代入不等式,我们得到 。确实增大了 。
一个现成的结论是:当我们给格子增加 x-shift polynomials,可以使得 。那么如果当我们使用第二种方案继续增加 呢?
Coppersmith 定理
设 , 是 次首一多项式,如果在有限域 下,有一个或多个根满足 ,那么我们就可以在与 相关的多项式时间内找到它。
证明过程省略(如果实在没找到过程可以找我一起探讨)。现成的结论:Coppersmith’ Method 的大致时间复杂度为: 。
既然气氛烘托到这,那么出道趣味题让大家练练手吧!
设 ,并且告诉我们 。请解出 。【不要用中国剩余定理,用构造格的方法。答案是:16384】
4. Attack
介绍 Coppersmith’ Method 这个寻小根算法之后,我们在来回到如果攻破改进之后低公共指数的 RSA 加密系统。
假设 是 个互素的整数。设 , 为最大阶为 的多项式。假设存在唯一的 ,使得 都成立。那么。如果 ,可以有效的从 中找到 。
其中 就对应“改进后”的低公共指数的 。它是一个低阶多项式,可以使用 coppersmith’ method 找到那个根 。
不久我会单独整理一份 jupyter notebook 用于记录 RSA 加密系统中的那些“高明”的攻击技巧。