DPDK用户态协议栈-TCP Posix API 2

tcp posix api

send发送

C 复制代码
ssize_t nsend(int sockfd, const void *buf, size_t len, __attribute__((unused))int flags) {

    ssize_t length = 0;

    void* hostinfo = get_host_fromfd(sockfd);

    if (hostinfo == NULL) {

        return -1;
    }

    struct ln_tcp_stream* stream = (struct ln_tcp_stream*)hostinfo;
    if (stream->proto == IPPROTO_TCP) {

        struct ln_tcp_fragment* fragment = rte_malloc("send frag", sizeof(struct ln_tcp_fragment), 0);

        if (fragment == NULL) {

            return 0;
        }
        memset(fragment, 0, sizeof(struct ln_tcp_fragment));

        fragment->sport = stream->dport;
        fragment->dport = stream->sport;

        fragment->acknum = stream->recv_next;
        fragment->seqnum = stream->send_next;

        fragment->windows = LN_TCP_INITIAL_WINDOWS;
        fragment->tcp_flags = RTE_TCP_ACK_FLAG | RTE_TCP_PSH_FLAG;
        fragment->hdr_off = 0x50;

        fragment->data = rte_malloc("frag data", len + 1, 0);

        if (fragment->data == NULL) {

            rte_free(fragment);
            return -1;
        }
        memset(fragment->data, 0, len + 1);

        rte_memcpy(fragment->data, buf, len);
        fragment->length = len;
        length = fragment->length;

        rte_ring_mp_enqueue(stream->send_next, (void*)fragment);
    }

    return length;
}

recv接收

C 复制代码
ssize_t nrecv(int sockfd, void *buf, size_t len, __attribute__((unused))int flags) {

    ssize_t length = 0;

    void* hostinfo = get_host_fromfd(sockfd);

    if (hostinfo == NULL) {

        return -1;
    }

    struct ln_tcp_stream* stream = (struct ln_tcp_stream*)hostinfo;

    if (stream->proto == IPPROTO_TCP) {

        struct ln_tcp_fragment* fragment = NULL;
        int nb_rev = 0;

        pthread_mutex_lock(&stream->mutex);
        while((nb_rev = rte_ring_mc_dequeue(stream->recvbuf, (void**)&fragment)) < 0) {

            pthread_cond_wait(&stream->cond, &stream->mutex);
        }
        pthread_mutex_unlock(&stream->mutex);

        if (fragment->length > len) {

            rte_memcpy(buf, fragment->data, len);

            int i;
            for (i = 0; i < fragment->length - len; i++) {

                fragment->data[i] = fragment->data[i + len];
            }

            fragment->length = fragment->length - len;
            length = fragment->length;

            rte_ring_mp_enqueue(stream->recvbuf, (void*)fragment);
        }
        else if (fragment->length == 0) {

            rte_free(fragment);
        }
        else {

            rte_memcpy(buf, fragment->data, len);

            length = fragment->length;

            rte_free(fragment->data);
            fragment->data = NULL;

            rte_free(fragment);
        }
    }

    return length;
}

数据包收发管理管理

接收

C 复制代码
static int ln_tcp_enqueue_recvbuf(struct ln_tcp_stream* stream, struct rte_tcp_hdr* tcphdr, int tcplen) {

    struct ln_tcp_fragment* fragment = rte_malloc("tcp frag", sizeof(struct ln_tcp_fragment));

    if (fragment == NULL) {

        return -1;
    }
    memset(fragment, 0, sizeof(struct ln_tcp_fragment));

    fragment->dport = ntohs(tcphdr->dst_port);
    fragment->sport = ntohs(tcphdr->src_port);

    uint8_t hdrlen = tcphdr->data_off >> 4;
    int payloadlen = tcplen - hdrlen * 4;

    if (payloadlen > 0) {

        uint8_t* payload = (uint8_t*)tcphdr + hdrlen * 4;

        fragment->data = rte_malloc("frag data", payloadlen + 1, 0);

        if (fragment->data == NULL) {

            rte_free(fragment);
            return -1;
        }
        memset(fragment, 0, payloadlen + 1);

        rte_memcpy(fragment->data, payload, payloadlen);
        fragment->length = payloadlen;
    }
    else if (payloadlen == 0) {

        fragment->length = 0;
        fragment->data = NULL;
    }

    rte_ring_mp_enqueue(stream->recvbuf, (void*)fragment);

    pthread_mutex_lock(&stream->mutex);
    pthread_cond_signal(&stream->cond);
    pthread_mutex_unlock(&stream->mutex);

    return 0;
}

发送

C 复制代码
static int ln_tcp_send_ackpkt(struct ln_tcp_stream* stream, struct rte_tcp_hdr* tcphdr) {

    struct ln_tcp_fragment* fragment = rte_malloc("ack frag", sizeof(struct ln_tcp_fragment), 0);

    if (fragment == NULL) {

        return -1;
    }
    memset(fragment, 0, sizeof(struct ln_tcp_fragment));

    fragment->sport = stream->dport;
    fragment->dport = stream->sport;

    fragment->acknum = stream->recv_next;
    fragment->seqnum = stream->send_next;

    fragment->windows = LN_TCP_INITIAL_WINDOWS;
    fragment->length = 0;
    fragment->data = NULL;
    fragment->hdr_off = 0x50;
    fragment->tcp_flags = RTE_TCP_ACK_FLAG;

    rte_ring_mp_enqueue(stream->sendbuf, (void*)fragment);

    return 0;
}

establish handle

C 复制代码
static int ln_tcp_handle_close_established(struct ln_tcp_stream* stream, struct rte_tcp_hdr* tcphdr, int tcplen) {

    if (tcphdr->tcp_flags & RTE_TCP_SYN_FLAG) {


    }

    if (tcphdr->tcp_flags & RTE_TCP_PSH_FLAG) {

        ln_tcp_enqueue_recvbuf(stream, tcphdr, tcplen);

        uint8_t hdrlen = tcphdr->data_off >> 4;
        int payloadlen = tcphdr - hdrlen * 4;

        stream->recv_next = stream->recv_next + payloadlen;
        stream->send_next = ntohl(tcphdr->recv_ack);
        ln_tcp_send_ackpkt(stream, tcphdr);
    }

    if (tcphdr->tcp_flags & RTE_TCP_ACK_FLAG) {


    }

    if (tcphdr->tcp_flags & RTE_TCP_FIN_FLAG) {

        stream->status = LN_TCP_STATUS_CLOSE_WAIT;

        ln_tcp_enqueue_recvbuf(stream, tcphdr, tcphdr->data_off >> 4);

        stream->recv_next = stream->recv_next + 1;
        stream->send_next = ntohl(tcphdr->recv_ack);

        ln_tcp_send_ackpkt(stream, tcphdr);
    }

    return 0;
}

tcp server

C 复制代码
static int tcp_server_entry(__attribute__((unused)) void* arg) {

    int sockfd = nsocket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0) {

        return -1;
    }

    printf("tcp sockfd is %d\n", sockfd);

    struct sockaddr_in servaddr;
    memset(&servaddr, 0, sizeof(struct sockaddr));
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(8888);

    nbind(sockfd, (struct sockaddr*)&servaddr, sizeof(servaddr));

    nlisten(sockfd, 10);

    while (1) {

        struct sockaddr_in client;
        socklen_t len = sizeof(client);

        int connfd = naccept(sockfd, (struct sockaddr*)&client, &len);

        char buff[BUFFER_SIZE] = {0};

        while (1) {

            int n = nrecv(connfd, buff, BUFFER_SIZE, 0); // block
            if (n > 0) {

                printf("recv: %s\n", buff);
                nsend(connfd, buff, n, 0);
            }
            else if (n == 0) {

                nclose(connfd);
                break;
            }
            else {

                //nonblock
            }
        }
    }

    nclose(sockfd);
}

效果展示

回去之后调好了,有点难调

总结

到目前为止,IP/TCP和IP/UDP的协议栈都写完了,但是没有并发效果;这个后面会解决。下一步是探索一下协议的扩展,写一个dns服务器来看一下如何基于tcp或者udp来扩展协议。

项目地址

项目地址

参考资料:https://github.com/0voice

相关推荐
白帽黑客沐瑶7 小时前
【网络安全就业】信息安全专业的就业前景(非常详细)零基础入门到精通,收藏这篇就够了
网络·安全·web安全·计算机·程序员·编程·网络安全就业
树码小子8 小时前
Java网络编程:(socket API编程:TCP协议的 socket API -- 回显程序的服务器端程序的编写)
java·网络·tcp/ip
绿箭柠檬茶9 小时前
Ubuntu 服务器配置转发网络访问
服务器·网络·ubuntu
FPGA_Linuxer10 小时前
FPGA 40 DAC线缆和光模块带光纤实现40G UDP差异
网络协议·fpga开发·udp
real 110 小时前
传输层协议UDP
网络·网络协议·udp
路由侠内网穿透11 小时前
本地部署 GPS 跟踪系统 Traccar 并实现外部访问
运维·服务器·网络·windows·tcp/ip
喵手14 小时前
玩转Java网络编程:基于Socket的服务器和客户端开发!
java·服务器·网络
徐子元竟然被占了!!15 小时前
实验-基本ACL
网络
ftpeak16 小时前
从零开始使用 axum-server 构建 HTTP/HTTPS 服务
网络·http·https·rust·web·web app
LabVIEW开发17 小时前
LabVIEW气体污染无线监测
网络·labview·labview知识·labview功能·labview程序