基于UDP的TFTP简单文本传输协议的下载和上传

news/2024/5/18 14:00:27 标签: udp, php, 网络协议

tftp 模型以及特点包的格式

 

 代码

#include <func.h>
#define PORT 69
#define IP "192.168.0.103"

int do_download(int sfd, struct sockaddr_in sin);
int do_upload(int sfd,struct sockaddr_in sin);

int main(int argc, const char *argv[])
{
    //创建报式套接字    
    int sfd = socket(AF_INET, SOCK_DGRAM, 0);

    if(sfd < 0)
    {
        ERR_MSG("sfd");
        return -1;
    }
    printf("socket create succes sfd=%d __%d__\n", sfd, __LINE__);
 //允许端口快速被复用---监测端口号是否真的有进程在占用,如果没有,则快速复用
    int reuse = 1;
    if(setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse)) < 0)
    {
        ERR_MSG("setsockopt");
        return -1;
    }
    printf("允许端口快速复用成功\n");

    //定义并填充地址信息结构体,方便发送数据
    //真实的地址信息结构体根据地址族指定 AF_INET :man ip(7)
    struct sockaddr_in sin;
    sin.sin_family      = AF_INET;
    sin.sin_port        = htons(PORT);      //端口号的网络字节序 : 1024-49151
    sin.sin_addr.s_addr = inet_addr(IP);    //本机号的网络字节序列
    char choose = 0;
    while(1)
    {
        printf("-------------------------\n");
        printf("----------1.下载---------\n");
        printf("----------2.上传---------\n");
        printf("----------3.退出---------\n");
        printf("-------------------------\n");
        printf("请输入---->");
        choose = getchar();
        while(getchar()!= '\n');

        switch(choose)
        {
        case '1' :
            //下载
            do_download(sfd, sin);
            break;
        case '2':
            //上传 do_upload
            do_upload(sfd, sin);
            break;
        case '3':
            close(sfd);
            return 0;
            //退出
        default:
            printf("输入错误请重新输出\n");
        }
    }
    //关闭套接字
    if(close(sfd) < 0)
    {
        ERR_MSG("close");
        return -1;
    }
    return 0;
}

int do_download(int sfd, struct sockaddr_in sin)
{

    char cuf[512] = "";
    char brr[200] = "";
    printf("-------->请输入下载的文件\n");
    scanf("%s", brr);

    short *ptr1 = (short *)cuf;
    *ptr1 = htons(1);

    char *ptr2 = cuf + 2;
    strcpy(ptr2, brr);

    char *ptr4 =ptr2+strlen(ptr2)+1;
    strcpy(ptr4,"octet");

    int len = 2+strlen(brr) + 1 + strlen(ptr4) + 1;
    if(sendto(sfd,cuf,len,0,(struct sockaddr*)&sin,sizeof(sin)) < 0 )
    {
        ERR_MSG("sendto");
        return-1;
    }
    //接受返回的地址信息结构体
    struct sockaddr_in cin;
    socklen_t clen = sizeof(cin);

    char buf[516];//数据包
    short * pcr1 = (short *)buf;
    *pcr1 = htons(3);
    short * pcr2 = pcr1 + 1;
    char arr[4];//ack
    short * ptrc1 = (short *)arr;
    *ptrc1 = htons(4);
    short * ptrc2 = ptrc1 + 1;

    ssize_t ter = 0;

    int fp = open("./1.png", O_WRONLY|O_CREAT|O_APPEND|O_TRUNC,0664);
    while(1)
    {
    //接受数据

    ter = recvfrom(sfd,buf,516,0,(struct sockaddr*)&cin,&clen);
    printf("ter=%ld __%d__\n", ter, __LINE__);
    *ptrc2 = *pcr2;
    write(fp,&buf[4], ter-4);

    sendto(sfd,arr,sizeof(arr), 0, (struct sockaddr*)&cin,sizeof(cin));
    if(ter < 516)
    {
    printf("下载成功\n");
        break;
    }

    }
    return 0;
}

int do_upload(int sfd,struct sockaddr_in sin)
{
    int ph = open("./2.png", O_RDONLY);
    char cuf[512] = "";
    char brr[200] = "";
    printf("-------->请输入传输的文件\n");
    scanf("%s", brr);

    short *ptr1 = (short *)cuf;
    *ptr1 = htons(2); //写请求

    char *ptr2 = cuf + 2;
    strcpy(ptr2, brr);

    char *ptr4 =ptr2+strlen(ptr2)+1;
    strcpy(ptr4,"octet");

    int len = 2+strlen(brr) + 1 + strlen(ptr4) + 1;
    if(sendto(sfd,cuf,len,0,(struct sockaddr*)&sin,sizeof(sin)) < 0 )
    {
        ERR_MSG("sendto");
        return-1;
    }
    //打包
    char demo[516] = "";
    short * ctr = (short*) demo;
    *ctr = htons(3);
    short* ctr2 = ctr+1;
    char arr[4];//ack
    short * ptrc1 = (short *)arr;
    *ptrc1 = htons(4);
    short * ptrc2 = ptrc1 + 1;

    ssize_t tev = 0, ter = 0;;                                                                                                  
    int i = 1;
    //定义接收的临时套接字
    struct sockaddr_in cin;
    socklen_t clen = sizeof(cin);

    while(1)
    {
        //发送数据
        if(recvfrom(sfd,arr,sizeof(arr),0,(struct sockaddr*)&cin,&clen)<0)//收到服务器创建的临时套接字地址
        {
           ERR_MSG("recvfrom");
           return -1;
        }
        *ctr2 = htons(i++);//编号
        tev = read(ph,(char*)(demo+4), 512);//读取文件
        if(ter = sendto(sfd, demo,tev+4,0, (struct sockaddr*)&cin, sizeof(cin)) < 0)//发送文件
        {
            ERR_MSG("sendto");
            return -1;
        }
        if(tev < 512)
        {
            printf("发送成功\n");
            break;
        }

        printf("ter = %ld __%d__\n", ter, __LINE__);
    }
    return 0;
}

 现象

下载

上传现象


http://www.niftyadmin.cn/n/5281830.html

相关文章

数据分析-19-Thera Bank信贷业务数据(包含数据代码)

文章目录 0. 数据代码下载1. 数据集介绍1.1 原始数据集变量的含义&#xff1a;1.2 原始数据的读入与清洗1.3 数据清洗及预处理 2. 探索数据变量2.1 数据相关系数的探索2.2 分类变量与开通贷款的关系探索2.2.1 银行存款证&#xff08;CD&#xff09;帐户与贷款之间的关系2.2.2 教…

Jenkins拉去Gitlab代码ssh方式凭证及私钥凭证

Jenkins要从Gitlab拉去代码&#xff0c;则需要在Jenkins配置凭证&#xff0c;同时在配置私钥 Jenkins->凭证->系统->全局凭证->添加凭证 同上图路径进入全局凭证配置页面 点击全局 再点左侧的添加凭证 选择"Username with password"&#xff0c;输入…

Torch 加速

loss.item() 1. 费时 loss.backward() loss_item loss.item() # todo loss.item() 这个比较费时间 0.2743 秒 2. 省时 loss_item loss.item() #todo 把它 移到 loss.backward() 前面&#xff0c;这个操作只需要 0.00027 秒 loss.backward() Index d_inv[torch.isin…

【Git-IDEA】在 IDEA 中使用 Git(clone、pull、push、merge、建立本地分支与远程分支的连接)

【Git-IDEA】在 IDEA 中使用 Git&#xff08;clone、pull、push、merge、建立本地分支与远程分支的连接&#xff09; 1&#xff09;Gitee2&#xff09;配置 Git3&#xff09;初始化本地仓库4&#xff09;连接远程仓库5&#xff09;clone5.1.方式一5.2.方式二 6&#xff09;分支…

window上编译和使用squashfs-tools工具

文章目录 前言背景一、squashfs-tools 工具如何使用二、源码编译步骤1.从github上下载源码2.搭建编译环境3.编译squashfs-tools三、遇到过的问题1.缺少库文件2.配置Makefile总结前言 SquashFS-Tools 是一个用于创建和管理 SquashFS 文件系统的工具集合。SquashFS 是一个只读文…

Python编程技巧 – 使用正则表达式

Python编程技巧 – 使用正则表达式 Python Programming Skills – Using Regular Expression By JacksonML Python以其强大的功能高居全球编程软件的榜首。它易于学习和使用&#xff0c;使其成为初学者绝佳语言。此外&#xff0c;Python还用于各种应用程序&#xff0c;包括We…

【数据结构和算法】---栈和队列的互相实现

目录 一、用栈实现队列1.1初始化队列1.2模拟入队列1.3模拟出队列1.4取模拟的队列头元素1.5判断队列是否为空 二、用队列实现栈2.1初始化栈2.2模拟出栈2.3模拟入栈2.4取模拟的栈顶元素2.5判读栈是否为空 一、用栈实现队列 具体题目可以参考LeetCode232. 用栈实现队列 首先要想到…

【模式识别】探秘判别奥秘:Fisher线性判别算法的解密与实战

​&#x1f308;个人主页&#xff1a;Sarapines Programmer&#x1f525; 系列专栏&#xff1a;《模式之谜 | 数据奇迹解码》⏰诗赋清音&#xff1a;云生高巅梦远游&#xff0c; 星光点缀碧海愁。 山川深邃情难晤&#xff0c; 剑气凌云志自修。 目录 &#x1f30c;1 初识模式识…