PCIe扫盲(六)

news2024/11/15 7:41:17

在这里插入图片描述

系列文章目录


PCIe扫盲(一)
PCIe扫盲(二)
PCIe扫盲(三)
PCIe扫盲(四)
PCIe扫盲(五)
PCIe扫盲(六)


文章目录

  • 系列文章目录
  • Flow Control基础(一)
  • Flow Control基础(二)
  • Quality of Service 简介
  • DLLP(数据链路层包)详解
    • Ack/Nak
    • Power Management
    • Flow Control
    • 厂商自定义
  • Ack/Nak 机制详解(一)
  • Ack/Nak 机制详解(二)
    • Example 1. Example of Ack
    • Example 2. Ack with Sequence Number Rollover
    • Example 3. Example of Nak
    • Example 4. Example of Lost TLPs
    • Example 5. Example of Bad Nak
  • 转载链接


Flow Control基础(一)

  Flow Control 即流量控制,这一概念起源于网络通信中。PCIe 总线采用 Flow Control 的目的是,保证发送端的 PCIe 设备永远不会发送接收端的 PCIe 设备不能接收的 TLP(事务层包)。也就是说,发送端在发送前可以通过 Flow Control 机制知道接收端能否接收即将发送的 TLP

  在 PCI 总线中,并没有 Flow Control 这样的机制,因此发送端并不知道当前时刻,接收端能否接收对应的 TLP。因此,发送端只能先尝试发送,期间可能会被插入多个等待周期(接收设备尚未就绪等原因),甚至是重发(Retries)等。

  PCIe Spec 规定,PCIe 设备的每一个端口(Ports)都必须支持 Flow Control 机制,在发送 TLP 之前,Flow Control 必须先检查接收端口是否有足够的 Buffer 空间来接收这个 TLP 。当 PCIe 设备支持多个 VCVirtual Channel)时,Flow Control 机制可以显著地提高总线的传输效率。

  PCIe Spec 规定,每个 PCIe 端口最多支持 8VC,并且每个 VCFlow Control Buffer 是完全独立的。也就是说,某一个 VCFlow Control Buffer 满了,并不会影响其他的 VC 的通信。

  注:一般 Endpoint 只有一个端口,Root 有一个或者多个端口,Switch 有一个 Upstream 端口和多个 Downstream 端口。

  前面的文章中介绍过,Flow Control 机制是通过相邻两个端口(Ports)的数据链路层之间发送 DLLPFlow Control DLLPs)来实现的。也就是说 Flow Control 是一种点到点(Point to Point)的方式,而非端到端(End to End)。在进行初始化的时候,接收端需要向发送端报告(reports)其 Buffer 的大小,在正常运行状态(Run-time)时,会周期性地通过 Flow Control DLLPs 来告知发送端,接收端的各个 Buffer 的大小。

  需要注意的是,虽然 Flow Control DLLP 只在相邻的数据链路层之间传输,但是相关的 Buffer 和计数器(FC Counter)确是在事务层(Transaction Layer)的,即事务层参与了 Flow Control 机制的管理。如下图所示:

在这里插入图片描述

  前面的文章中多次介绍过,TLP 一共有三大类:Posted Transactions(包括 Memory WritesMessages)、Non-Posted Transactions(包括 Memory ReadsConfiguration Reads and WritesIO Reads and Writes)以及 Completions(包括 Read and Write Completion)。并且知道,TLP 可以分为两个部分,HeaderData 部分。Flow Control 为了获得更高的数据传输效率,将这三类 TLP 分开存放,同时将 HeaderData 部分也分开存放。因此,一共存在 6 种不同的 Flow Control Buffer 类型,如下图所示:

在这里插入图片描述

  Flow Control Buffer 的存储单元(Unit)被称作 Flow Control Credits。对于 Header 来说,Requests TLP 每个 unit 等于 5DW,而 Completions TLP 每个 unit 等于 4DW 。对于 Data 来说,每个 unit 等于 4DW,即 Data Buffer 是按照 16 个字节对齐的。对于各种类型的 Buffer 的最小值如下表所示:

在这里插入图片描述

  最大值如下表所示:

在这里插入图片描述

  注:0 unit 表示无限(Infinite)。

Flow Control基础(二)

  在任何事务层包(TLP)发送之前,PCIe 总线必须要先完成 Flow Control 初始化。当物理层完成链路初始化后,便会将 LinkUp 信号变为有效,告知数据链路层可以开始 Flow Control 初始化了。如下图所示:

在这里插入图片描述

  注:由于 VC0 是默认使能的,所以当 Flow Control 初始化开始时,其会被自动的初始化。其他的 Virtual Channel 是可选的,只有当被配置为使能的时候才会被初始化。

  Flow Control 初始化被分为两个步骤,FC_Init1FC_Init2,其在整个数据链路控制和管理状态机(Data Link Control & Management State Machine)的位置如下图所示:

在这里插入图片描述

  在 FC_Init1 步骤中,PCIe 设备会连续地发送三个 InitFC1 类型的 Flow Control DLLP 来报告其接收 Buffer 的大小。三个 DLLP 的顺序是固定的:PostedNon-Posted 然后是 Completions。如下图所示:

在这里插入图片描述

  FC_Init2FC-Init1 类似,同样是连续的发送三个 InitFC2 类型的 DLLP,当完成后,DLCMSM(上一篇文章中提到的状态机)会切换到 DL_Active 状态,表明数据链路层初始化完成。

  注:可能有人会有疑惑了,FC_Init1FC_Init2 干的活不是差不多嘛,为什么还需要 FC_Init2 呢?原因是,不同的设备完成 FC_Init1 的时间可能是不同的,增加 FC_Init2 是为了保证每个设备都能收到 FC 初始化 DLLP

  FC_Init DLLP 的格式如下图所示:

在这里插入图片描述

  在完成 FC 初始化之后,相邻的两个设备之间会周期性的通过 Updated FC DLLP 更新接收 Buffer 的大小。如下图所示:

在这里插入图片描述

  Update FC DLLP 的格式与 FC_Init 的格式是类似的,具体如下:

在这里插入图片描述

  前面说到。Update FC DLLP 是周期性发送的,周期的值可以通过以下公式计算得:

在这里插入图片描述

  具体可以参考 PCIe 的 Spec ,这里不再详细介绍,下面给出 Gen1Gen2 的周期表格(根据公式计算的结果)。其中 UFUpdateFactor

  • Gen1 (2.5GT/s)如下表所示:

在这里插入图片描述

  • Gen2(5GT/s)如下表所示:

在这里插入图片描述

  • Gen3 (8GT/s)如下表所示:

在这里插入图片描述

  注:虽然 UpdateFactor 是指的 Flow Control 中的系数,而 AckFactor 指的是 Ack/Nak 中的系数,但实际上他们的值是一样的。

Quality of Service 简介

  前面的文章中介绍过,为了保证视频、音频等数据得到优先传输,PCIe 总线实现了一种叫做 Quality of ServiceQoS)的机制。QoS 可以满足视频、音频等对 Latency 和实时性(Isochronous)要求比较高(一般不可以被打断)的应用需求。QoS 主要通过 VCVirtual Channel)和 TCTraffic Class)来实现。

  VC 的相关寄存器位于 PCIe 配置空间的扩展部分(PCIe Extended Capability Space),如下图所示:

在这里插入图片描述

  前面的文章中介绍过,每一个 VC 都有独立的 Buffer,某一个 VC Buffer 满了并不会影响其他 VC 的使用。但是只靠 VC 并不能实现 QoS 中的优先级的功能,这还需要 TCTraffic Class)的支持。TC 的值由 TLP Header 中的 Byte1bit[6:4] 定义,如下图所示。显然 TC 值的范围为 0~7,值越大优先级越高,默认为 0(优先级最低)。在初始化的时候,PCIe 驱动程序会为每一种类型的包分配好合适的 TC 值(优先级)。

在这里插入图片描述

  如果 PCIe 驱动程序没有找到 PCIe Extended Capability Space,则认为该设备只有一个 VC,即 VC0。此时再为每一个 TLP 分配不同的 TC 值,显然是没有意义的。因此会默认采用 TC0/VC0 组合,即不支持 QoS 功能。换一句话说,如果某一个 PCIe 设备只支持一个 VC(VC0),那么就没有 QoS 什么事了。

   注:本次连载的博客只是简单地介绍 QoS 的功能和应用,关于 QoS 的详细内容,如 VC 仲裁,端口仲裁,实时性(Isochronous)等相关内容,还请参考 PCIe Spec 的相关章节。

  PCIe 驱动程序(配置软件)通过修改 VC 资源控制寄存器(VC Resource Control Register)中的 TC/VC Map 位来实现 TC/VC Mapping 。同时通过 VC ID 位来选择相应的 VC 。如下图所示:

在这里插入图片描述

  图中的例子,TC0TC1 对应 VC0,而 TC2~TC4 对应的是 VC3

  TC/VC Mapping 采用了一种灵活的机制,但是仍然需要注意以下几点:

  • TC/VC Mapping 是针对 Link 两端的端口(Ports)的;

  • TC0 会被自动地 MapVC0,且只能 MapVC0

  • 其他的 TC 可以被 Map 到任意的 VC 上;

  • 一个 TC 一般最多只能 Map 到一个 VC 上;

  • 可以有 TC 或者 VC 不被使用。

  如果 Link 的两个端口(Ports)中,VC 数量不一致,则该 Link 只能服从 VC 数量少的端口,如下图所示:

在这里插入图片描述

  PCIe 驱动程序可以通过查询扩展配置空间中的 Extended VC Count 来确定该端口支持的 VC 数量,如下图所示:

在这里插入图片描述

DLLP(数据链路层包)详解

  首先说明一下,在本次连载的博文中,DLLP 一般指的是由发送端的数据链路层发送,接收端的数据链路层接收的数据包,其和事务层(Transaction Layer)一般没有什么关系。本文将要介绍的 DLLP 指的正是这样的数据包,其一般用于 Ack/Nak 机制、功耗管理、Flow Control(流量控制)和一些厂商自定义用途等。示意图如下:

在这里插入图片描述

  DLLP 的格式是固定的,一共有 8 个字节,包括 FramingSDP & END)。和 TLP 不一样的地方是,DLLP 并未携带任何路由信息,原因很简单,因为 DLLP 只在相邻的两个设备的数据链路层之间通信,根本不需要路由。并且 DLLP 一般也不需要和事务层交换信息。

   注:前面文章中介绍的 Flow ControlAck/Nak 等都是针对 TLP,并不会对 DLLP 产生影响。但是这些功能正是借助 DLLP 才得以实现的。

  DLLP 的一般格式如下图所示:

在这里插入图片描述

   DLLP 的类型与目标应用如下表所示:

在这里插入图片描述

Ack/Nak

  其中,用于 Ack/NakDLLP 的格式如下:

在这里插入图片描述

在这里插入图片描述

Power Management

  用于功耗管理(Power Management)的 DLLP 的格式如下:

在这里插入图片描述
在这里插入图片描述

Flow Control

  用于 Flow ControlDLLP 的格式如下:

在这里插入图片描述

在这里插入图片描述

厂商自定义

  厂商自定义的 DLLP 的格式如下:

在这里插入图片描述

Ack/Nak 机制详解(一)

  前面在数据链路层入门的文章中简单地提到过Ack/Nak机制的原理和作用,接下来的两篇文章中将对Ack/Nak机制进行详细地介绍。

  Ack/Nak是一种由硬件实现的,完全自动的机制,目的是保证TLP有效可靠地传输。Ack DLLP用于确认TLP被成功接收,Nak DLLP则用于表明TLP传输中遇到了错误。

在这里插入图片描述

  如上图所示,发送方会对每一个TLP在Replay Buffer中做备份,直到其接收到来自接收方的Ack DLLP,确认该DLP已经成功的被接受,才会删除这个备份。如果接收方发现TLP存在错误,则会向发送发发送Nak DLLP,然后发送方会从Replay Buffer中取出数据,重新发送该TLP。

  Ack/Nak机制内部的详细结构图如下图所示:

在这里插入图片描述

  下面对图中的各个Elements分别做一个简单地介绍。

  首先是发送端的Elements:

  来个大图特写:

在这里插入图片描述

  • NEXT_TRANSMIT_SEQ Counter

  NEXT_TRANSMIT_SEQ Counter,即NTS计数器,是一个12位的计数器。当数据链路层处于DL-Down状态或者复位时,该计数器会被初始化为0。该计数器只会执行加一操作,也就是说当其到达最大值4095时,在进行加一操作则会变成0(Roll Over)。该计数器用于产生下一个待发送的TLP的序列号(Sequence Number)。每一个序列号都是与一个TLP所唯一对应的,可以说这个序列号正是整个Ack/Nak机制的关键。

  • LCRC Generator
      LCRC产生器用于产生一个32位的CRC值,其作用于整个TLP和其对应的序列号。

  • Replay Buffer
      Replay Buffer是Mindshare书中的叫法,在PCIe Spec中,这个Buffer的名称叫做Retry Buffer。Replay Buffer中按照传输顺序,存储了整个TLP、序列号和LCRC,如下图所示:
      
    在这里插入图片描述
      
      当发送端收到来自接收端的Ack DLLP时,会将Buffer中相应的TLP(包括对应的序列号和LCRC)移除;如果接收到的是Nak DLLP,则会将Buffer中响应的TLP(包括对应的序列号和LCRC)重新发送给接收端。

  • REPLAY_TIMER Count
      REPLAY_TIMER是一种看门狗定时器,当该定时器溢出,则表明发送端已经发送了一个或者多个TLP,但是并未收到来自接收端的应答信号(Ack/Nak)。此时,发送端会将Replay Buffer中的TLP重新发送,并将看门狗定时器重启。
      
      只要发送端发送了任何TLP,该定时器便会启动,在接收到来自接收端的应答信号之前都会持续地运行。当收到应答信号之后,定时器会立即被清零。此时如果Replay Buffer仍然有TLP(表明还有TLP被发送,但是仍未得到应答),定时器又会被立即被重新启动。如果Buffer中是空的,则定时器不会被重新启动,直到新的TLP被发送。

  • REPLAY_NUM Count
      这是一个2位的计数器,用于记录同一个TLP发送失败的次数,当其值从11b变为00b时(溢出了,表示尝试发送某个TLP失败了4次),数据链路层会自动地强制物理层重新进行链路训练(即LTSSM进入Recovery状态)。当完成链路训练之后,便会重新发送之前发送失败的TLP。
      
      当发送端接收到来自接收端的Nak DLLP或者发送端的看门狗定时器(REPLAY_TIMER)溢出时,该计数器都会被加一;当接收到Ack DLLP时,该计数器则会被清零。

  • ACKD_SEQ Register
      ACKD_SEQ寄存器用于存储最近接收到的Ack或者Nak DLLP中的序列号。当复位或者数据链路层处于无效状态时,该寄存器会被初始化为全1。关于ACKD_SEQ寄存器的具体用法会在后续的文章中,用例子的形式详细说明。

  • DLLP CRC Check
      接收端在接收到来自发送端的DLLP后,首先会检查其DLLP CRC,如果发现有错误,则会直接将其丢弃,认为其实无效的DLLP。

  然后是接收端的Elements:

  首先来一张大图特写:

在这里插入图片描述

  • LCRC Error Check
      顾名思义,LCRC Error Check用于检查接收到的TLP是否存在错误。如果存在错误,则将对应的TLP直接丢弃,然后产生一个Nak DLLP发送给发送端,让其重新发送该TLP。

  • NEXT_RCV_SEQ Counter
      NEXT_RCV_SEQ是一个12位的计数器,即Next Receive Sequence Number,其值为已经成功接收的TLP的序列号加1。主要用于检查当前接收到的TLP是不是应该接收到的TLP。

如果NEXT_RCV_SEQ和当前接收到的TLP中的序列号的值相等,则认为这是一个有效的TLP,但是接收端并不会立即向发送端发送Ack DLLP,而是等到AckNak_LATENCY_TIMER溢出时才向发送端发送Ack DLLP。也就是说,一个Ack DLLP可能会对应多个TLP,接收端不会每成功接收到一个TLP便向发送端发送Ack DLLP。
  
  如果当前接收到的TLP中的序列号小于NEXT_RCV_SEQ(且差值不超过2048),则认为该TLP之前已经成功发送过了,此次是重复发送。需要注意的是,PCIe Spec认为重复发送并不是一个错误,只是直接将该TLP丢弃,并没有Nak或者Error Reporting,但是会返回一个包含有上一次成功接收的TLP的序列号(NRS-1)的Ack DLLP给发送端。
  
  如果当前接收到的TLP的序列号大于NEXT_RCV_SEQ,表明传输过程中漏掉了一些TLP。此时,接收端会返回Nak DLLP,并直接丢弃该TLP。
  
  一个简单的例子如下图所示:
  
在这里插入图片描述

  • NAK_SCHEDULED Flag
      NAK_SCHEDULED是一个标志位,当接收端产生Nak DLLP时,该标志位会被置位。当接收端成功接收到有效的TLP时,该标志位会被清零。需要特别注意的是,当该标志位处于置位状态时,接收端不应产生其他的Nak DLLP。

  • AckNak_LATENCY_TIMER
      AckNak_LATENCY_TIMER定时器会在接收端成功接收到有效的TLP,且并未向发送端返回Ack DLLP之前运行。当AckNak_LATENCY_TIMER定时器溢出时,接收端会立即向发送端返回Ack DLLP(携带的序列号为NRS-1,即一个Ack对应多个有效的TLP)。无论接收端返回Ack还是Nak,该定时器都会被复位,但是只有当接收端再次收到有效的TLP时,该定时器才会被重新启动。
      
      该定时器(REPLAY_TIMER)的值是由PCIe Spec规定的和Lane的数量与Max_Payload有关,Gen1的值如下图所示:
      
    在这里插入图片描述
      
    Gen2(5GT/s)如下图所示:
      
    在这里插入图片描述
      
       注:该定时器(REPLAY_TIMER)的值是AckNak_LATENCY_TIMER定时器值的三倍。而REPLAY_TIMER的值则如下表所示(Gen1和Gen2):
      
    在这里插入图片描述
    在这里插入图片描述

  • Ack/Nak Generator
      显然,Ack/Nak Generator的功能是产生Ack或者Nak DLLP。其格式如下:
      
    在这里插入图片描述
    在这里插入图片描述
      
    最后,介绍一下PCIe Spec中推荐的包优先级顺序。我们知道,PCIe总线通信中,存在多种类型的包,包括TLP、DLLP和Ordered Sets等。为了能够是总线达到最优的传输效率,PCIe Spec推荐对这些包的优先级做如下的设置:(当然这只是推荐,并没有强制厂商一定要这要去实现)
      

  1. Completion of any TLP or DLLP currently in progress (highest priority)
  2. Ordered Set
  3. Nak
  4. Ack
  5. Flow Control
  6. Replay Buffer re‐transmissions
  7. TLPs that are waiting in the Transaction Layer
  8. All other DLLP transmissions (lowest priority)

   注:这里说的优先级和QoS中说的优先级是两码事,千万不要搞混了。

Ack/Nak 机制详解(二)

  这一篇文章来简单地分析几个Ack/Nak机制的例子。

Example 1. Example of Ack

在这里插入图片描述

  Step1 设备A准备依次向设备B发送5个TLP,其对应的序列号分别为3,4,5,6,7;

  Step2 设备B成功的接收到了TLP3,并将NEXT_RCV_SEQ从3加到4,但是设备B没有立即向设备A返回Ack(此时AckNak_LATENCY_TIMER尚未溢出);

  Step3 设备B又成功地接收到了TLP4和TLP5;

  Step4 假设此时AckNak_LATENCY_TIMER溢出了,则设备B会向设备A返回一个带有序列号为5的Ack DLLP。同时,设备B将AckNak_LATENCY_TIMER复位,但是并未重新启动,直到设备B成功地接收到了TLP6。

  Step5 设备A接收到了Ack5,将REPLAY_TIMER和REPLAY_NUM复位,然后将Buffer中的序列号5(和5之前)的TLP备份移除;

  Step6 一旦设备B接收到TLP6,AckNak_LATENCY_TIMER又会被重新启动。

Example 2. Ack with Sequence Number Rollover

在这里插入图片描述

  Step1 设备A准备依次向设备B发送序列号为4094,4095,0,1,2的TLP,注意第一个发送的是TLP4094,最后一个发送的是TLP2。也就是说序列号Rollover了;

  Step2 设备B成功接收到TLP4094~TLP1后,假设此时AckNak_LATENCY_TIMER溢出了,则设备B向设备A返回Ack1 DLLP;

  Step3 设备A接收到Ack1,并将Buffer中的序列号为1(和之前的,包括TLP4094~TLP1)的TLP备份移除。同时将REPLAY_TIMER和REPLAY_NUM复位。

Example 3. Example of Nak

在这里插入图片描述

  Step1 假设设备A准备依次向设备B发送序列号为4094,4095,0,1,2的TLP;

  Step2 设备B成功地接受了TLP4094,并将NEXT_RCV_SEQ加1,变为4095;

  Step3 设备B接收到了TLP4095,但是该TLP并未通过CRC校检(即存在错误)。此时无论AckNak_LATENCY_TIMER处于何种状态,设备B都会立即向设备A返回Nak4094(注意返回的Nak DLLP中的序列号为上一次成功接收的TLP的序列号)。同时设备B将AckNak_LATENCY_TIMER停止并复位;

  Step4 设备B会一直等待设备A向其发送TLP4095,但是设备A却并不知发生了什么,在接收到设备B向其返回的Ack/Nak之前,会继续发送TLP0~TLP2,只是设备B会直接忽略这些TLP。

  Step5 当设备A接收到来自设备B的Nak4094 DLLP时,会将Buffer中的TLP4094(和之前的TLP)移除,并从TLP4095从新开始发送。同时,将REPLAY_TIMER和REPLAY_NUM复位。

  Step6 由于设备A接收到的是Nak,而不是Ack,因此设备A会重新启动REPLAY_TIMER并将REPLAY_NUM加一;

  Step7 一旦设备B成功地接收到TLP4095,设备B便会清除NAK_SCHEDULED标志位,并将NEXT_RCV_SEQ计数器加一,同时重启AckNak_LATENCY_TIMER。

Example 4. Example of Lost TLPs

在这里插入图片描述

  Step1 假设设备A准备依次向设备B发送TLP 4094,4095,0,1,2;

  Step2 设备B成功地接收了TLP4094~TLP0,并向设备A返回Ack0,此时设备B的NEXT_RCV_SEQ为1;

  Step3 设备A接收到设备B返回的Ack0,从Buffer中移除相应的TLP备份;

  Step4 设备B接收到了TLP2(而不是TLP1),也就是说TLP1在传输过程中丢失了。此时,设备B会直接将TLP2丢弃,并将NAK_SCHEDULED标志位置位,同时向设备A返回Nak0 DLLP;

  Step5 设备A接收到Nak0 DLLP后,会将Buffer中的TLP0(以及之前的,如果有的话)移除。同时,从TLP1开始,重新向设备B发送。

Example 5. Example of Bad Nak

在这里插入图片描述

  Step1 设备A准备依次向设备B发送TLP 4094,4095,0,1,2;

  Step2 设备B成功的接收了TLP4094~TLP0,但是由于AckNak_LATENCY_TIMER尚未溢出,所以设备B没有立即向设备A返回Ack DLLP;

  Step3 设备B发现TLP1中存在错误,于是向设备A返回Nak0 DLLP,并将NAK_SCHEDULED标志位置位;

  Step4 设备A发现其接收到的Nak0 DLLP中也存在错误(CRC校检不通过),于是直接丢弃了Nak0;

  Step5 然而设备B却一直在等待设备A向其发送TLP1,在其成功接收TLP1之前,设备B不会返回任何Ack或者Nak,不管设备A向其发送什么(除TLP1之外的)。设备B的NAK_SCHEDULED标志位也一直保持置位;

  Step6 尴尬的是,设备A并不知道设备B想要其重发TLP1(由于没有成功接收到Nak0)。因此,设备A会继续向设备B发送之后的TLP,但是由于一直没有接收到设备B的Ack/Nak DLLP,设备A的REPLAY_TIMER会在一段时间后溢出;

  Step7 当设备A的REPLAY_TIMER溢出后,设备A会向Buffer中的所有TLP都重新发送一遍,并重启REPLAY_TIMER,同时将REPLAY_NUM计数器加一;

  Step8 设备B会再次接收到TLP4094~TLP0,但是这在之前就已经成功接受到过了。因此设备B会直接将其丢弃,且不会像设备A返回任何的Ack或者Nak

  Step9 此时,设备B再次接收到了TLP1,并未发现错误(成功接收)。于是,设备B将NAK_SCHEDULED标志位清零,并重启AckNak_LATENCY_TIMER,将NEXT_RCV_SEQ加一。

转载链接

  1. Flow Control基础(一)
  2. Flow Control基础(二)
  3. Quality of Service 简介
  4. DLLP(数据链路层包)详解
  5. Ack/Nak 机制详解(一)
  6. Ack/Nak 机制详解(二)

  
 


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2144540.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

io多路复用:epoll水平触发(LT)和边沿触发(ET)的区别和优缺点

在进行ET模式的正式分析之前,我们来举个例子简单地了解下ET和LT: 假设我们通过fork函数创建了父子两个进程,并通过匿名管道来通信,在子进程中,我们一次向管道写入10个字符数据,为"aaaa\nbbbb\n";每隔5s写入…

深入解析JSP会话跟踪:从原理到实践

什么是会话? 生活中:一次电话.一次取款过程Web应用一次会话:一个客户端浏览器与web服务器之间连续发生的一系列请求和相应的过程 什么是会话状态? 会话状态是指Web服务器与浏览器在会话过程中产生的状态信息。 什么是Cookie? 引言:浏览购…

windows使用tcpdump.exe工具进行抓包教程

windows主机安装一些抓包工具可能有些不方便,这里有一个tcpdump.exe工具直接免安装,可以直接使用进行抓包。(工具下载见 附件) tcpdump.exe使用教程 如下: 1:tcpdump -D 可查看网络适配器(注意前面的编号)…

htaccess转换nginx工具

115工具网为您提供htaccess与nginx在线转换,apache伪静态文件转为nginx重写规则,htaccess伪静态规则换nginx,apache RewriteRule转rewrite,apache伪静态文件转nginx重写,apache转nginx重写规则,本工具支持所有的htaccess伪静态、基本的配置规则、重定向等转换为ngin…

java多线程编程示例

程序功能 程序展示了 Java 中如何使用多线程来并行执行任务。具体功能如下: 程序创建了三个线程,每个线程执行相同的任务类 Task。 每个线程在运行时输出自身名称,并模拟执行五次任务,每次任务间隔 1 秒。 主线程在启动这三个线程…

技术周总结 09.09~09.15周日(C# WinForm WPF 软件架构)

文章目录 一、09.09 周一1.1) 问题01: Windows桌面开发中,WPF和WinForm的区别和联系?联系:区别: 二、09.12 周四2.1)问题01:visual studio的相关快捷键有哪些?通用快捷键编辑导航调试窗口管理 2…

C++:日期类的实现

目录 一、前言 二、头文件 三、各个函数的实现 打印、检查日期及获取日期 、、-、-、 、<、<、>、>、 &#xff01; 日期-日期 >>、<< 一、前言 前面几篇讲了关于类和对象的一些知识&#xff0c;本篇就来实现一下前面用到的日期类。 二、头文…

Java 中 List 常用类和数据结构详解及案例示范

1. 引言 在 Java 开发中&#xff0c;List 是最常用的数据结构接口之一&#xff0c;它用于存储有序的元素集合&#xff0c;并允许通过索引进行随机访问。电商系统中&#xff0c;如购物车、订单列表和商品目录等功能都依赖 List 进行数据管理。选择适当的 List 实现类能够显著提…

三菱变频器变更电流最大输入(20mA 初始值)时的频率(60Hz初始值)

变更最高频率。变更示例 在4~ 20mA 输入频率设定器中&#xff0c;将 20mA 时的频率从 60Hz(初始值)变更为 50Hz。 输入 20mA 电流时调整为输出 50Hz。 将Pr.126 设定为“50Hz” NOTE 4mV 时的频率设定可通过校正参数 C5 设定。 其他的频率设定电流增益的调整方法&#xff0c;还…

计算机网络(七) —— https协议与网络安全证书

目录 一&#xff0c;关于https 二&#xff0c;关于加密 2.1 明文&#xff0c;密钥 2.2 对称和非对称加密 2.3 数据摘要&#xff0c;数据指纹&#xff0c;数字签名 三&#xff0c;https过程过程探究 四&#xff0c;证书 4.1 CA认证 4.2 证书大致内容和申请流程 4.3 签…

试用完几十款ETL工具后的经验总结,ETL工具用这三款就足够了

1.ETL选型前言 市面上ETL工具国内外加起来估计得有30种之多&#xff0c;其中近20款工具都花时间试用过&#xff0c;现在把试用后总结出来的经验分享一下&#xff0c;目前很多企业在选择ETL工具时不知道怎么选择适合自己的工具也不可能一款一款的去试用&#xff0c;试用成本非常…

3.postman脚本语言、接口关联(json引用(变量)、脚本用正则表达式)、断言封装、自动化构造接口请求(Postman工具)

一、Postman的脚本语言 1.使用js语言 2.pm变量 pm.sendRequest():发送HTTP请求 二、自动化实现接口关联 1.JSON引用 2.正则表达式&#xff08;在test下编写如下脚本&#xff09; //获取响应 console.log(responseBody) //re的方式获取token let token responseBody.match(&quo…

佰朔资本:股票中什么叫龙头?怎么找龙头股?

龙头&#xff0c;也便是龙头股&#xff0c;指的是某一工作中有必定影响力和号召力的股票&#xff0c;龙头股的涨跌一般对其他同工作板块股票的涨跌有必定演示和引导效果&#xff0c;是一种风向标一般的存在。龙头股的技能面表现和成交量都会比一同间的大盘和地块要强。 龙头股…

三菱变频器以模拟量电流进行频率设定(电流输入)

POINT 1、在 STF(STR)信号 ON 时&#xff0c;发出启动指令。2、请将 AU 信号置为 ON。 3、请设定 Pr.79 运行模式选择 “2”(外部运行模式)。 接线示例 重点&#xff1a;请将 AU 信号置为 ON。 操作示例&#xff1a;以 60Hz 运行。 1、接通电源时的画面&#xff0c;监视器显…

伪工厂模式制造敌人

实现效果 1.敌人方实现 敌人代码 using UnityEngine; using UnityEngine.UI;public class EnemyBasics : MonoBehaviour {public int EnemySpeed { get; internal set; }public int EnemyAttackDistance { get; internal set; }public int EnemyChaseDistance { get; interna…

YOLOv8 OBB win10+ visual 2022移植部署

前言 想做一个目标旋转角度检测的工程&#xff0c;但是网上多少python的&#xff0c;或者linux的。在win10 visual 2022移植部署&#xff0c;记录一下。 参考 这篇文章没有C win10 环境下的部署教程&#xff0c;我相对于是对此做了补充。 1、下载工程 https://github.com/sh…

Matlab 的.m 文件批量转成py文件

在工作中碰到了一个问题&#xff0c;需要将原来用matlab gui做出来的程序改为python程序&#xff0c;因为涉及到很多文件&#xff0c;所以在网上搜了搜有没有直接能转化的库。参考了【Matlab】一键Matlab代码转python代码详细教程_matlab2python-CSDN博客 这位博主提到的matla…

Redis安装 ▎Redis详细知识点

前言: Redis是一个开源的内存数据结构存储&#xff0c;支持丰富的数据类型&#xff0c;如字符串、哈希、列表、集合和有序集合,作为一个键值对数据库&#xff0c;Redis能提供毫秒级的响应时间&#xff0c;适合高并发应用场景。它还支持持久化&#xff0c;将内存数据定期保存到…

路由器接口配置DHCP实验简述

一、路由器配置 [Huawei]undo info-center enable Info: Information center is disabled. [DHCP-SERVER]sysname DHCP-Server [DHCP-Server]dis this sysname DHCP-Server undo info-center enable return [DHCP-Server]dhcp enable Info: The operation may take a few secon…

【pytorch学习笔记,利用Anaconda安装pytorch和paddle深度学习环境+pycharm安装---免额外安装CUDA和cudnn】

作者链接: link 一、安装pytorch环境 1.打开打开anaconda的终端后 conda env list然后创建一个名字叫pytorch&#xff0c;python是3.8版本的环境 conda create -n pytorch python3.8再次看环境 conda env list# conda environments: #显示如下环境 base …