TFTP协议实现

   依据TFTP协议的规定,实现通过TFTP来实现文件的传输功能。
TFTP Server:
实现功能:实现TFTP服务器,可以让用户下载文件
命令:   tftps rootdir(请求的文件路径)
实现思路:
监听69端口
接收到数据后,fork子进程服务该客户端
分析客户端的请求报文,检查是否有扩展参数
如果有扩展参数,则回应OACK,并等待客户端对OACK做回应
启动传输数据
文件传输结束后子进程退出

tftps.c


int main(int argc,char **argv)
{
    
int server_sockfd,client_sockfd;
    
int server_len,client_len;
    
struct sockaddr_in server_address;
    
struct sockaddr_in client_address;
    
char recv_buf[516] = { };
    
char send_buf[516] = { };
    
char recv_cmd[4] = { };
    
int byte;
    
int n = 0;
    
int pid,fd;

    server_sockfd
= socket(AF_INET,SOCK_DGRAM,0);
    bzero
(&server_address,sizeof(server_address));
    server_address
.sin_family = AF_INET;
    server_address
.sin_addr.s_addr = inet_addr("192.168.220.59");

//我的PC机IP地址
    server_address
.sin_port = htons(69);
    server_len
= sizeof(server_address);
    
bind(server_sockfd,(struct sockaddr *)&server_address,server_len);

    
if((byte = recvfrom(server_sockfd,recv_buf,516,0,(struct sockaddr *)&client_address,sizeof(struct sockaddr))) == -1)
    
{
        
perror("recvfrom");
        
exit(0);
    
}

//创建一个进程来分析收到的报文信息
    
if((pid = fork()) == 0)
    
{
        
if(recv_buf[1] == 1)  //为下载文件请求
        
{
            
printf("this is a read request:\n");
            
if((fd = open(recv_buf[2],O_RDONLY)) < 0)
            

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
TFTP(Trivial File Transfer Protocol)是一种简单的文件传输协议,常用于将文件从服务器传输到客户端。下面是一个使用C语言实现TFTP协议的例子: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <unistd.h> #define TFTP_OPCODE_RRQ 1 #define TFTP_OPCODE_DATA 3 #define TFTP_OPCODE_ACK 4 #define TFTP_OPCODE_ERROR 5 #define TFTP_ERROR_UNDEFINED 0 #define TFTP_ERROR_FILE_NOT_FOUND 1 #define TFTP_ERROR_ACCESS_VIOLATION 2 #define TFTP_ERROR_DISK_FULL 3 #define TFTP_ERROR_ILLEGAL_OPERATION 4 #define TFTP_ERROR_UNKNOWN_TID 5 #define TFTP_ERROR_FILE_EXISTS 6 #define TFTP_ERROR_NO_SUCH_USER 7 #define TFTP_BLOCK_SIZE 512 struct tftp_packet { unsigned short opcode; union { struct { char filename[1]; char mode[1]; } rrq; struct { unsigned short block_num; char data[TFTP_BLOCK_SIZE]; } data; struct { unsigned short block_num; } ack; struct { unsigned short error_code; char error_msg[1]; } error; } data; }; int main(int argc, char *argv[]) { if (argc != 4) { fprintf(stderr, "Usage: %s <ip> <port> <file>\n", argv[0]); exit(EXIT_FAILURE); } int sock_fd = socket(AF_INET, SOCK_DGRAM, 0); if (sock_fd < 0) { perror("socket"); exit(EXIT_FAILURE); } struct sockaddr_in server_addr; memset(&server_addr, 0, sizeof(server_addr)); server_addr.sin_family = AF_INET; server_addr.sin_addr.s_addr = inet_addr(argv[1]); server_addr.sin_port = htons(atoi(argv[2])); char *filename = argv[3]; char mode[] = "octet"; int filename_len = strlen(filename); int mode_len = strlen(mode); int packet_len = 2 + filename_len + 1 + mode_len + 1; char *packet_buf = malloc(packet_len); if (!packet_buf) { perror("malloc"); exit(EXIT_FAILURE); } struct tftp_packet *packet = (struct tftp_packet *)packet_buf; packet->opcode = htons(TFTP_OPCODE_RRQ); memcpy(packet->data.rrq.filename, filename, filename_len); memcpy(packet->data.rrq.filename + filename_len, "\0", 1); memcpy(packet->data.rrq.mode, mode, mode_len); memcpy(packet->data.rrq.mode + mode_len, "\0", 1); int ret = sendto(sock_fd, packet_buf, packet_len, 0, (struct sockaddr *)&server_addr, sizeof(server_addr)); if (ret < 0) { perror("sendto"); exit(EXIT_FAILURE); } int block_num = 1; FILE *file = fopen(filename, "wb"); if (!file) { perror("fopen"); exit(EXIT_FAILURE); } while (1) { char recv_buf[TFTP_BLOCK_SIZE + 4]; struct sockaddr_in client_addr; socklen_t client_len = sizeof(client_addr); ret = recvfrom(sock_fd, recv_buf, sizeof(recv_buf), 0, (struct sockaddr *)&client_addr, &client_len); if (ret < 0) { perror("recvfrom"); exit(EXIT_FAILURE); } struct tftp_packet *recv_packet = (struct tftp_packet *)recv_buf; unsigned short opcode = ntohs(recv_packet->opcode); if (opcode == TFTP_OPCODE_DATA) { unsigned short recv_block_num = ntohs(recv_packet->data.data.block_num); if (recv_block_num == block_num) { int data_len = ret - 4; fwrite(recv_packet->data.data.data, 1, data_len, file); block_num++; char ack_buf[4]; struct tftp_packet *ack_packet = (struct tftp_packet *)ack_buf; ack_packet->opcode = htons(TFTP_OPCODE_ACK); ack_packet->data.ack.block_num = htons(recv_block_num); ret = sendto(sock_fd, ack_buf, sizeof(ack_buf), 0, (struct sockaddr *)&client_addr, client_len); if (ret < 0) { perror("sendto"); exit(EXIT_FAILURE); } if (data_len < TFTP_BLOCK_SIZE) { break; } } } else if (opcode == TFTP_OPCODE_ERROR) { unsigned short error_code = ntohs(recv_packet->data.error.error_code); fprintf(stderr, "TFTP error: %d\n", error_code); exit(EXIT_FAILURE); } else { fprintf(stderr, "Unexpected TFTP opcode: %d\n", opcode); exit(EXIT_FAILURE); } } fclose(file); close(sock_fd); free(packet_buf); return 0; } ``` 这个例子实现TFTP的读取请求(RRQ)和数据(DATA)包的处理,以及确认(ACK)包的发送。它会将服务器上的文件传输到本地,并保存为和服务器上的文件名相同的文件。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值