zoukankan      html  css  js  c++  java
  • linux SSSocket 简单封装

    //
    //  simple_socket.h
    //  SSSocket
    //
    
    #ifndef SSSocket_simple_socket_h
    #define SSSocket_simple_socket_h
    
    struct ssso;
    
    enum SSSO_STATUS{
        SSSO_STATUS_CONNECTING,
        SSSO_STATUS_CONNECTED,
        SSSO_STATUS_CLOSED,
        SSSO_STATUS_ERROR,
    };
    
    //create a socket , ready to connect to host define by host_name and port
    struct ssso* ssso_new(const char* host_name, int port);
    
    //close socket
    void  ssso_free(struct ssso* so);
    
    //update socket, write data from buffer to socket, and if there is data readable, it return 1
    int   ssso_check(struct ssso* so);
    
    //peek whether has data of size in read buffer, return data or NULL
    void* ssso_peek(struct ssso* so, size_t size);
    
    //discard data of size in read buffer, mainly use after geting integrate message data using ssso_peek
    void  ssso_discard(struct ssso* so, size_t size);
    
    //write data to buffer, after do ssso_check, the data may be send through socket
    void  ssso_write(struct ssso* so, const void* data, size_t size);
    
    enum SSSO_STATUS ssso_status(struct ssso* so);
    #endif


    #include <sys/socket.h>
    #include <sys/types.h>
    #include <netinet/in.h>
    #include <arpa/inet.h>
    #include <netdb.h>
    #include <unistd.h>
    #include "stdio.h"
    #include "stdlib.h"
    #include "string.h"
    #include "fcntl.h"
    #include "errno.h"
    #include "assert.h"
    
    #include "simple_socket.h"
    
    //simple stupid buffer
    struct ssbf{
            char* buf;
            size_t buf_size;
            size_t dat_size;
    };
    
    struct ssbf* ssbf_new(size_t size);
    void  ssbf_free(struct ssbf* bf);
    void  ssbf_put(struct ssbf* bf, const void* data, size_t size);
    void* ssbf_data(struct ssbf* bf);
    size_t   ssbf_datasize(struct ssbf* bf);
    void  ssbf_discard(struct ssbf* bf, size_t size);
    
    struct ssbf* ssbf_new(size_t size){
        struct ssbf* bf = malloc(sizeof(struct ssbf));
        bf->buf = malloc(size);
        bf->buf_size = size;
        bf->dat_size = 0;
        
        return bf;
    }
    
    void  ssbf_free(struct ssbf* bf){
        assert(bf != NULL);
        
        free(bf->buf);
        free(bf);
    }
    
    void  ssbf_put(struct ssbf* bf, const void* data, size_t put_size){
        assert(bf != NULL);
        assert(bf->dat_size <= bf->buf_size);
        
        if(bf->buf_size - bf->dat_size < put_size){
            size_t new_size = (bf->dat_size + put_size)*2;
            bf->buf = (char*)realloc(bf->buf, new_size);
            assert(bf->buf != NULL);
            bf->buf_size = new_size;
        }
        
        memcpy(bf->buf + bf->dat_size, data, put_size);
        bf->dat_size += put_size;
    }
    
    void* ssbf_data(struct ssbf* bf) { return bf->buf; }
    size_t   ssbf_datasize(struct ssbf* bf) {return bf->dat_size; }
    
    void  ssbf_discard(struct ssbf* bf, size_t discard_size){
        assert(bf != NULL);
        
        if(discard_size == -1) discard_size = bf->dat_size;
        assert(bf->dat_size >= discard_size);
        
        memcpy(bf->buf, bf->buf + discard_size, bf->dat_size - discard_size);
        bf->dat_size -= discard_size;
    }
    
    #define MAX_RECV_BUF 1024*10
    
    //simple stupid socket
    struct ssso{
            int fd;
            struct ssbf* wbf;
        struct ssbf* rbf;
        char recv_buf[MAX_RECV_BUF];
        enum SSSO_STATUS status;
        int error_code;
    };
    
    void set_error_code(struct ssso* so, int error_code){
        so->error_code = error_code;
        so->status = SSSO_STATUS_ERROR;
    }
    
    struct ssso* ssso_new(const char* host_name, int port){
            struct sockaddr_in pin;
        struct hostent *nlp_host;
        
        //解析域名,如果是IP则不用解析,如果出错,显示错误信息
        while ((nlp_host=gethostbyname(host_name))==0){
            printf("Resolve Error!
    ");
            return NULL;
        }
        
        bzero(&pin,sizeof(pin));
        pin.sin_family=AF_INET;                 //AF_INET表示使用IPv4
        pin.sin_addr.s_addr=htonl(INADDR_ANY);
        pin.sin_addr.s_addr=((struct in_addr *)(nlp_host->h_addr))->s_addr;
        pin.sin_port=htons(port);
        
        //建立socket
        int sd=socket(AF_INET,SOCK_STREAM,0);
        
        int flags = fcntl(sd, F_GETFL, 0);
        fcntl(sd, F_SETFL, flags | O_NONBLOCK);
    
        int connected = 1;
        if(connect(sd,(struct sockaddr*)&pin,sizeof(pin)) == -1){
            if(errno != EINPROGRESS){
                assert(0);
                return NULL;
            }
            else{
                connected = 0;
            }
        }
        
        struct ssso* so = malloc(sizeof(struct ssso));
        so->fd = sd;
        so->wbf = ssbf_new(MAX_RECV_BUF);
        so->rbf = ssbf_new(MAX_RECV_BUF);
        so->status = (connected == 1) ? SSSO_STATUS_CONNECTED : SSSO_STATUS_CONNECTING;
        return so;
    }
    
    void  ssso_free(struct ssso* so){
        assert(so != NULL);
        close(so->fd);
        free(so->wbf);
        free(so->rbf);
        free(so);
    }
    
    int check_sock_error(int fd){
        int conn_error;
        socklen_t len = sizeof(conn_error);
        int ret = getsockopt(fd, SOL_SOCKET, SO_ERROR, &conn_error, &len);
        
        if(ret == -1){
            return ret;
        }
        return conn_error;
    }
    
    int  ssso_check_connect(struct ssso* so){
        assert(so->status == SSSO_STATUS_CONNECTING);
        
        fd_set wfds;
        struct timeval timeout;
        timeout.tv_sec = 0;
        timeout.tv_usec = 0;
        
        FD_ZERO(&wfds);
        FD_SET(so->fd, &wfds);
        
        int maxfd = so->fd + 1;
        int rc = select(maxfd, NULL, &wfds, NULL, &timeout);
        
        if(rc == -1){
            //error
            set_error_code(so, errno);
            return -1;
        }
        else if(rc == 0){
            //time out, no event
        }
        else{
            if(FD_ISSET(so->fd, &wfds)){
                //writeable
                if((so->error_code = check_sock_error(so->fd)) != 0) {
                    printf("sock error %d
    ",so->error_code);
                    set_error_code(so, so->error_code);
                    return -1;
                }
                else{
                    so->status = SSSO_STATUS_CONNECTED;
                    printf("sock connected 
    ");
                    return 0;
                }
            }
        }
        
        return 0;
    }
    
    int  ssso_check(struct ssso* so){
        assert(so != NULL);
        
        if(so->status == SSSO_STATUS_CONNECTING){
            if(ssso_check_connect(so) != 0)
                return -1;
        }
        
        if(so->status != SSSO_STATUS_CONNECTED)
            return -1;
        
        fd_set rfds;
        fd_set wfds;
        struct timeval timeout;
        timeout.tv_sec = 0;
        timeout.tv_usec = 0;
        
        FD_ZERO(&rfds);
        FD_SET(so->fd, &rfds);
        
        FD_ZERO(&wfds);
        FD_SET(so->fd, &wfds);
        
        int maxfd = so->fd + 1;
        int rc = select(maxfd, &rfds, &wfds, NULL, &timeout);
        
        int readable = (ssbf_datasize(so->rbf) > 0) ? 1 : 0;
        if(rc == -1){
            //error
            set_error_code(so, errno);
            return -1;
        }
        else if(rc == 0){
            //time out, no event
        }
        else{
            if(FD_ISSET(so->fd, &rfds)){
                //readable
                size_t readed = recv(so->fd, so->recv_buf, MAX_RECV_BUF, 0);
                if(readed == 0){
                    printf("sock recv return 0, may be closed by server
    ");
                    so->status = SSSO_STATUS_CLOSED;
                    
                }
                else if(readed == -1){
                    printf("sock read error %d
    ",errno);
                    set_error_code(so, errno);
                    return -1;
                }
                else{
                    ssbf_put(so->rbf, so->recv_buf, readed);
                    readable = 1;
                }
            }
            if(FD_ISSET(so->fd, &wfds)){
                //writeable
                if(ssbf_datasize(so->wbf) > 0){
                    size_t wrote = write(so->fd, ssbf_data(so->wbf), ssbf_datasize(so->wbf));
                    if(wrote == -1){
                        printf("sock write error %d
    ",errno);
                        set_error_code(so, errno);
                        return -1;
                    }
                    else
                        ssbf_discard(so->wbf, wrote);
                }
            }
        }
        
        return readable;
    }
    
    void* ssso_peek(struct ssso* so, size_t size){
        assert(so != NULL);
        
        if(ssbf_datasize(so->rbf) < size)
            return NULL;
        
        return ssbf_data(so->rbf);
    }
    
    void  ssso_discard(struct ssso* so, size_t size){
        assert(so != NULL);
        
        ssbf_discard(so->rbf, size);
    }
    
    void  ssso_write(struct ssso* so, const void* data, size_t size){
        assert(so != NULL);
        
        ssbf_put(so->wbf, data, size);
    }
    
    enum SSSO_STATUS ssso_status(struct ssso* so){
        assert(so != NULL);
        
        return so->status;
    }



  • 相关阅读:
    2-SAT·hihoCoder音乐节
    Music in Car
    Game with a Strip
    Oleg and Little Ponies
    组合数性质求K个数选取i*j个数分成j组的方案数
    Python学习笔记03
    Python学习笔记02
    Python 学习笔记01
    欺骗侦测
    Oracle 使用小计(4)
  • 原文地址:https://www.cnblogs.com/hzcya1995/p/13318672.html
Copyright © 2011-2022 走看看