计算机网络学习笔记4-TFTP编程
Posted studying~
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了计算机网络学习笔记4-TFTP编程相关的知识,希望对你有一定的参考价值。
TFTP协议
简单文件传送协议,是基于UDP的应用层协议,被设计用来传输小文件
通信过程:
TFTP通信过程总结(无选项)
1、服务器在69号端口等待客户端的请求
2、服务器若批准此请求,则使用临时端口与客户端进行通信
3、每个数据包的编号都有变化(从1开始)、逐次递增
4、每个数据包都要得到ACK的确认,如果出现超时,则需要客户端重新发送ACK,若干次重发后,若服务器还未收到ACK,则客户端则发送ERROR
5、数据包的长度以516Byte传输
小于516Byte的数据意味着传输结束
通信协议:
数据传输模式:
octet:二进制模式
netascii:文本模式
mail:已经不再支持
注意:
以上的0代表的是’\\0’
不同的差错码对应不同的错误信息
错误码:
0 未定义,参见错误信息
1 File not found.-------------------------------找不到文件
2 Access violation.----------------------------访问冲突
3 Disk full or allocation exceeded.--------磁盘已满或超出分配
4 illegal TFTP operation.--------------------非法的TFTP操作
5 Unknown transfer ID.----------------------未知的传输ID
6 File already exists.-------------------------文件已存在
7 No such user.--------------------------------没有这个用户
8 Unsupported option(s) requested.------请求了不支持的选项
传输的数据的大小一定是516Byte吗?
不一定,在向服务器发送的读写请求报文没有无选项的情况下,传输数据的大小=516Byte,但是在有选项的情况下,我们可以根据自己的设定来更改,
同时在无选项的时候,最后发送的数据包通常小于516Byte。
由于网络的原因,一方收不到另一方的数据怎么办?
重发,若干次后若还是没收到,则发送ERROR
TFTP带选项:
如果发送带选项的读写请求:
常用选项:
- tsize选项 当读操作时,tsize选项的参数必须为“0”,服务器会返回待读取的文件的大小,当写操作时,tsize选项参数应为待写入文件的大小,服务器会回显该选项
- blksize选项 修改传输文件时使用的数据块的大小(范围:8~65464)
- timeout选项 修改默认的数据传输超时时间(单位:秒)
注意:我写的是客户端 服务器 是windows下的一个程序
windows下服务器:
运行状态:
tftp客户端代码:
#include<stdio.h>
#include<string.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
int main(int argc,char *argv[])
{
if(argc != 3)
{
printf("./a.out server_ip file_name\\n");
return 0;
}
//tftp是基于udp 所以是udp编程流程
int sockfd = socket(AF_INET, SOCK_DGRAM ,0);
if(sockfd <0)
{
perror("socket");
return 0;
}
//给tftp服务器发送 下载文件的请求
//组tftp 文件读取请求报文
unsigned char cmd[128]="";
int len = sprintf(cmd,"%c%c%s%c%s%c",0x00,0x01,argv[2],0,"octet",0);
//将请求cmd发给服务器的 69号端口
struct sockaddr_in server;
bzero(&server,sizeof(server));
server.sin_family = AF_INET;
server.sin_port = htons(69);
inet_pton(AF_INET,argv[1], &server.sin_addr.s_addr);
sendto(sockfd, cmd, len, 0, (struct sockaddr *)&server, sizeof(server));
//打开一个本地空的文件
int fd = open(argv[2],O_WRONLY|O_CREAT,0666);
if(fd < 0)
{
perror("open");
return 0;
}
//不同读取服务器传过来的文件数据
unsigned short num=0;
while(1)
{
unsigned char buf[1024]="";
struct sockaddr_in from;
socklen_t from_len = sizeof(from);
int len = recvfrom(sockfd,buf,sizeof(buf),0, \\
(struct sockaddr *)&from, &from_len);
//判断收到的数据的操作码 必须是00 03表示文件数据
if(buf[1]==0x03)//文件数据
{
//将文件数据 写入 本地址文件中
//防止写入重复数据
if((num+1) == ntohs(*(unsigned short *)(buf+2)) )
{
write(fd, buf+4 , len-4);
num = ntohs(*(unsigned short *)(buf+2));
printf("recv:%d\\n",num);
}
//给服务器发送ACK回应
buf[1]=4;
sendto(sockfd, buf , 4 ,0,(struct sockaddr *)&from,sizeof(from));
if(len < 516)//这是最后一个文件数据
break;
}
}
//关闭套接字
close(sockfd);
//关闭文件
close(fd);
return 0;
}
以上是关于计算机网络学习笔记4-TFTP编程的主要内容,如果未能解决你的问题,请参考以下文章