tftp协议的实现

一、tftp协议介绍

 
TFTP是一个传输文件的简单协议,它其于UDP协议而实现,但是我们也不能确定有些TFTP协议是基于其它传输协议完成的。此协议设计的 时候是进行小文件传输的。因此它不具备通常的FTP的许多功能,它只能从文件服务器上获得或写入文件,不能列出目录,不进行认证,它传输8位数据。传输中 有三种模式:netascii,这是8位的ASCII码形式,另一种是octet,这是8位源数据类型;最后一种mail已经不再支持,它将返回的数据直 接返回给用户而不是保存为文件。

二、 基于TFTP协议的网络数据包格式

---------------------------------------------------
| Local Medium | Internet | Datagram | TFTP |
---------------------------------------------------
可以看出,TFTP是应用层的协议,我们在linux进行编程时,只需要把TFTP包封装好,然后通过UDP协议进行发送或接收进行了。

三、TFTP数据包类型

TFTP支持六种类型的包:
opcode operation(2bytes)

1 Read request (RRQ)
2 Write request (WRQ)

 RRQ/WRQ包(带扩展选项)
 
 
通过读其RFC文档可以知道:
 
 
例如:客户端请求服务器端下载文件zImage,其扩展选项opt1 : timeout  5,opt 2 : blksize 1462

optcode         filename         mode             opt1                 opt2
---------------------------------------------------------------------------------
       1        |      zImage\0 |    octet\0    |   timeout\05\0 | blksize\01462\0 |
---------------------------------------------------------------------------------
注意:optcode占用2bytes,如果不带扩展选项,opt1,opt2就不需要添加了。如果使用扩展选项,每个扩展选项以NULL结束.

使用扩展选项可以使tftp包的收发更灵活,比如如果不使用扩展选项的话,blksize的默认值为512,即每次携带的数据大小为512byte。如果小于512就认为是最后一个数据包了。如果使用扩展选项的话,我们可以和服务器端商量,比如说让blksize的值大一点,常用的是1462。这样传输同样大小的文件,显然是使用扩展选项更快一点。

3 Data (DATA)

数据包:

2 bytes   2 bytes   n bytes
----------------------------------
| Opcode | Block # | Data |
----------------------------------

例如:发送第一个数据包,内容为"hello word"
    
    opcode        Block #          Data
------------------------------------------------
        3        |       1         |     hello word\0 | 
-------------------------------------------------

//应答包
4 Acknowledgment (ACK)

2 bytes 2 bytes
---------------------
| Opcode | Block # |
--------------------- 

例如:服务器端发送完一个数据包给客户端,客户端收到后应做出应答

     Opcode        Block#
-----------------------------
        4            |       1      |
------------------------------

5 Error (ERROR)

2 bytes       2 bytes           string  
-----------------------------------------
| Opcode   | ErrorCode | ErrMsg \0 |
-----------------------------------------

一个ERROR包,它的操作码是5,它的格式如上所示。此包可以被其它任何类型的包确认。错误码指定错误的类型。错误的值和错误的意义在附录中。错误信息是供程序员使用的。

/*
 *TFTP error code
 */

enum{
TFTP_ERR_UNDEFINED  = 0,
TFTP_ERR_FILE_NOT_FOUND          = 1,
TFTP_ERR_ACCESS_DENIED = 2,
TFTP_ERR_DISK_FULL = 3,
TFTP_ERR_UNEXPECTED_OPCODE  = 4,
TFTP_ERR_UNKNOWN_TRANSFER_ID = 5,
TFTP_ERR_FILE_ALREADY_EXISTS         = 6,
};

6  OACK(带扩展选项)
 
 
例如:
我们上面客户端用可选选项进行发读请求,服务器端收到后,做出应答如下:

optcode            opt1                 opt2
----------------------------------------------------
       1        |    timeout\05\0 | blksize\01462\0 |
----------------------------------------------------

注意:如果客户端请求时所带的选项,服务器端并不支持,此时服务器端将发出错误包
 
四、客户端和服务器端的交互
 
实例用wirshak抓包如下:
 
 
  • 1
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 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、付费专栏及课程。

余额充值