PCIe专题学习——3.2(数据链路层Ack/Nak机制解析)

2024-03-15 00:30

本文主要是介绍PCIe专题学习——3.2(数据链路层Ack/Nak机制解析),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

之前我们讲了对PCIe的一些基础概念作了一个宏观的介绍,了解了PCIe是一种封装分层协议(packet-based layered protocol),主要包括事务层(Transaction layer), 数据链路层(Data link layer)和物理层(Physical layer)。

一:Ack/Nak机制

在上一篇文章“DLLP结构与类型”中,我们有说到,数据链路层会产生多个DLLP,其中有两个DLLP分别是Ack DLLP和Nak DLLP,这两个DLLP均是由接收端传至发送端,也可以理解为一种反馈机制。

Ack DLLP: 表示接收端收到了来自发送端的正确的TLP报文;

Nak DLLP: 表示接收端有来自发送端的TLP报文没有被正确接收,需要发送端重新发送;

我们通过下面这张图,先来大致了解一下Ack/Nak的工作机制:

  1. 发送端数据链路层传送一个TLP(Sequence+TLP+LCRC),通过Link,到达接收端。

  2. 接收端接收到来自发送端数据链路层的TLP报文之后,先检验LCRC,在检验Sequence ID。

  3. 当Sequence ID和LCRC检验均正确时,接收端返回Ack DLLP告知发送端:"您发送的TLP,我方已正确接收,请知悉!"。

  4. 当Sequence ID或者LCRC检验中,发现哪怕一个错误,接收端都会返回Nak DLLP告知发送端:"对不起,您发送的TLP没有被正确接收,请您再发送一下"。

现在大概知道Ack/Nak的工作机制了吧?继续王往下看,再来一张图:

是不是有点晕,小编第一次看到的时候也晕~

上面这幅图是数据链路层Ack/Nak机制详细的结构图,理解了上面这副图就基本掌握了Ack/Nak机制;

不要被上面这幅图吓到,我们下面就一步一步地分析:

1:从发送端事务层传送的TLP到达数据链路层

如下图红丝圈内所示,不过这里需要注意的是:当Retry buffer是满的或者正在执行重新发送TLP的状态,数据链路层将会锁定TLP传送,不再接收。

2:为TLP分配Sequence ID

当TLP到达数据链路层之后,第一件事就是被分配Sequence ID,也可以理解为给TLP一个身份编号,以便于后续的检验工作;

这里还要提一个参数:NEXT_TRANSMIT_SEQ, 简称NTS,是一个12位的Sequence ID计数器,初始值为0,最大取值为4095。一个TLP被分配Sequence ID之后,NTS会加1,然后把累加后的数值再赋值给下一个TLP。

3:为TLP增加LCRC

TLP分配Sequence ID后,下一步就是生成LCRC

LCRC为32bit,基于事务层传送的TLP和数据链路层分配的Sequence ID生成;

4:将TLP在Retry Buffer备份

TLP在加上前缀Sequence ID和后缀LCRC之后,会在Retry buffer里面完整备份

  • 单个TLP最大占用的Retry Buffer大小为:4122 Bytes ( 2 bytes Sequence ID + 16 bytes Header + 4096 bytes Data + 4 bytes ECRC +  4 bytes LCRC ).

  • PCIe Spec中并没有规定Retry Buffer大小,不同的设计采用的大小不同,但是必须保证在TLP传输的过程中不能遇到瓶颈.

 5:接收端对接收的TLP进行LCRC检查

接收端接收到发送端传来的TLP后会先根据Sequence ID,Header,Data,ECRC计算LCRC然后再跟传进来的LCRC对比,检查是否一致;

6:LCRC检查fail

当LCRC检查fail时,会舍弃刚才传进来的TLP,并将NAK_SCHEDULED标志位置起来,给发送端回报NAK DLLP,此外将期望接收到。

7:LCRC检查OK,检查Sequence ID

当TLP的LCRC检查OK之后,接收端继续检查Sequence ID,这里出现了新的参数:NEXT_RSV_SEQ,简称NRS,用于追踪下一个期望获得的TLP的Sequence ID,NRS有12位,取值范围为0~4095;

检查Sequence ID时分为三个情况:

  • Sequence ID=NRS, 代表是TLP接收正确;

  • Sequence ID<NRS, 代表是TLP是重复的;

  • Sequence ID>NRS, 代表是TLP有发生丢失的情况;

(1)当Sequence ID = NRS时

这个情况下,正确接收TLP,并将TLP传送至上层事务层,同时NRS要加1,准备接收下一个TLP,另外还要给发送端回报Ack DLLP告知发送端已正确接收TLP;

 (2)当Sequence ID < NRS时

这个情况下,代表接收端收到了重复的TLP,当下收到的这个TLP会被舍弃,此时NRS保持原有数值,并给发送端报上一个有效TLP的Ack DLLP;

(3)当Sequence ID > NRS时

这个情况下,代表当下Sequence ID之前的TLP丢失,当下收到的这个TLP会被舍弃,此时NRS保持原有数值,NAK_SCHEDULED标志位被置起,并给发送端回报上一个有效TLP的Ack DLLP;

8:Ack/Nak Latency Timer

接收端还有一个重要的参数:Ack/Nak Latency Timer,延迟时间不是固定的,与Link Width,Max payload有关。

Latency timer超时,Ack/Nak生成器会给发送端发送Ack DLLP。发送Ack DLLP之后,Latency Timer会重置。 

9:发送端检查接收端返回的DLLPs

当发送端收到接收端返回的Ack/Nak DLLPs之后,会先检查其CRC,如下图中红色圈内所示:

发送端会根据Ack/Nak DLLPs的byte0-3计算CRC,并与传进来的CRC做比对,验证是否一致;

(1)CRC检查fail

只要CRC检查fail,当下的Ack/Nak DLLPs就会被舍弃;

(2)CRC检查OK之后,继续后续步骤

10,发送端检查AS参数

这里有一个新的参数:Acknowledged Sequence Numbers,简化标记为AckD_SEQ,缩写为AS,AckD_SEQ是一个12位的计数器,用于记载最近收到的Ack/Nak DLLP中的Sequence ID;

当发送端收到的Ack/Nak DLLP中的Sequence ID大于AS时,代表TLPs传输正在进行中。

11:接收到Nak DLLP,TLP retry

当发送端接收到一个Nak DLLP时,代表前面传送的TLP有问题,需要重新发送,此时会有两种情况:

(1)如果Nak DLLP的Sequence ID=AS:

这个情况下,说明没有新的TLP传输,此时需要将Retry buffer中所有的TLPs重新发送,并且更新Relay_NUM+1;

(2)如果Nak DLLP的Sequence ID > AS:

这个情况下,说明有新的TLP传输,此时需要Retry buffer中Nak DLLP中Sequence ID之前的TLP全部清空,并将当下的TLP重新发送,与此同时,将Replay_TIMER以及Replay_NUM重置,并且Replay_NUM重置后加1;

12:NTS-AS >=2048?

发送端在接收到Ack/Nak DLLPs最后一步要检验NTS-AS差值,NTS-AS差值最小为2048;

  • 如果NTS-AS>=2048不成立,则说明数据链路层有协议错误。

  • 如果NTS-AS>=2048成立,则数据链路层继续传输TLPs

看完上面的理论之后是不是还有点晕晕的,我们再来看两个例子加深一下理解:

例1:TLPs丢失 

(1)下图中,Device A要给DeviceB传输5个TLPs,Sequence ID分别是4094,4095,0,1,2;

(2)TLP 4094第一个被成功接收,返回Ack DLLP给Device A,同时Next_RCV_SEQ加1(也就是=4095)

(3)TLP 4095第二个被成功接收,返回Ack DLLP给Device A,同时Next_RCV_SEQ加1(也就是=0,因为4095+1超过了Next_RCV_SEQ的最大取值4095,从0开始计)

(4)TLP 4094第三个被成功接收,返回Ack DLLP给Device A,同时Next_RCV_SEQ加1(也就是=1)

(5)在TLP 0被成功接收之后,Ack/Nak_LATENCY_TIMER超时,重新发送Sequnce ID=0对应的Ack DLLP;

(6)TLP1在传输的过程中由于某些原因(比如物理层的Error)丢失,TLP2继续传输,但是在Device B端在等待TLP1,比较Sequnce ID(=2)与NRS(=1)发现,Sequence ID > NRS,Device B端才知道有TLP丢失,此时将TLP2丢失,并回报NRS-1(1-1=0)对应的Nak,也就是Nak 0;

(7)Device A端接收到Sequence ID=0对应的Ack DLLP之后,重新发送TLP1和TLP2,并且将Sequence ID=0之前的TLPs(4094,4095,0)全部从Retry buffer里面清除。

 例2:Nak DLLP错误

(1)下图中,Device A要给Device B传输5个TLPs,Sequence ID分别是4094,4095,0,1,2.

(2)TLP 4094,4095,0被依次成功接收,此时Next_RCV_SEQ= 1

(3)在TLP 1 到达Device B端,检查32-bit LCRC fail,此时,返回NRS-1对应的Nak DLLP,也就是Nak 0;

(4)Nak 0 到达Device A端,检查16-bit CRC fail,Nak 0 则被丢弃;

(5)Nak 0 被丢弃后,Device B不会再发送任何Ack或者Nak,由于长时间没有收到Device B的反馈(Ack/Nak),Replay_TIMER超时,Device A 将Retry buffer中所有的TLPs重新发送。

(6)TLP 4094,4095,0重新发送后在Device B端会被识别到时重复的TLPs,然后被丢弃,TLP1,2继续正确传输。

这篇关于PCIe专题学习——3.2(数据链路层Ack/Nak机制解析)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/810217

相关文章

网页解析 lxml 库--实战

lxml库使用流程 lxml 是 Python 的第三方解析库,完全使用 Python 语言编写,它对 XPath表达式提供了良好的支 持,因此能够了高效地解析 HTML/XML 文档。本节讲解如何通过 lxml 库解析 HTML 文档。 pip install lxml lxm| 库提供了一个 etree 模块,该模块专门用来解析 HTML/XML 文档,下面来介绍一下 lxml 库

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

大模型研发全揭秘:客服工单数据标注的完整攻略

在人工智能(AI)领域,数据标注是模型训练过程中至关重要的一步。无论你是新手还是有经验的从业者,掌握数据标注的技术细节和常见问题的解决方案都能为你的AI项目增添不少价值。在电信运营商的客服系统中,工单数据是客户问题和解决方案的重要记录。通过对这些工单数据进行有效标注,不仅能够帮助提升客服自动化系统的智能化水平,还能优化客户服务流程,提高客户满意度。本文将详细介绍如何在电信运营商客服工单的背景下进行

基于MySQL Binlog的Elasticsearch数据同步实践

一、为什么要做 随着马蜂窝的逐渐发展,我们的业务数据越来越多,单纯使用 MySQL 已经不能满足我们的数据查询需求,例如对于商品、订单等数据的多维度检索。 使用 Elasticsearch 存储业务数据可以很好的解决我们业务中的搜索需求。而数据进行异构存储后,随之而来的就是数据同步的问题。 二、现有方法及问题 对于数据同步,我们目前的解决方案是建立数据中间表。把需要检索的业务数据,统一放到一张M

关于数据埋点,你需要了解这些基本知识

产品汪每天都在和数据打交道,你知道数据来自哪里吗? 移动app端内的用户行为数据大多来自埋点,了解一些埋点知识,能和数据分析师、技术侃大山,参与到前期的数据采集,更重要是让最终的埋点数据能为我所用,否则可怜巴巴等上几个月是常有的事。   埋点类型 根据埋点方式,可以区分为: 手动埋点半自动埋点全自动埋点 秉承“任何事物都有两面性”的道理:自动程度高的,能解决通用统计,便于统一化管理,但个性化定

使用SecondaryNameNode恢复NameNode的数据

1)需求: NameNode进程挂了并且存储的数据也丢失了,如何恢复NameNode 此种方式恢复的数据可能存在小部分数据的丢失。 2)故障模拟 (1)kill -9 NameNode进程 [lytfly@hadoop102 current]$ kill -9 19886 (2)删除NameNode存储的数据(/opt/module/hadoop-3.1.4/data/tmp/dfs/na

异构存储(冷热数据分离)

异构存储主要解决不同的数据,存储在不同类型的硬盘中,达到最佳性能的问题。 异构存储Shell操作 (1)查看当前有哪些存储策略可以用 [lytfly@hadoop102 hadoop-3.1.4]$ hdfs storagepolicies -listPolicies (2)为指定路径(数据存储目录)设置指定的存储策略 hdfs storagepolicies -setStoragePo

Hadoop集群数据均衡之磁盘间数据均衡

生产环境,由于硬盘空间不足,往往需要增加一块硬盘。刚加载的硬盘没有数据时,可以执行磁盘数据均衡命令。(Hadoop3.x新特性) plan后面带的节点的名字必须是已经存在的,并且是需要均衡的节点。 如果节点不存在,会报如下错误: 如果节点只有一个硬盘的话,不会创建均衡计划: (1)生成均衡计划 hdfs diskbalancer -plan hadoop102 (2)执行均衡计划 hd

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi