目录
1. tftp协议概述
简单文件传输协议,适用于在网络上进行文件传输的一套标准协议,使用UDP传输
特点:
是应用层协议
基于UDP协议实现
数据传输模式
octet:二进制模式(常用)
mail:已经不再支持
2. tftp下载模型
TFTP通信过程总结
-
服务器在69号端口等待客户端的请求
-
服务器若批准此请求,则使用 ==临时端口== 与客户端进行通信。
-
每个数据包的编号都有变化(从1开始)
-
每个数据包都要得到ACK的确认,如果出现超时,则需要重新发送最后的数据包或ACK包
-
数据长度以512Byte传输的,小于512Byte的数据意味着数据传输结束。
3. tftp协议分析
差错码:
0 未定义,参见错误信息
1 File not found.
2 Access violation.
3 Disk full or allocation exceeded.
4 illegal TFTP operation.
5 Unknown transfer ID.
6 File already exists.
7 No such user.
8 Unsupported option(s) requested.
代码
#include<stdio.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<unistd.h>
#include<string.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <fcntl.h>
//打印错误信息的宏函数
#define ERR_MSG(msg) do{\
fprintf(stderr,"__%d__",__LINE__);\
perror(msg);\
}while(0);
int main(int argc, const char *argv[])
{
if(argc<2)
{
fprintf(stderr,"请输入filename\n");
return -1;
}
//创建报式套接字
int sfd=socket(AF_INET,SOCK_DGRAM,0);
if(sfd<0)
{
ERR_MSG("socket");
return -1;
}
//填充服务器的IP地址以及端口号
struct sockaddr_in sin;
sin.sin_family =AF_INET;
sin.sin_port =htons(69);
sin.sin_addr.s_addr=inet_addr("192.168.0.110");
//绑定服务器的地址信息结构体-->非必须绑定
char buf[128]="";
char* ptr=buf;
short int* pa=(short int*)ptr;
*pa=htons(1);
char* pb=ptr+2;
char filename[128]="";
strcpy(filename,argv[1]);
strcpy(pb,filename);
char* pc=pb+strlen(pb);
char* pd=pc+1;
strcpy(pd,"octet");
int size=2+strlen(pb)+1+strlen("octet")+1;
if(sendto(sfd,buf,size,0,(struct sockaddr*)&sin,sizeof(sin))<0)
{
ERR_MSG("sendto");
return -1;
}
printf("sendto success\n");
int fd=open(argv[2],O_WRONLY|O_CREAT|O_TRUNC,0664);
if(fd<0)
{
ERR_MSG("open");
return -1;
}
int rcv_len;
unsigned short num=1;
while(1)
{
//接收
unsigned char buf1[1024]="";
struct sockaddr_in cin;
socklen_t addrlen = sizeof(cin);
if((rcv_len=recvfrom(sfd,buf1,sizeof(buf1),0,(struct sockaddr*)&cin,&addrlen))<0)
{
ERR_MSG("recvfrom");
return -1;
}
if(buf1[1]==3)
{
if(num==ntohs(*(unsigned short *)(buf1+2)))
{
write(fd,buf1+4,rcv_len-4);
num++;
}
//发送ACK包给服务器
buf1[1]=4;
sendto(sfd,buf1,4,0,(struct sockaddr*)&cin,sizeof(cin));
if(rcv_len<516)
break;
}
}
close(fd);
//关闭套接字
close(sfd);
return 0;
}