我的编程空间,编程开发者的网络收藏夹
学习永远不晚

在.NET中如何使用FixedTimeEquals应对计时攻击

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

在.NET中如何使用FixedTimeEquals应对计时攻击

这篇“在.NET中如何使用FixedTimeEquals应对计时攻击”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“在.NET中如何使用FixedTimeEquals应对计时攻击”文章吧。

计时攻击

在计算机安全中,计时攻击(Timing attack)是旁道攻击 (Side-channel attack) 的一种,而旁道攻击是根据计算机处理过程发出的信息进行分析,包括耗时,声音,功耗等等,这和一般的暴力破解或者利用加密算法本身的弱点进行攻击是不一样的。

举个例子

假如您有一个后端 webapi, GetConfig 接口用来获取配置信息,调用时需要在 Header 中传入一个秘钥,然后判断是否正确并进行返回,如下

X-Api-Key: x123
[HttpGet]public IActionResult GetConfig(){    var key = Request.Headers["X-Api-Key"].FirstOrDefault();    if (key != "x123")    {        return Unauthorized();    }    return Ok(configuration);}

注意,这里我们为了判断两个字符串相等,通常会使用 == 或者 != , 实际上背后使用了 String 的 Equals() 方法,如下

// Determines whether two Strings match.public static bool Equals(string? a, string? b){    if (object.ReferenceEquals(a, b))    {        return true;    }    if (a is null || b is null || a.Length != b.Length)    {        return false;    }    return EqualsHelper(a, b);}

而内部又使用了 SequenceEqual() 方法

[MethodImpl(MethodImplOptions.AggressiveInlining)]private static bool EqualsHelper(string strA, string strB){    Debug.Assert(strA != null);    Debug.Assert(strB != null);    Debug.Assert(strA.Length == strB.Length);    return SpanHelpers.SequenceEqual(            ref Unsafe.As<char, byte>(ref strA.GetRawStringData()),            ref Unsafe.As<char, byte>(ref strB.GetRawStringData()),            ((uint)strA.Length) * sizeof(char));}

大概的逻辑是,先判断两个字符串长度是否一致,如果不是,直接返回 false,然后循环字符串进行逐位对比,一旦发现不相同,直接返回 false,伪代码如下

public bool Equals(string str1, string str2){    if (str1.Length != str2.Length)     {        return false;    }     for (var i = 0; i < str1.Length; i++)    {        if (str1[i] != str2[i])        {            return false;        }    }     return true;}

这里有一个问题是,如果字符串第一位不相同,直接就返回 false,如果最后一位不相同,那就需要遍历到最后,然后返回 false。不一样的字符串,计算的时长可能不一致。

尝试破解

假如用户知道了我们的秘钥的固定长度是 4 位。

GET /GetConfig  X-Api-Key:a000Cost: 2ns

本次耗时了 2ns, 接下来又输入 b000, c000....

GET /GetConfig  X-Api-Key:b000Cost: 2ns GET /GetConfig  X-Api-Key:c000Cost: 2ns ...GET /GetConfig  X-Api-Key:x000Cost: 4ns

直到输入了 x000, 发现其他的耗时都是 2ns, 而这里是 4ns,大概率判定第一位是 x。

注意,这里的测试进行了放大,可能每个 case 分别调用了 100 次,然后统计了 P50(中位数)得出的结果。

然后用同样的方法,测试第二位,第三位....., 最终破解拿到了秘钥。

使用固定时间的算法

虽然看上去有点扯,但确实是真实存在的,包括大名鼎鼎的针对 TLS 的 Lucky 13 攻击,有兴趣的同学可以看一下。在安全性要求比较高的场景中,确实要考虑到计时攻击,当涉及到安全时,还是宁可信其有。

所以我们的算法的执行耗时应该是固定的,不应该在不匹配时,就立即返回,我们尝试改造一下代码

public bool Equals(string str1, string str2){    if (str1.Length != str2.Length)    {        return false;    }    bool reult = true;    for (var i = 0; i < str1.Length; i++)    {        if (str1[i] != str2[i])        {            reult = false;        }    }    return reult;}

不管怎么样,都会遍历完整个字符串,然后返回结果,看上去没什么问题,时间总是固定的,但在现代的 CPU 和 .NET 上却不是的,因为我们要考虑到分支预测,特别是 if 条件。

好吧,那我们调整一下代码

public bool Equals(string str1, string str2){    if (str1.Length != str2.Length)    {        return false;    }    bool reult = true;    for (var i = 0; i < str1.Length; i++)    {        reult &= str1[i] == str2[i];     }    return reult;}

我们用了运算符 &,来代替 If, 只有全部为 true 时,才会返回 true,其中任意一个字符不匹配,就会返回 false,看上去不错。

但是,还有一些问题,对于bool类型的 result (true/false), 我们的 .NET JIT 和 x86 指令执行仍然会进行一些优化,我们再调整一下代码

public bool Equals(string str1, string str2){    if (str1.Length != str2.Length)    {        return false;    }    int reult = 0;    for (var i = 0; i < str1.Length; i++)    {        reult |= str1[i] ^ str2[i];     }    return reult == 0;}

我们把 bool 改成了 int 类型,然后使用了运算符 ^ 和 |,同样的,只有字符串全部匹配时,result 为 0,,才会返回 true, 其中任意一个不匹配,result 就不为 0,会返回 false。

最后,为了防止 JIT 对我们的代码进行其他的优化,我们可以加一个特性,告诉 JIT 不要管它,就像这样

[MethodImpl(MethodImplOptions.NoInlining | MethodImplOptions.NoOptimization)]public bool Equals(string str1, string str2){    if (str1.Length != str2.Length)    {        return false;    }    int reult = 0;    for (var i = 0; i < str1.Length; i++)    {        reult |= str1[i] ^ str2[i];     }    return reult == 0;}

上面我们实现了一个针对字符串比较的固定时间的算法,来应对计时攻击。

实际上, 从 .NET Core 2.1 开始就已经做了内置支持,我们可以直接使用 FixedTimeEquals 方法, 看一下它的实现

[MethodImpl(MethodImplOptions.NoInlining | MethodImplOptions.NoOptimization)]public static bool FixedTimeEquals(ReadOnlySpan<byte> left, ReadOnlySpan<byte> right){     if (left.Length != right.Length)    {        return false;    }    int length = left.Length;    int accum = 0;    for (int i = 0; i < length; i++)    {        accum |= left[i] - right[i];    }    return accum == 0;}

现在用起来也很方便:

var result = CryptographicOperations.FixedTimeEquals(   Encoding.UTF8.GetBytes(str1), Encoding.UTF8.GetBytes(str2));

以上就是关于“在.NET中如何使用FixedTimeEquals应对计时攻击”这篇文章的内容,相信大家都有了一定的了解,希望小编分享的内容对大家有帮助,若想了解更多相关的知识内容,请关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

在.NET中如何使用FixedTimeEquals应对计时攻击

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

在.NET中如何使用FixedTimeEquals应对计时攻击

这篇“在.NET中如何使用FixedTimeEquals应对计时攻击”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“在.NET
2023-06-30
操作系统安全审计:威胁无处不在!教你如何应对,防范系统被攻击
2024-02-28

如何在Android应用中使用C++对Bitmap对象进行处理

这篇文章将为大家详细讲解有关如何在Android应用中使用C++对Bitmap对象进行处理,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。步骤如下:a.编写JNI接口函数//java接口函数p
2023-05-31

如何在canvas中使用环形倒计时组件

这篇文章将为大家详细讲解有关如何在canvas中使用环形倒计时组件,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。Canvas环形倒计时组件Canvas环形倒计时是基于Canvas实现的倒计时
2023-06-09

如何在Android应用中使用AlertDialog实现一个对话框

本篇文章为大家展示了如何在Android应用中使用AlertDialog实现一个对话框,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。一、确定对话框AlertDialog.Builder builde
2023-05-31

在Android应用中使用Handler时出现 leak如何解决

这篇文章给大家介绍在Android应用中使用Handler时出现 leak如何解决,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。 在Android中,Handler类应该是静态的,否则,可能发生泄漏。在应用程序线
2023-05-31

在Android应用中AlertDialog的点击按钮无法关闭对话框如何解决

在Android应用中AlertDialog的点击按钮无法关闭对话框如何解决?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。这里的问题:当我点击确定按钮,也就是
2023-05-31

如何在android应用中利用service实现一个计时器功能

本篇文章给大家分享的是有关如何在android应用中利用service实现一个计时器功能,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。1、首先建立主页面的设计:activity
2023-05-31

如何在java中使用DateUtils工具类对时间进行转换

本文章向大家介绍如何在java中使用DateUtils工具类对时间进行转换,主要包括如何在java中使用DateUtils工具类对时间进行转换的使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。J
2023-05-30

在Android 应用中使用TextView时出现imageview被压缩如何解决

在Android 应用中使用TextView时出现imageview被压缩如何解决?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。代码示例如下:
2023-05-31

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录