【正点原子FPGA连载】 第三十三章基于lwip的tftp server实验 摘自【正点原子】DFZU2EG_4EV MPSoC之嵌入式Vitis开发指南

第三十三章基于lwip的tftp server实验

文件传输是网络环境中的一项基本应用,其作用是将一台电子设备中的文件传输到另一台可能相距很远的电子设备中。TFTP作为TCP/IP协议族中的一个用来在客户机与服务器之间进行文件传输的协议,常用于无盘工作站、路由器以及远程测控设备从主机上获取引导配置文件,实现远程升级。由于TFTP简单且易实现,本实验我们使用lwip协议栈实现TFTP Server的功能。本章包括以下几个部分:
3333.1简介
33.2实验任务
33.3硬件设计
33.4软件设计
33.5下载验证

33.1简介

一、TFTP简介(基于RFC1350版本)
简单文件传输协议TFTP (Trivial File Transfer Protocol) 是TCP/IP协议族中的一个用来在客户机与服务器之间进行简单文件传输,基于UDP实现的应用层协议,提供不复杂、开销不大的文件传输服务,端口号为 69。为了保证文件可靠传输TFTP有自己的差错改正措施。TFTP 只支持文件传输、不支持交互、没有庞大的命令集,也没有目录列表功能,以及不能对用户进行身份鉴别。
与常用的文件传送协议 FTP (File Transfer Protocol) 相比,FTP基于TCP协议,提供交互式的访问,允许客户指明文件的类型与格式、允许执行对目录和文件的访问,并且可以完成特定类型的目录操作以及需要进行身份验证。
可以说FTP是完整的、面向会话、常规用途的文件传输协议,而TFTP相当于用作特殊目的简化版的FTP。
TFTP的主要优点有两个。
第一,TFTP可用于UDP环境。例如,当需要将程序或文件同时向许多机器下载时就往往需要使用TFTP。
第二,TFTP代码所占的内存较小。这对较小的计算机或某些特殊用途的设备(如无盘工作站等)是很重要的。这些设备不需要硬盘,只需要固化了TFTP、UDP和IP的小容量只读存储器即可。当接通电源后,设备执行只读存储器中的代码,在网络上广播一个TFTP请求。网络上的TFTP服务器就发送响应,其中包括可执行二进制程序。设备收到此文件后将其放入内存,然后开始运行程序。这种方式增加了灵活性,也减少了开销。
TFTP的主要特点如下:
(1)每次传送的数据报文中有512字节的数据,但最后一次可不足512字节。
(2)数据报文按序编号,从1开始。
(3)支持ASCII码或二进制传送。
(4)可对文件进行读或写。
(5)使用很简单的首部。
(6)实现简单而不是高的系统吞吐量
二、TFTP的五种报文
TFTP的报文格式如图 34.1.1所示,可以看到TFTP有五种报文,每种报文有不同的操作码,这五种报文分别是:RRQ、WRQ、DATA、ACK和ERROR报文。下面我们简单的介绍下这五种报文。
RRQ/WRQ报文
模式字段中,包含两种字符串中的一种,"netascii"表示ASCII文件,"octet"表示二进制文件。对于RRQ,客户向TFTP服务器发送读请求后,服务器返回一个块编号为1的DATA报文。而对于WRQ,客户向TFTP服务器发送写请求后,服务器返回的是块编号为1的ACK报文。总之,不管是RRQ还是WRQ,接收DATA数据的一方发送ACK确认,而发送DATA数据的一方只负责发送数据。
在这里插入图片描述

图 34.1.1 TFTP报文格式
a)DATA报文
发送方用于传送数据块。所有的块都用数字顺序编码,从1开始。在所有的DATA报文中,这个块必须准确地等于512Byte,但最后一个块可以小于或等于512Byte。当发送的DATA报文中数据部分的长度小于512Byte,表示DATA报文发送完毕,所以小于数据部分512Byte的DATA数据报可以作为文件结束的标志。特殊的情况是,当文件中的数据正好是512Byte的整数倍时,那么发送端必须再发送一个具有数据部分为0Byte的额外的DATA数据块以表示传输的结束。数据可以采用ASCII码或二进制来传送。
b)ACK报文
块号表示它所收到的块号(不是下一个期待的块号,这与TCP中的ACK序号不同)。特殊情况是,当客户向服务器发送一个WRQ请求后,服务器返回给客户的是一个块号为0的ACK报文,表示服务器已经准备好了接收来自客户的数据报。
c)EEROR报文(差错报文)
ERROR报文既可以由客户发送,也可以由服务器发送,当一条连接(如读连接或写连接)不能建立或在数据传输中出现问题时使用。差错码定义了差错的类型,差错信息是一个可变字节,包含原文中的差错数据。
从上面的报文格式中可以看出,TFTP报文没有差错检验和字段,所以接收端检验数据是否出现差错的唯一方法是通过该TFTP数据报的UDP首部中的检验和字段。
三、TFTP传输过程
以TFTP客户端向 TFTP 服务器发送写请求为例,说明整个过程。
1)服务器使用默认端口号69被动打开连接;
2)客户主动打开连接,向服务器进程发送WRQ报文,报文中包含写入文件的文件名;
3)TFTP服务器进程选择一个新的端口和TFTP客户进程进行通信,并向TFTP客户进程发送块编号为0的的ACK报文;
4)客户端收到服务器的ACK报文后发送DATA报文,数据段为512Byte,少于512Byte表明是文件的最后的数据,块编号逐次递增;
5)TFTP服务器校验收到的DATA报文的块编号,如果校验正确则将数据写入文件,并发送ACK报文表明已接收到数据,ACK报文的块编号为本次接收的DATA报文的块编号。另外还判断数据段长度是否小于512 Byte,小于则表明文件传输完成,关闭连接,如果等于512Byte,则重复步骤4-5,直到所有请求的数据发送完毕。
从上面的传输过程可以看出,TFTP 是一种类似于停止等待协议(不是真正的停止等待协议,在停止等待协议中,接收方发送的 ack 表示期望收到的下一个分组,而在 TFTP 的 ACK 报文中,ACK的块号表示的是本次成功收到的数据块,而不是下一个期望的下一个数据块)。TFTP 客户端只有收到服务器的确认报文ACK后才会接着向服务器发送新的数据。
另外需要注意的是TFTP 协议中,用于读文件的连接和用于写文件的连接的建立方式不同:建立读连接的时候,客户首先向服务器发送 RRQ 读报文,服务器收到该报文后,直接发回给该客户 DATA 报文,并且包含第一个数据块(块号为 1)。而建立写连接的时候,客户首先先服务器发送 WRQ 写报文,服务器收到该报文后,则发回给客户 ACK 报文,使用的块号为 0;当然上面两种情况如果遇到请求报文出错时,均会发回 ERROR 报文作为响应。
33.2实验任务
本章的实验任务是使用LWIP协议栈搭建TFTP服务器,PC电脑上的客户端可以从TFTP服务器读取文件也可向TFTP服务器写入文件,文件存放在SD卡中。
33.3硬件设计
根据实验任务我们可以画出本次实验的系统框图,如下图所示:
在这里插入图片描述

图 34.3.1 系统框图
在图 34.3.1中,UART用于打印程序相关的信息,LWIP通过以太网传输数据,TF卡用于存放文件,包括服务器创建的文件和客户端写入的文件。
step1:创建Vivado工程
本次实验的硬件设计可以在《lwip echo server》实验的基础上添加SD卡。
1-1 我们先打开《lwip echo server》实验的Vivado工程,打开后将工程另存为“lwip_tftp_server”工程,然后点击“OK”按钮。
step2:使用IP Integrator创建Processing System
2-1 在Vivado界面左侧的Flow Navigator中,点击IP INTEGRATOR下的Open Block Design以打开Diagram窗口。
2-2 在打开的下图Diagram窗口,双击打开Zynq UltraScale+ MPSOC重定义窗口。
在这里插入图片描述

图 34.3.2 重定义Zynq UltraScale+ MPSOC
2-3 在下图所示的重定义窗口,如同《SD卡读写TXT文本实验》那样配置SD卡。点击左侧的I/O Configuration,在右侧的界面中找到SD卡控制器配置选项,并勾选“SD1”,Slot Type选择SD2.0,然后选择MIO46…51,并勾选CD选项,然后点击“OK”,如下图所示。
在这里插入图片描述

图 34.3.3 SD卡接口配置界面
2-4 由于不需要添加其它IP,点击Validate Design验证成功后,按Ctrl+S快捷键保存Diagram。此时我们的第二步完成,进入第三步
step3:生成顶层HDL
在sources面板中,右键点击Block Design设计文件“design_1.bd”,然后执行“Generate Output Products”。
step4:生成Bitstream文件并导出到VITIS
由于本实验未用到PL部分,所以无需生成Bitstream文件,只需导出Hardware硬件平台文件即可。如果使用到PL,则需要添加引脚约束以及对该系统进行综合、实现并生成Bitstream文件。
4-1 导出硬件。
在菜单栏中选择 File > Export > Export hardware。注意导出路径的选择并取消勾选“Include bitstream”,然后点击“OK”按钮。
新建vitis文件夹,将导出的平台文件移动到该文件夹下。
4-2 硬件导出完成后,选择菜单Tools->Launch Vitis,指定工作空间,启动Vitis开发环境。
33.4软件设计
下面我们开始第五步——创建应用工程。
step5:在Vitis中创建应用工程
5-1在菜单栏中选择“File->New->Application Project”,
在弹出的界面中,输入工程名“lwip_tftp_server”,点击“Next >”,添加应用平台文件,添加完成后,接下来依次点击“Next>”,直到弹出选择模板界面,选择“Empty Application”空应用工程,然后点击“Finish”按钮。
5-2 设置板级支持包(BSP)。
打开板级支持包设置界面,具体步骤可参考前面的实验。在弹出的BSP设置界面,勾选“lwip212”和“xilffs”以启用lwip和文件系统,如图 34.4.1所示。
如果没有开启DHCP服务可以开启DHCP服务,点击standalone下的lwip212,在右侧界面中到“dhcp_options”,将其下的两个选项的“Value”设置为“true”,如图 34.4.2所示。
在这里插入图片描述

图 34.4.1 BSP的设置界面
在这里插入图片描述

图 34.4.2 开启DHCP
5-3 由于Xilinx提供的lwip例程里有tftp server的源代码,所以我们无需自己手动编写,直接添加即可。
双击硬件平台工程design_1_wrapper下的platform.spr,在打开的界面中,单击standalone on psu_cortexa53_0下的Board Support Package,然后展开Libraries。可以看到Libraries标题下有lwip211和xilffs两个库,单击lwip211库后的“Import Example”选项,如下图所示。
在这里插入图片描述

图 34.4.3 Import lwip Example
5-4 在弹出的下图所示界面中,点击下方的“Examples Directory”。
在这里插入图片描述

图 34.4.4打开例程文件目录
5-5 打开例程所在文件的目录,里面有Xilinx关于lwip的全部例程源文件。我们选择本次实验需要的源文件,如图 34.4.5所示,并单击鼠标右键选择复制。复制完成后,关闭该目录,并在打开的图 34.4.4界面中,点击“Cancel”退出。
在这里插入图片描述

图 34.4.5 例程所在文件的目录
5-6 单击Vitis软件的lwip_tftp_server/src目录,按下粘贴快捷键“Ctrl-v”,将复制的文件粘贴到该src目录下,如下图所示。
在这里插入图片描述

图 34.4.6 src目录
5-7 为了方便分析,我们将刚才复制到src目录的源文件重命名,主要是删除不需要的前缀,其中“lwip_example_tftpserver_common.h”改为“lwip_tftp_server.h”,然后进行编译,如下图所示:
在这里插入图片描述

图 34.4.7 删除不相关前缀后的src文件夹内容
5-8 现在我们打开main.c文件,为了方便分析源代码,在main.c文件中将带有下图箭头所指的预编译指令删除。
在这里插入图片描述

图 34.4.8 删除不需要的预编译指令
删除不适用的预编译指令后的main.c代码与我们《lwip echo server实验》的main.c代码基本相同,区别在于本次TFTP server实验没有使用IPv6,所以没有IPv6的预编译指令,其他完全相同,main.c代码讲解见《lwip echo server实验》。
5-9本实验可以说是在《lwip echo server实验》的基础上增加了文件系统,然后将Echo server的实现文件echo.c文件改写成了TFTP Server的实现文件。因而本实验的主要代码是TFTP Server的实现,该实现在lwip_tftp_server.h和lwip_tftp_server.c中,由于这两个文件的总代码有500多行,因此我们挑选部分代码进行讲解。此处以客户端写文件为例讲解lwip_tftp_server.c中的写文件实现源码。讲解以函数调用顺序进行。
首先我们看main函数中调用的start_application函数,该函数实现如下:

352 void start_application()
353 {
354     struct udp_pcb *pcb;
355     err_t err;
356 
357     //创建测试文件用于客户端读取
358     err = tftp_create_test_file();
359     if (err) {
360         xil_printf("Unable to create test file\r\n");
361         return;
362     }
363 
364     //创建新的UDP PCB
365     pcb = udp_new();
366     if (!pcb) {
367         xil_printf("Error creating PCB. Out of Memory\r\n");
368         return;
369     }
370 
371     //绑定端口
372     err = udp_bind(pcb, IP_ADDR_ANY, TFTP_PORT);
373     if (err != ERR_OK) {
374         xil_printf("Unable to bind to port %d; err %d\r\n",
375                 TFTP_PORT, err);
376         udp_remove(pcb);
377         return;
378     }
379 
380     //设置接收回调函数
381     udp_recv(pcb, (udp_recv_fn) tftp_server_recv_cb, NULL);
382 }

可以看到该函数首先通过调用tftp_create_test_file函数创建了测试文件,用于tftp客户端读取tftp服务器的文件数据,测试文件名为sample#.txt,其中“#”为数字1、2、3中的任一值,其文件内容为“----- This is a test file for TFTP server application -----”。如果不执行客户端的读取文件请求,可删除该函数的调用及其实现。
由于TFTP基于UDP协议,从start_application函数可以看到lwip中使用UDP协议很简单。首先通过udp_new函数创建一个新的UDP PCB,然后调用udp_bind函数绑定端口号,IP_ADDR_ANY表明为任意本地地址,TFTP_PORT是在lwip_tftp_server.h宏定义的端口号,其值为69,即TFTP的默认端口。最后调用udp_recv函数设置接收回调函数就完成了UDP服务的创建,服务端的功能即TFTP协议由回调函数实现。回调函数代码如下:

270 static void tftp_server_recv_cb(void *arg, struct udp_pcb *upcb, struct pbuf *p,
271         ip_addr_t *ip, u16_t port)
272 {
273     tftp_opcode op = tftp_get_opcode(p->payload);
274     char fname[512];
275     struct udp_pcb *pcb;
276     err_t err;
277 
278     pcb = udp_new();
279     if (!pcb) {
280         xil_printf("Error creating PCB. Out of Memory\r\n");
281         goto cleanup;
282     }
283 
284     //绑定到端口0以接收下一个可用的空闲端口
285     err = udp_bind(pcb, IP_ADDR_ANY, 0);
286     if (err != ERR_OK) {
287         xil_printf("Unable to bind to port %d; err %d\r\n", port, err);
288         goto cleanup;
289     }
290 
291     switch (op) {
292     case TFTP_RRQ:
293         //从payload中获取文件名
294         strcpy(fname, p->payload + FIL_NAME_OFFSET);
295         printf("TFTP RRQ (read request): %s\r\n", fname);
296         tftp_process_read(pcb, ip, port, fname);
297         break;
298     case TFTP_WRQ:
299         //从payload中获取文件名
300         strcpy(fname, p->payload + FIL_NAME_OFFSET);
301         printf("TFTP WRQ (write request): %s\r\n", fname);
302         tftp_process_write(pcb, ip, port, fname);
303         break;
304     default:
305         //发送访问冲突消息
306         tftp_send_error_packet(pcb, ip, port, TFTP_ERR_ILLEGALOP);
307         printf("TFTP unknown request op: %d\r\n\r\n", op);
308         udp_remove(pcb);
309         break;
310     }
311 
312 cleanup:
313     pbuf_free(p);
314 }

当TFTP客户端发起写入或读取文件的请求后,lwip协议栈调用回调函数tftp_server_recv_cb。该回调函数通过tftp_get_opcode宏获取客户端发送报文的操作码,不同的操作码执行该函数switch分支中的不同的case,如对于写入文件请求,则执行“case TFTP_WRQ”分支语句,该分支语句调用TFTP处理写文件请求函数tftp_process_write,该函数实现如下:

233 //TFTP 处理写文件请求
234 static int tftp_process_write(struct udp_pcb *pcb, ip_addr_t *ip, int port,
235         char *fname)
236 {
237     tftp_connection_args *conn;
238     FIL w_fil;
239     FRESULT Res;
240 
241     Res = f_open(&w_fil, fname, FA_CREATE_ALWAYS | FA_WRITE);
242     if (Res) {
243         xil_printf("Unable to open file %s for writing %d\r\n", fname,
244                Res);
245         tftp_send_error_packet(pcb, ip, port, TFTP_ERR_DISKFULL);
246         udp_remove(pcb);
247         return -1;
248     }
249 
250     conn = mem_malloc(sizeof *conn);
251     if (!conn) {
252         xil_printf("Unable to allocate memory for tftp conn\r\n");
253         tftp_send_error_packet(pcb, ip, port, TFTP_ERR_DISKFULL);
254         udp_remove(pcb);
255         return -1;
256     }
257 
258     memcpy(&conn->fil, &w_fil, sizeof(w_fil));
259     conn->block = 0;
260 
261     //为该pcb设置接收回调
262     udp_recv(pcb, (udp_recv_fn) tftp_server_write_req_recv_cb, conn);
263 
264     //通过发送第一个ACK来启动传输
265     tftp_send_ack_packet(pcb, ip, port, conn->block);
266 
267     return 0;
268 }

该函数首先在文件系统中创建一个文件,文件名为客户端写入的文件名,然后为新创建的UDP PCB设置接收回调函数,用于处理后面接收客户端传入的文件,最后发送块编号为0的ACK报文以应答客户端启动传输。TFTP写入请求的接收回调函数实现如下:

187 //TFTP 写入请求的接收回调函数
188 static void tftp_server_write_req_recv_cb(void *_args, struct udp_pcb *upcb,
189         struct pbuf *p, ip_addr_t *addr, u16_t port)
190 {
191     ip_addr_t ip = *addr;
192     tftp_connection_args *args = (tftp_connection_args *)_args;
193 
194     if (p->len != p->tot_len) {
195         xil_printf("TFTP_WRQ: Tftp server does not support "
196                 "chained pbufs\r\n");
197         pbuf_free(p);
198         return;
199     }
200 
201     //确保数据块是我们所期望的
202     if ((p->len >= TFTP_PACKET_HDR_LEN) &&
203         (tftp_get_block_value(p->payload) ==
204          (u16_t) (args->block + 1))) {
205 
206         //将接收的数据写入文件
207         unsigned int n;
208         f_write(&args->fil, p->payload + TFTP_PACKET_HDR_LEN,
209                 p->len - TFTP_PACKET_HDR_LEN, &n);
210         if (n != p->len - TFTP_PACKET_HDR_LEN) {
211             xil_printf("TFTP_WRQ: Write to file error\r\n");
212             tftp_send_error_packet(upcb, &ip, port,
213                         TFTP_ERR_DISKFULL);
214             pbuf_free(p);
215             return tftp_cleanup(upcb, args);
216         }
217         args->block++;
218     }
219 
220     tftp_send_ack_packet(upcb, &ip, port, args->block);
221 
222     
223      
224      //如果接收到的数据段长度小于指定的字节数,则表明已经接收了整个文件,因此可以退出
225     if (p->len < TFTP_DATA_PACKET_MSG_LEN) {
226         xil_printf("TFTP_WRQ: Transfer completed\r\n\r\n");
227         return tftp_cleanup(upcb, args);
228     }
229 
230     pbuf_free(p);
231 }

从该回调函数可以看到,TFTP服务端对客户端发送的数据报文的块编号进行校验,如果不是我们期望的块编号就重发上一次发送的ACK报文,如果是期望的块编号,就将数据写入文件中,然后递增块编号,并发送ACK报文给客户端以确认收到数据。
在该函数的最后判断接收到的数据段长度是否小于指定的字节数TFTP_DATA_PACKET_MSG_LEN,如果是,则表明已经接收了整个文件,因此可以结束连接。TFTP_DATA_PACKET_MSG_LEN在lwip_tftp_server.h宏定义为512。
以上大概的讲解了TFTP Server接收客户端写入文件的实现。下面我们进行实际操作,看看TFTP客户端是否能向服务器写入文件。
33.5下载验证
首先我们将下载器与MPSOC开发板上的JTAG接口连接,下载器另外一端与电脑连接。然后使用USB连接线将USB UART接口与电脑连接,用于串口通信。使用网线一端连接MPSOC开发板的以太网接口,另一端与电脑或路由器连接。连接完成后,在开发板上插入TF 卡(SD 卡插槽位于开发板背面)。最后连接开发板的电源,给开发板上电。如下图所示:
在这里插入图片描述

图 34.5.1 MPSOC开发板实物图
现在进入最后一步。
step6:板级验证
6-1 在Vitis软件的下方的Vitis Serial Terminal窗口中点击右上角的加号连接串口。
6-2 下载程序。下载完成后,可以看到串口打印的结果如下:
在这里插入图片描述

图 34.5.2 显示打印结果
其中“File system initialization successful”表明SD卡可以正常工作。打印的最后一句表明了该实验如何使用。由于是TFTP服务器实验,所以我们需要TFTP客户端,可以从网上下载,也可以使用Windows系统的CMD命令行界面,如果开启了TFTP客户端,开启方法见步骤6-5。
6-3 下面我们先创建一个文件用来传输到TFTP服务器。文件存放位置任意,文件内容任意。
我们在Vivado工程目录新建一个名为“test”的文件夹,里面新建一个名为testfile.txt的文件,文件内容为“这只是一个测试文件。”,如下图所示:
在这里插入图片描述

图 34.5.3 新建一个名为test_file.txt的文件
6-4 我们打开电脑的CMD(按win+r键后输入cmd),然后输入命令“cd /D F:\ZYNQ\Embedded_System\lwip_tftp_server\test”切换到 “F:\ZYNQ\Embedded_System\lwip_tftp_server\test”目录下,如下图所示:
在这里插入图片描述

图 34.5.4 切换到上传文件所在的目录
然后输入“tftp -i 192.168.1.10 PUT testfile.txt”命令,回车,会显示传输成功字样,如下图所示:
在这里插入图片描述

图 34.5.5 进行tftp连接
此时VITIS串口终端也会打印如下信息:
在这里插入图片描述

图 34.5.6 串口终端打印写入完成信息
如果回车后出现像下图所示界面所示“tftp不是内部或外部命令,也不是可运行的程序或批处理文件”,则表明未开启Windows的tftp客户端功能,开启方式见6-5。
在这里插入图片描述

图 34.5.7 未启用tftp客户端时的界面
如果回车后出现连接请求失败或者串口终端打印“Error creating PCB.Out of Memory”信息时,检查防火墙是否全部关闭。
向服务器写入文件刚才测试完成了,现在测试从服务器端读取文件,可以读取刚才写入的文件,也可以读取服务器程序创建的测试文件。下面我们以读取服务器程序创建的测试文件为例,进行读取文件测试。
在CMD中输入“tftp -i 192.168.1.10 GET sample1.txt”命令,然后回车,会显示传输成功字样,如下图所示:
在这里插入图片描述

图 34.5.8 输入读取文件命令
此时VITIS串口终端也会打印如下信息:
在这里插入图片描述

图 34.5.9 读取成功
此时我们打开test文件夹,会看到其中新增了sample1.txt,双击打开,其内容如下:
在这里插入图片描述

图 34.5.10 读取的sample1.txt文件
可以看到读取文件测试成功。现在我们把SD卡插到电脑上,查看其内容如下:
在这里插入图片描述

图 34.5.11 SD卡上的文件
可以看到客户端上传给TFTP服务器的文件确实写到SD卡中。
6-5 下面我们介绍一下如何开启Windows的tftp客户端功能。在Win10或Win7系统中,按“Win+r”快捷键后,在下图所示界面中输入“control”。
在这里插入图片描述

图 34.5.12 打开控制面板界面
进入下图所示控制面板界面,将查看方式设置为“类别”,单击“程序”下的“卸载程序”,如下图所示:
在这里插入图片描述

图 34.5.13 点击进入“程序和功能”界面
在弹出的界面中,单击“启用或关闭Windows功能”,如下图所示:
在这里插入图片描述

图 34.5.14 点击“启用或关闭Windows功能”
在弹出的“Windows功能”界面中,找到“Tftp Client”,并勾选,如下图所示:
在这里插入图片描述

图 34.5.15 勾选tftp client
单击确定后,如果出现“Windows需要重启电脑才能完成安装所请求的更改”字样,重新启动电脑即可。现在 Windows的tftp客户端服务已启用。
至此,本实验完成。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Java基础:笔试题

文章目录Java 基础题目1. 如下代码输出什么&#xff1f;2. 当输入为2的时候返回值是多少?3. 如下代码输出值为多少?4. 给出一个排序好的数组&#xff1a;{1,2,2,3,4,5,6,7,8,9} 和一个数&#xff0c;求数组中连续元素的和等于所给数的子数组解析第一题第二题第三题第四题方案…

复制带随机指针的复杂链表

目录一、题目题目链接二、题目分析三、解题思路四、解题步骤4.1 复制结点并链接到对应原节点的后面4.2 处理复制的结点的随机指针random4.3 分离复制的链表结点和原链表结点并重新链接成为链表五、参考代码六、总结一、题目题目链接 ​​​​ ​ 题目链接&#xff1a;https://…

【二】一起算法---队列:STL queue、手写循环队列、双端队列和单调队列、优先队列

纸上得来终觉浅&#xff0c;绝知此事要躬行。大家好&#xff01;我是霜淮子&#xff0c;欢迎订阅我的专栏《算法系列》。 学习经典算法和经典代码&#xff0c;建立算法思维&#xff1b;大量编码让代码成为我们大脑的一部分。 ⭐️已更系列 1、基础数据结构 1.1、链表➡传送门 1…

【微信小程序】-- 分包 - 独立分包 分包预下载(四十五)

&#x1f48c; 所属专栏&#xff1a;【微信小程序开发教程】 &#x1f600; 作  者&#xff1a;我是夜阑的狗&#x1f436; &#x1f680; 个人简介&#xff1a;一个正在努力学技术的CV工程师&#xff0c;专注基础和实战分享 &#xff0c;欢迎咨询&#xff01; &…

Java 到底是值传递还是引用传递?

C 语言是很多变成语言的母胎&#xff0c;包括 Java。对于 C 语言来说&#xff0c;所有的方法参数都是通过 “值” 传递的&#xff0c;也就是说&#xff0c;传递给被调用方法的参数值存放在临时变量中&#xff0c;而不是存放在原来的变量中。这就意味着&#xff0c;被调用的方法…

Windows安装部署nginx

Windows安装部署nginx 1、官网下载安装包&#xff1a; 官网地址&#xff1a;nginx下载地址 下载好后&#xff0c;解压即可&#xff1a; 2、启动nginx&#xff1a; 启动nginx时&#xff0c;运行cmd&#xff0c;使用命令进行操作&#xff1b;不要直接双击nginx.exe&#xff0c…

【JavaEE】进程和线程

目录 1.2进程调度 1.2.1进程状态 1.2.2进程优先级 1.2.3进程的上下文 1.2.4进程的记账信息 2.线程 2.1线程的定义&#xff1a; 2.2为什么有线程 3.进程和线程的区别 什么进程&#xff1f;举一个很直观的例子&#xff0c;我们打开任务管理器&#xff0c;打开之后&…

自动驾驶TPM技术杂谈 ———— 超声波雷达系统测距

文章目录概述参数传播速度(v)测量误差飞行时间工作模式直接测量间接测量探测时序AK1AK2概述 超声波雷达是利用声波&#xff08;一种机械波&#xff0c;速度340m/s 空气&#xff09;在空气中传播&#xff0c;遇到障碍物会把部分声波进行折回&#xff0c;超声波雷达再对折回的声波…

C++修炼之筑基期第一层——认识类与对象

文章目录&#x1f337;专栏导读&#x1f337;什么是面向对象&#xff1f;&#x1f337;类的引入&#x1f337;什么是类&#x1f337;类的定义方式&#x1f337;类的访问限定符与封装&#x1f33a;访问限定符&#x1f33a;封装&#x1f337;类的作用域&#x1f337;类的实例化&a…

大环境不好,找工作太难?三面阿里,幸好做足了准备,已拿offer

三面大概九十分钟&#xff0c;问的东西很全面&#xff0c;需要做充足准备&#xff0c;就是除了概念以外问的有点懵逼了&#xff08;呜呜呜&#xff09;。回来之后把这些题目做了一个分类并整理出答案&#xff08;强迫症的我狂补知识&#xff09;分为软件测试基础、Python自动化…

【Java web】-转发和重定向

作者&#xff1a;学Java的冬瓜 博客主页&#xff1a;☀冬瓜的主页&#x1f319; 专栏&#xff1a;【Java Web】 分享&#xff1a;从大小凉山&#xff0c;到金沙江畔&#xff0c;从乌蒙山脉&#xff0c;到红河两岸。——山鹰组合 主要内容&#xff1a;转发和重定向的区别、优缺点…

C++单继承和多继承

C单继承和多继承继承单继承写法继承中构造函数的写法写法构造和析构的顺序问题多继承继承 1.继承&#xff0c;主要是遗传学中的继承概念 2.继承的写法&#xff0c;继承中的权限问题 3.继承中的构造函数的写法 继承&#xff1a;子类没有新的属性&#xff0c;或者行为的产生 父类…

智能生活垃圾检测与分类系统(UI界面+YOLOv5+训练数据集)

摘要&#xff1a;智能生活垃圾检测与分类系统用于日常生活垃圾的智能监测与分类&#xff0c;通过图片、视频和摄像头识别生活垃圾&#xff0c;对常见的可降解、纸板、玻璃、金属、纸质和塑料等类别垃圾进行检测和计数&#xff0c;以协助垃圾环保分类处理。本文详细介绍基于YOLO…

Kubernetes学习(七)补充:基于自定义指标进行扩缩容

资源指标只包含CPU、内存&#xff0c;一般来说也够了。但如果想根据自定义指标:如请求qps/5xx错误数来实现HPA&#xff0c;就需要使用自定义指标了&#xff0c;目前比较成熟的实现是 Prometheus Custom Metrics。自定义指标由Prometheus来提供&#xff0c;再利用k8s-prometheus…

【C语言】字符串函数和内存函数

前言&#x1f338;在我们编写C程序时&#xff0c;除了使用自定义函数&#xff0c;往往还会使用一些库函数&#xff0c;例如标准输入输出函数printf&#xff0c;scanf&#xff0c;字符串函数strlen&#xff0c;内存函数memset等等&#xff0c;使用这些系统自带的库函数可以轻松地…

【Spring】我抄袭了Spring,手写一套MySpring框架。。。

这篇博客实现了一个简单版本的Spring&#xff0c;主要包括Spring的Ioc和Aop功能 文章目录这篇博客实现了一个简单版本的Spring&#xff0c;主要包括Spring的Ioc和Aop功能&#x1f680;ComponentScan注解✈️Component注解&#x1f681;在spring中ioc容器的类是ApplicationConte…

来到CSDN的一些感想

之所以会写下今天这篇博客&#xff0c;是因为心中实在是有很多话想说&#xff01;&#xff01;&#xff01; 认识我的人应该都知道&#xff0c;我是才来CSDN不久的&#xff0c;也可以很清楚地看见我的码龄&#xff0c;直到今天&#xff1a;清清楚楚地写着&#xff1a;134天&…

狄拉克符号系统

狄拉克符号系统&#xff1a;狄拉克构造了一个抽象的、一般矢量--态矢和一整套狄拉克符号以描述量子力学体系的状态一般狄拉克符号任何力学量的完全集的本征函数系作为基矢构成希尔伯特空间&#xff0c;微观体系的状态波函数作为该空间的一个态矢&#xff0c;有且仅有。态矢在所…

手把手教你基于HTML、CSS搭建我的相册(上)

The sand accumulates to form a pagoda写在前面HTML是什么&#xff1f;CSS是什么&#xff1f;demo搭建写在最后写在前面 其实有过一些粉丝咨询前端该从什么开始学&#xff0c;那当然是我们的前端基础三件套开始学起&#xff0c;HTML、CSS、javaScript&#xff0c;前端的大部分…

Qt学习_11_构建内嵌子界面与独立子界面的框架

0 前言 对于较大的Qt项目而言&#xff0c;弹出的独立子界面 与 根据菜单在主窗口内切换的内嵌子界面是我们所必须面对的问题。那么在项目框架对这两部分内容应该如何布局&#xff0c;才能让项目更清爽&#xff0c;更规整。是本文着重讨论的内容。下图给出我们的最终设计框架。…
最新文章