tftp客户端实现(一)-发出一条RRQ消息

一、简介

TFTP(Trivial File Transfer Protocol,简单文件传输协议)是TCP/IP协议族中的一个用来在客户机与服务器之间进行简单文件传输的协议,提供不复杂、开销不大的文件传输服务。端口号为69。

二、报文格式

tftp报文格式

TFTP支持5种类型的包,TFTP数据报的头两个字节为操作码,通过操作码来标识这5种类型的包:
1. Read request(RRQ)
2. Write request(WRQ)
3. Data(DATA)
4. Acknowledgment(ACK)
5. Error(ERROR)
而模式域包括了字符串”netascii”,”octet”或”mail”,名称不分大小写,这里先不一一细说。

三、发送过程

tftp客户端RRQ请求

首先,作为tftp客户端,主机A从X端口发送一条RRQ消息(读请求)到tftp服务器监听的69端口。服务器收到这条RRQ消息后就开始往主机A回发文件数据,需注意的是:回发文件数据的端口不再是69,而是服务器上随机的某个端口,图中为Y端口。主机A从本地的X端口接收完服务器的数据后,就会发送一条ACK响应消息给服务器。这就是一次RRQ的请求过程。

四、主要代码:

tftp.h:
#ifndef __TFTPC_H__
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <strings.h>
#include <sys/stat.h>
#include <sys/socket.h>
#include <netdb.h>
#include <sys/ioctl.h>
#include <assert.h>

#define OPCODE_RRQ (1)
#define OPCODE_WRQ (2)
#define OPCODE_DATA (3)
#define OPCODE_ACK (4)
#define OPCODE_ERR (5)

#define BLOCKSIZE (512)

struct TFTPHeader{
    short opcode;
}__attribute__((packed));

struct TFTPWRRQ{
    struct TFTPHeader header;
    char *filename; 
    char *mode; 
}__attribute__((packed));

struct TFTPData{
    struct TFTPHeader header;
    short block;
    char data[];
}__attribute__((packed));

struct TFTPACK{
    struct TFTPHeader header;
    short block;
}__attribute__((packed));

struct TFTPERR{
    struct TFTPHeader header;
    short errcode;
    char *errmsg;   
}__attribute__((packed));
#endif


tftp_client.c:
#include "tftp.h"
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <errno.h>

/************************************************************
FileName:    tftp_client.c
Date:        2017-03-05
Description: Linux Debian系统中,实现tftp协议的RRQ消息的发送,通过WireShark捕获并分析
***********************************************************/

int main(int argc, char **argv)
{
    struct hostent *serv_host;
    struct sockaddr_in serv_addr;

    int sockfd = -1;
    int opt = 1;
    int ret = -1;


    if(argc != 5)       /*判断入口参数是否为5个*/
    {
        fprintf(stderr,"Usage: %s host port [get|put] filename\n", argv[0]);
        exit(1);

    }

    char *host = argv[1];
    int serv_port = atoi(argv[2]);
    int opt_code = strcmp(argv[3], "get") == 0 ? 1 : 2;
    char *filename = argv[4];

    printf("DBG: host = %s , port = %d , opt = %d , filename = %s \n",
    host, serv_port, opt_code, filename);

    /*获取tftp服务器的信息*/
    if((serv_host = gethostbyname(argv[1])) == NULL)
    {
        printf("host is incorrect!\n");
        exit(1);
    }

    serv_addr.sin_family = PF_INET;
    serv_addr.sin_addr = *((struct in_addr *)serv_host->h_addr_list[0]);
    serv_addr.sin_port = htons(serv_port);

    /*获取udp的协议号*/
    struct protoent *proto;
    if((proto = getprotobyname("udp")) == NULL)
    {
        fprintf(stderr,"getprotoname(), failed to find udp protoco info\n");
        exit(1);
    }

    /*创建udp套接字*/
    if((sockfd = socket(PF_INET, SOCK_DGRAM, proto->p_proto)) < 0)
    {
        perror("ERROR opening socket\n");
        exit(1);
    }

    /*构造一条RRQ消息*/
    struct TFTPHeader header;
    header.opcode = htons(OPCODE_RRQ);

    int filenamelen = strlen(filename) + 1;
    int packetsize = sizeof(header) + filenamelen + 5 + 1;     /*5为数据报模式(octet)的长度,1为字符串结束符长度*/

    void *packet = malloc(packetsize);   
    memcpy(packet, &header, sizeof(header));
    memcpy(packet + sizeof(header), filename, filenamelen);

    char *mode = "octet";
    memcpy(packet + sizeof(header) + filenamelen, mode, strlen(mode) + 1);

    if(sendto(sockfd, packet, packetsize, 0,
        (struct sockaddr*)&serv_addr, sizeof(struct sockaddr)) < 0){
        perror("sendto()");
        return 1;
    }
    return 0;
}

五、结果分析:

Linux Debian下执行:

Linux Debian执行

这里以本地主机作为tftp服务器端,1234为服务器监听端口,获取服务器上的testfile文件。
WireShark抓包结果分析:

WireShark抓包结果分析

  • 1
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
TFTP协议是一种简单的文件传输协议,它可以用于在计算机之间传输文件。TFTP协议基于UDP协议,因此它不具备TCP协议的可靠传输保证。本文将介绍如何使用C语言实现TFTP协议的客户端和服务端。 ## TFTP协议概述 TFTP协议是一种基于UDP协议的文件传输协议,它采用简单的请求/响应模式进行通信。TFTP协议主要有两种模式,分别是Netascii模式和Octet模式。 Netascii模式是一种ASCII码模式,它将传输的文件视为ASCII码字符流进行传输。在Netascii模式下,每个文件的换行符会被转换成Telnet的行结束符CR-LF(回车换行)。 Octet模式是一种二进制模式,它将传输的文件视为二进制数据流进行传输。在Octet模式下,每个文件的换行符会被保留。 TFTP协议定义了五种不同的请求/响应类型,分别是: - RRQ:读请求 - WRQ:写请求 - DATA:数据包 - ACK:确认包 - ERROR:错误包 ## TFTP客户端实现 TFTP客户端主要有两个功能,分别是向服务器请求文件和向服务器发送文件。下面是使用C语言实现TFTP客户端的代码示例。 ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <sys/socket.h> #include <netinet/in.h> #define SERVER_IP "127.0.0.1" #define SERVER_PORT 69 #define BUF_SIZE 512 void error(char *msg) { perror(msg); exit(1); } int main(int argc, char *argv[]) { int sockfd, n; struct sockaddr_in servaddr; char buf[BUF_SIZE]; if (argc < 3) { fprintf(stderr,"usage: %s <filename> <mode>\n", argv[0]); exit(1); } // 创建套接字 sockfd = socket(AF_INET, SOCK_DGRAM, 0); if (sockfd < 0) error("ERROR opening socket"); // 设置服务器地址 memset(&servaddr, 0, sizeof(servaddr)); servaddr.sin_family = AF_INET; servaddr.sin_addr.s_addr = inet_addr(SERVER_IP); servaddr.sin_port = htons(SERVER_PORT); // 发送读请求 sprintf(buf, "%c%c%s%c%s%c", 0, 1, argv[1], 0, argv[2], 0); n = sendto(sockfd, buf, strlen(buf), 0, (struct sockaddr *)&servaddr, sizeof(servaddr)); if (n < 0) error("ERROR sending read request"); // 接收数据 FILE *fp = fopen(argv[1], "wb"); while (1) { // 接收数据包 struct sockaddr_in cliaddr; socklen_t len = sizeof(cliaddr); n = recvfrom(sockfd, buf, BUF_SIZE, 0, (struct sockaddr *)&cliaddr, &len); if (n < 0) error("ERROR receiving data"); // 解析数据包 unsigned short opcode = ntohs(*(unsigned short *)buf); unsigned short block = ntohs(*(unsigned short *)(buf + 2)); if (opcode == 3) { // 写入文件 fwrite(buf + 4, 1, n - 4, fp); // 发送确认包 sprintf(buf, "%c%c%c%c", 0, 4, buf[2], buf[3]); n = sendto(sockfd, buf, 4, 0, (struct sockaddr *)&cliaddr, len); if (n < 0) error("ERROR sending ACK"); if (n < BUF_SIZE - 4) break; } else if (opcode == 5) { // 接收到错误包 fprintf(stderr, "ERROR: %s\n", buf + 4); break; } else { // 接收到无法识别的数据包 fprintf(stderr, "ERROR: unrecognized packet\n"); break; } } fclose(fp); close(sockfd); return 0; } ``` 上面的代码实现TFTP客户端的读请求功能,它会向服务器发送一个RRQ请求,并接收服务器返回的数据包。如果接收到的数据包是一个数据包,则将数据写入指定的文件中,并发送一个ACK确认包。 ## TFTP服务端实现 TFTP服务端主要有两个功能,分别是接收客户端的文件请求和接收客户端发送的文件。下面是使用C语言实现TFTP服务端的代码示例。 ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <sys/socket.h> #include <netinet/in.h> #define SERVER_PORT 69 #define BUF_SIZE 512 void error(char *msg) { perror(msg); exit(1); } int main(int argc, char *argv[]) { int sockfd, n; struct sockaddr_in servaddr, cliaddr; socklen_t len; char buf[BUF_SIZE]; // 创建套接字 sockfd = socket(AF_INET, SOCK_DGRAM, 0); if (sockfd < 0) error("ERROR opening socket"); // 设置服务器地址 memset(&servaddr, 0, sizeof(servaddr)); servaddr.sin_family = AF_INET; servaddr.sin_addr.s_addr = INADDR_ANY; servaddr.sin_port = htons(SERVER_PORT); // 绑定套接字 if (bind(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0) error("ERROR on binding"); while (1) { // 接收请求 len = sizeof(cliaddr); n = recvfrom(sockfd, buf, BUF_SIZE, 0, (struct sockaddr *)&cliaddr, &len); if (n < 0) error("ERROR receiving request"); // 解析请求 unsigned short opcode = ntohs(*(unsigned short *)buf); if (opcode != 1) { // 接收到非读请求 sprintf(buf, "%c%c%c%c%s%c", 0, 5, 0, 4, "Unsupported request", 0); n = sendto(sockfd, buf, strlen(buf), 0, (struct sockaddr *)&cliaddr, len); if (n < 0) error("ERROR sending error packet"); } else { // 发送数据 FILE *fp = fopen(buf + 2, "rb"); if (fp == NULL) { // 文件不存在 sprintf(buf, "%c%c%c%c%s%c", 0, 5, 0, 1, "File not found", 0); n = sendto(sockfd, buf, strlen(buf), 0, (struct sockaddr *)&cliaddr, len); if (n < 0) error("ERROR sending error packet"); } else { // 文件存在,发送数据包 unsigned short block = 1; while (1) { // 读取文件 n = fread(buf + 4, 1, BUF_SIZE - 4, fp); if (n < BUF_SIZE - 4) { if (feof(fp)) { // 文件读取完成 sprintf(buf, "%c%c%c%c", 0, 3, block >> 8, block & 0xFF); n += 4; break; } else { // 文件读取出错 sprintf(buf, "%c%c%c%c%s%c", 0, 5, 0, 2, "Error reading file", 0); n = sendto(sockfd, buf, strlen(buf), 0, (struct sockaddr *)&cliaddr, len); if (n < 0) error("ERROR sending error packet"); fclose(fp); break; } } else { // 发送数据包 sprintf(buf, "%c%c%c%c", 0, 3, block >> 8, block & 0xFF); n += 4; block++; } n = sendto(sockfd, buf, n, 0, (struct sockaddr *)&cliaddr, len); if (n < 0) { fclose(fp); error("ERROR sending data"); } // 接收ACK确认包 n = recvfrom(sockfd, buf, BUF_SIZE, 0, (struct sockaddr *)&cliaddr, &len); if (n < 0) { fclose(fp); error("ERROR receiving ACK"); } opcode = ntohs(*(unsigned short *)buf); unsigned short ack_block = ntohs(*(unsigned short *)(buf + 2)); if (opcode != 4 || ack_block != block - 1) { // 接收到错误的ACK确认包 sprintf(buf, "%c%c%c%c%s%c", 0, 5, 0, 0, "ACK packet error", 0); n = sendto(sockfd, buf, strlen(buf), 0, (struct sockaddr *)&cliaddr, len); if (n < 0) error("ERROR sending error packet"); fclose(fp); break; } } fclose(fp); } } } close(sockfd); return 0; } ``` 上面的代码实现TFTP服务端的读请求功能,它会接收客户端发来的RRQ请求,并将指定的文件内容发送给客户端。如果接收到的数据包是一个ACK确认包,则继续发送下一个数据包。如果接收到的ACK确认包有误,则发送一个错误包告知客户端

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值