首页 > 编程语言 >[米联客-安路飞龙DR1-FPSOC] UDP通信篇连载-07 ICMP层程序设计

[米联客-安路飞龙DR1-FPSOC] UDP通信篇连载-07 ICMP层程序设计

时间:2024-08-09 18:54:28浏览次数:19  
标签:UDP end DR1 begin echo pkg 安路 icmp data

软件版本:Anlogic -TD5.9.1-DR1_ES1.1

操作系统:WIN10 64bit

硬件平台:适用安路(Anlogic)FPGA

实验平台:米联客-MLK-L1-CZ06-DR1M90G开发板

板卡获取平台:https://milianke.tmall.com/

登录"米联客"FPGA社区 http://www.uisrc.com 视频课程、答疑解惑!

 

3.6 ICMP层

该层在程序中为IP层的子层,设计了接收ICMP请求包并回复的功能。

3.6.1 ICMP接收模块

该模块首先过滤掉ICMP报文头部,并且对头部数据进行校验,校验方式和IP首部校验相同。

//ICMP报文的校验和

always@(posedge I_clk or posedge I_reset) begin

    if(I_reset) begin

        accum1              <=  16'd0;

        accum2              <=  32'd0;

        checksum_state      <=  2'd0;

        checksum_correct    <=  1'b1;

    end

    else begin

        case(checksum_state)

            0:begin

                if(I_icmp_pkg_valid) begin

                    accum1[15:8]    <=  I_icmp_pkg_data;

                    checksum_state  <=  2'd1;

                end

                else begin

                    accum1[15:8]    <=  8'd0;

                    checksum_state  <=  2'd0;

                end

            end

            1:begin

                accum1[7:0]     <=  I_icmp_pkg_data;

                checksum_state  <=  2'd2;

            end

            2:begin

                if(!I_icmp_pkg_valid) begin

                    checksum_state      <=  2'd3;

                    if((tmp_accum1[15:0] + tmp_accum1[31:16]) != 16'hffff)

                        checksum_correct    <=  1'b0;

                    else

                        checksum_correct    <=  1'b1;

                end

                else begin

                    accum2          <=  tmp_accum1;

                    accum1[15:8]    <=  I_icmp_pkg_data;

                    checksum_state  <=  2'd1;

                end

            end

            3:begin

                accum1          <=  16'd0;

                accum2          <=  32'd0;

                checksum_state  <=  2'd0;

            end

        endcase

    end

end

 

将报文的附加数据存入FIFO中,并向icmp_pkg_tx模块发送ping应答信息,请求发送ping应答包。

//以下模块完成ICMP报文包echo ping应答的请求,并且先缓存到ip_layer的FIFO中

always@(posedge I_clk or posedge I_reset) begin

    if(I_reset) begin

        cnt                         <=  4'd0;

        type1                       <=  8'd0;

        code                        <=  8'd0;

        echo_data_cnt               <=  10'd0;

        checksum                    <=  16'd0;

        O_icmp_req_en               <=  1'b0;

        O_icmp_req_id               <=  16'd0;

        O_icmp_req_sq_num           <=  16'd0;

        O_icmp_ping_echo_data_valid <=  1'b0;

        O_icmp_ping_echo_data       <=  8'd0;

        O_icmp_req_checksum         <=  16'd0;

        STATE                       <=  RECORD_ICMP_HEADER;

    end

    else begin

        case(STATE)

            RECORD_ICMP_HEADER:begin

                O_icmp_req_en       <=  1'b0;

                echo_data_cnt       <=  10'd0;

                if(I_icmp_pkg_valid)        //ICMP报文有效

                    case(cnt)

                        0:  begin   type1                   <=  I_icmp_pkg_data;    cnt <=  cnt + 1'b1;end  //ICMP报文首部-类型:8位数表示错误类型的差错报文或者查询类型的报告报文,一般是查询报文(0代表回显应答(ping应答);1代表查

                        1:  begin   code                    <=  I_icmp_pkg_data;    cnt <=  cnt + 1'b1;end  //ICMP报文首部-类型:代码占用8位数据,根据ICMP差错报文的类型,进一步分析错误的原因

                        2:  begin   checksum[15:8]          <=  I_icmp_pkg_data;    cnt <=  cnt + 1'b1;end  //ICMP报文首部-校验和:16位校验和的计算方法与IP首部校验和计算方法一致,该校验和需要对ICMP首部和ICMP数据做校验

                        3:  begin   checksum[7:0]           <=  I_icmp_pkg_data;    cnt <=  cnt + 1'b1;end  //ICMP报文首部-校验和:16位校验和的计算方法与IP首部校验和计算方法一致,该校验和需要对ICMP首部和ICMP数据做校验

                        4:  begin   O_icmp_req_id[15:8]     <=  I_icmp_pkg_data;    cnt <=  cnt + 1'b1;end  //ICMP报文首部-标识符:16位标识符对每一个发送的数据报进行标识

                        5:  begin   O_icmp_req_id[7:0]      <=  I_icmp_pkg_data;    cnt <=  cnt + 1'b1;end  //ICMP报文首部-标识符:16位标识符对每一个发送的数据报进行标识

                        6:  begin   O_icmp_req_sq_num[15:8] <=  I_icmp_pkg_data;    cnt <=  cnt + 1'b1;end  //ICMP报文首部-序列号:16位对发送的每一个数据报文进行编号

                        7:  begin   O_icmp_req_sq_num[7:0]  <=  I_icmp_pkg_data;    cnt <=  cnt + 1'b1;end  //ICMP报文首部-序列号:16位对发送的每一个数据报文进行编号

                        8:  begin

                            if(type1 == PING_REQUEST && code == 8'h00) begin        //如果是远端主机发的ping请求包,那么本地主机需要返回一个ping应答包      

                                O_icmp_ping_echo_data_valid <=  1'b1;               //ping应答有效

                                O_icmp_ping_echo_data       <=  I_icmp_pkg_data;

                            end

                            else begin

                                O_icmp_ping_echo_data_valid <=  1'b0;

                                O_icmp_ping_echo_data       <=  8'd0;

                            end

                            cnt     <=  4'd0;

                            STATE   <=  WAIT_PACKET_END;

                        end

                        default:    STATE   <=  RECORD_ICMP_HEADER;

                    endcase

                else

                    STATE   <=  RECORD_ICMP_HEADER;

            end

            WAIT_PACKET_END:begin

                if(I_icmp_pkg_valid) begin          //继续接收ICMP 报文

                    if(O_icmp_ping_echo_data_valid) //ping应答有效

                        echo_data_cnt   <=  echo_data_cnt + 1'b1;//ICMP包计数器

                    else

                        echo_data_cnt   <=  10'd0;

                    O_icmp_ping_echo_data_valid <=  O_icmp_ping_echo_data_valid;

                    O_icmp_ping_echo_data       <=  I_icmp_pkg_data;

                    STATE               <=  WAIT_PACKET_END;

                end

                else begin

                    if(O_icmp_ping_echo_data_valid) begin

                        O_icmp_req_en       <=  1'b1;           //通知ip_tx模块接收到ICMP报文包ping请求,并且发送一个echo ping应答

                        O_icmp_req_checksum <=  checksum_temp;  //输出校验和

                    end

                    else begin

                        O_icmp_req_en       <=  1'b0;

                        O_icmp_req_checksum <=  16'd0;                      

                    end

                    echo_data_cnt               <=  echo_data_cnt;

                    O_icmp_ping_echo_data_valid <=  1'b0;

                    O_icmp_ping_echo_data       <=  8'd0;

                    STATE                       <=  RECORD_ICMP_HEADER;

                end

            end

        endcase

    end

end

 

3.6.2 ICMP发送模块

该模块收到icmp_pkg_ctrl模块发送的ping应答包使能信号,寄存接收到的信息包括标识符、序列号、校验和、地址和数据长度,通过计数器添加报文头部,再将附加数据从FIFO中读出,组成ping应答包发送到ip_tx模块。

always@(posedge I_clk or posedge I_reset) begin

    if(I_reset) begin

        cnt1                    <=  4'd0;

        cnt2                    <=  10'd0;

        request_id              <=  16'd0;

        request_sq_num          <=  16'd0;

        request_ip_taddress     <=  32'd0;

        checksum                <=  16'd0;

        echo_data_length        <=  10'd0;

        O_icmp_pkg_req          <=  1'b0;

        O_icmp_pkg_valid        <=  1'b0;

        O_icmp_pkg_data         <=  8'd0;

        O_icmp_ping_echo_ren    <=  1'b0;

        STATE                   <=  WAIT_ICMP_PACKET;

    end

    else begin

        case(STATE)

            WAIT_ICMP_PACKET:begin

                if(I_icmp_req_en) begin//当接收到ICMP echo ping包,先保存该包的基本信息到寄存器

                    request_id              <=  I_icmp_req_id;              //ICMP包的标识符

                    request_sq_num          <=  I_icmp_req_sq_num;          //ICMP包的序列号

                    request_ip_taddress     <=  I_icmp_req_ip_addr;         //ICMP包的地址

                    checksum                <=  I_icmp_req_checksum;        //ICMP包的校验和

                    echo_data_length        <=  I_icmp_ping_echo_data_len;  //ICMP包的长度      

                    O_icmp_pkg_req          <=  1'b1;                       //请求ip_tx模块发送部分,发送ICMP报文

                    STATE                   <=  WAIT_PACKET_SEND;           //发送ICMP包状态

                end

                else begin

                    request_id              <=  16'd0;

                    request_sq_num          <=  16'd0;

                    request_ip_taddress     <=  32'd0;

                    checksum                <=  16'd0;

                    echo_data_length        <=  10'd0;

                    O_icmp_pkg_req          <=  1'b0;

                    STATE                   <=  WAIT_ICMP_PACKET;                  

                end

            end

            WAIT_PACKET_SEND:begin

                if(I_icmp_pkg_busy) begin//该信号来自ip_tx模块,当有效代表ip_tx模块已经开始准备发送ICMP包,这里需要对ip_tx代码部分时序逻辑确保数据正确给到ip_tx模块

                    O_icmp_pkg_req          <=  1'b0;

                    O_icmp_pkg_valid        <=  1'b1;

                    O_icmp_pkg_data         <=  PING_REPLY_TYPE;//回显应答(ping应答)的类型

                    STATE                   <=  SEND_PACKET;

                end

                else begin

                    O_icmp_pkg_req          <=  1'b1;

                    O_icmp_pkg_valid        <=  1'b0;

                    O_icmp_pkg_data         <=  8'd0;

                    STATE                   <=  WAIT_PACKET_SEND;

                end

            end

            SEND_PACKET:begin

                case(cnt1)

                    0   :begin  O_icmp_pkg_data <=  8'h00;                  cnt1    <=  cnt1 + 1'b1;end//回显应答(ping应答)的代码

                    1   :begin  O_icmp_pkg_data <=  checksum[15:8];         cnt1    <=  cnt1 + 1'b1;end//ICMP报文包校验和,直接获取远程主机发送的Ping包校验和

                    2   :begin  O_icmp_pkg_data <=  checksum[7:0];          cnt1    <=  cnt1 + 1'b1;end//ICMP报文包校验和,直接获取远程主机发送的Ping包校验和

                    3   :begin  O_icmp_pkg_data <=  request_id[15:8];       cnt1    <=  cnt1 + 1'b1;end//ICMP报文标识符,直接获取远程主机发送的Ping包标识符

                    4   :begin  O_icmp_pkg_data <=  request_id[7:0];        cnt1    <=  cnt1 + 1'b1;end//ICMP报文标识符,直接获取远程主机发送的Ping包标识符

                    5   :begin  O_icmp_pkg_data <=  request_sq_num[15:8];   cnt1    <=  cnt1 + 1'b1;end//ICMP报文编码,直接获取远程主机发送的Ping序列号

                    6   :begin  //从echo FIFO中读取ICMP echo报文的数据部分

                        O_icmp_pkg_data         <=  request_sq_num[7:0];    

                        cnt1                    <=  cnt1 + 1'b1;

                        O_icmp_ping_echo_ren    <=  1'b1;

                    end

                    7   :begin//ICMP报文包的数据有效部分

                        O_icmp_pkg_valid        <=  1'b1;

                        O_icmp_pkg_data         <=  I_icmp_ping_echo_data;

                        if(cnt2 == (echo_data_length - 1)) begin

                            cnt2                    <=  10'd0;

                            O_icmp_ping_echo_ren    <=  1'b0;

                            cnt1                    <=  cnt1 + 1'b1;

                        end

                        else begin

                            cnt2                    <=  cnt2 + 1'b1;

                            O_icmp_ping_echo_ren    <=  1'b1;

                            cnt1                    <=  cnt1;                          

                        end

                    end

                    8   :begin

                        cnt1                    <=  4'd0;

                        O_icmp_pkg_data         <=  8'd0;

                        O_icmp_pkg_valid        <=  1'b0;

                        STATE                   <=  WAIT_ICMP_PACKET;

                    end

                    default:;

                endcase

            end

        endcase

    end

end

标签:UDP,end,DR1,begin,echo,pkg,安路,icmp,data
From: https://www.cnblogs.com/milianke/p/18351347

相关文章

  • [米联客-安路飞龙DR1-FPSOC] UDP通信篇连载-01 以太网协议介绍
    软件版本:Anlogic-TD5.9.1-DR1_ES1.1操作系统:WIN1064bit硬件平台:适用安路(Anlogic)FPGA实验平台:米联客-MLK-L1-CZ06-DR1M90G开发板板卡获取平台:https://milianke.tmall.com/登录"米联客"FPGA社区http://www.uisrc.com视频课程、答疑解惑! ​1概述本文介绍了基于XILIN......
  • [米联客-安路飞龙DR1-FPSOC] UDP通信篇连载-02 MAC层程序设计
    软件版本:Anlogic-TD5.9.1-DR1_ES1.1操作系统:WIN1064bit硬件平台:适用安路(Anlogic)FPGA实验平台:米联客-MLK-L1-CZ06-DR1M90G开发板板卡获取平台:https://milianke.tmall.com/登录"米联客"FPGA社区http://www.uisrc.com视频课程、答疑解惑! 3程序设计前面我们介绍了以太......
  • [米联客-安路飞龙DR1-FPSOC] SDK入门篇连载-02 FPSoc程序固化入门
    软件版本:Anlogic-TD5.9.1-DR1_ES1.1操作系统:WIN1064bit硬件平台:适用安路(Anlogic)FPGA实验平台:米联客-MLK-L1-CZ06-DR1M90G开发板板卡获取平台:https://milianke.tmall.com/登录"米联客"FPGA社区http://www.uisrc.com视频课程、答疑解惑! 1概述在应用的调试过程中,我们......
  • [米联客-安路飞龙DR1-FPSOC] SDK入门篇连载-03 GPIO PS/PL实验
    软件版本:Anlogic-TD5.9.1-DR1_ES1.1操作系统:WIN1064bit硬件平台:适用安路(Anlogic)FPGA实验平台:米联客-MLK-L1-CZ06-DR1M90G开发板板卡获取平台:https://milianke.tmall.com/登录"米联客"FPGA社区http://www.uisrc.com视频课程、答疑解惑! 1概述本课对FPSoC芯片GPIO进......
  • [米联客-安路飞龙DR1-FPSOC] SDK入门篇连载-01 FPSoc开发入门
    软件版本:Anlogic-TD5.9.1-DR1_ES1.1操作系统:WIN1064bit硬件平台:适用安路(Anlogic)FPGA实验平台:米联客-MLK-L1-CZ06-DR1M90G开发板板卡获取平台:https://milianke.tmall.com/登录"米联客"FPGA社区http://www.uisrc.com视频课程、答疑解惑!1概述1.1背景介绍由于FPS......
  • 简析传输层协议——TCP、UDP协议
    TCP/IP协议族的传输层协议TCP(TransmissionControlProtocol)传输控制协议UDP(UserDatagramProtocol)用户数据报协议TCP协议介绍:TCP是面向连接的、可靠的进程到进程通信的协议TCP提供全双工服务,即数据可在同一时间双向传输TCP报文段:TCP将若干个字节构成一个分......
  • 为什么 DDoS 攻击偏爱使用 TCP 和 UDP 包?
    DistributedDenialofService(DDoS)攻击是指攻击者利用多个计算机系统或网络设备(通常是被恶意软件感染的计算机,被称为“僵尸网络”)来淹没目标服务器的资源,导致合法用户无法访问服务。TCP和UDP是两种最常见的用于DDoS攻击的网络协议。1.TCP和UDP的特性TCP(Tr......
  • Transport Layer Security for UDP&TCP(TLS/DTLS1.2)
    参考文章:https://blog.csdn.net/alwaysrun/article/details/89076492 https://www.jianshu.com/p/fd0a624d0912 https://cloud.tencent.com/developer/article/1928677文档:https://www.rfc-editor.org/rfc/rfc6347  https://www.rfc-editor.org/rfc/rfc52461.SSL/TL......
  • [米联客-安路飞龙DR1-FPSOC] FPGA基础篇连载-26 RS485串口程序收发环路设计
    软件版本:Anlogic-TD5.9.1-DR1_ES1.1操作系统:WIN1064bit硬件平台:适用安路(Anlogic)FPGA实验平台:米联客-MLK-L1-CZ06-DR1M90G开发板板卡获取平台:https://milianke.tmall.com/登录"米联客"FPGA社区http://www.uisrc.com视频课程、答疑解惑! 1概述在前面的课程中,我......