zoukankan      html  css  js  c++  java
  • libevent学习六(Connect listeners )



     
    创建与释放
    //backlog需要查询平台说明,在linux2.2以后 backlog就变成了已完成连接但未accept的队列的最大值(原来是处于syn状态的,现在换成sysctl 控制的参数tcp_max_syn_backlog)
     
     
    struct evconnlistener *evconnlistener_new(struct event_base *base,
        evconnlistener_cb cb, void *ptr, unsigned flags, int backlog,
        evutil_socket_t fd);
     
    //evconnlistener_new_bind 可以分配和bind socket fd
    struct evconnlistener *evconnlistener_new_bind(struct event_base *base,
        evconnlistener_cb cb, void *ptr, unsigned flags, int backlog,
        const struct sockaddr *sa, int socklen);
    void evconnlistener_free(struct evconnlistener *lev);
     
     
    flags:
    LEV_OPT_LEAVE_SOCKETS_BLOCKING

    By default, when the connection listener accepts a new incoming socket, it sets it up to be nonblocking so that you can use it with the rest of Libevent. Set this flag if you do not want this behavior.

    LEV_OPT_CLOSE_ON_FREE

    If this option is set, the connection listener closes its underlying socket when you free it.

    LEV_OPT_CLOSE_ON_EXEC

    If this option is set, the connection listener sets the close-on-exec flag on the underlying listener socket. See your platform documentation for fcntl and FD_CLOEXEC for more information.

    LEV_OPT_REUSEABLE

    By default on some platforms, once a listener socket is closed, no other socket can bind to the same port until a while has passed. Setting this option makes Libevent mark the socket as reusable, so that once it is closed, another socket can be opened to listen on the same port.

    LEV_OPT_THREADSAFE

    Allocate locks for the listener, so that it’s safe to use it from multiple threads. New in Libevent 2.0.8-rc.

    LEV_OPT_DISABLED

    Initialize the listener to be disabled, not enabled. You can turn it on manually with evconnlistener_enable(). New in Libevent 2.1.1-alpha.

    LEV_OPT_DEFERRED_ACCEPT
    If possible, tell the kernel to not announce sockets as having been accepted until some data has been received on them, and they are ready for reading. Do not use this option if your protocol doesn’t start out with the client transmitting data, since in that case this option will sometimes cause the kernel to never tell you about the connection. Not all operating systems support this option: on ones that don’t, this option has no effect. New in Libevent 2.1.1-alpha.
     
     
    //设置接收连接的回调函数
     
    typedef void (*evconnlistener_cb)(struct evconnlistener *listener,
        evutil_socket_t sock, struct sockaddr *addr, int len, void *ptr);
     
    void evconnlistener_set_cb(struct evconnlistener *lev,
        evconnlistener_cb cb, void *arg);
     
    //开关控制
    int evconnlistener_disable(struct evconnlistener *lev);
    int evconnlistener_enable(struct evconnlistener *lev);
     
     
    //错误诊断
    typedef void (*evconnlistener_errorcb)(struct evconnlistener *lis, void *ptr);
    void evconnlistener_set_error_cb(struct evconnlistener *lev,
        evconnlistener_errorcb errorcb);
     
    //其他
    evutil_socket_t evconnlistener_get_fd(struct evconnlistener *lev);
    struct event_base *evconnlistener_get_base(struct evconnlistener *lev);
     
    #include <event2/listener.h>
    #include <event2/bufferevent.h>
    #include <event2/buffer.h>

    #include <arpa/inet.h>

    #include <string.h>
    #include <stdlib.h>
    #include <stdio.h>
    #include <errno.h>

    static void
    echo_read_cb(struct bufferevent *bev, void *ctx)
    {
            /* This callback is invoked when there is data to read on bev. */
            struct evbuffer *input = bufferevent_get_input(bev);
            struct evbuffer *output = bufferevent_get_output(bev);

            /* Copy all the data from the input buffer to the output buffer. */
            evbuffer_add_buffer(output, input);
    }

    static void
    echo_event_cb(struct bufferevent *bev, short events, void *ctx)
    {
            if (events & BEV_EVENT_ERROR)
                    perror("Error from bufferevent");
            if (events & (BEV_EVENT_EOF | BEV_EVENT_ERROR)) {
                    bufferevent_free(bev);
            }
    }

    static void
    accept_conn_cb(struct evconnlistener *listener,
        evutil_socket_t fd, struct sockaddr *address, int socklen,
        void *ctx)
    {
            /* We got a new connection! Set up a bufferevent for it. */
            struct event_base *base = evconnlistener_get_base(listener);
            struct bufferevent *bev = bufferevent_socket_new(
                    base, fd, BEV_OPT_CLOSE_ON_FREE);

            bufferevent_setcb(bev, echo_read_cb, NULL, echo_event_cb, NULL);

            bufferevent_enable(bev, EV_READ|EV_WRITE);
    }

    static void
    accept_error_cb(struct evconnlistener *listener, void *ctx)
    {
            struct event_base *base = evconnlistener_get_base(listener);
            int err = EVUTIL_SOCKET_ERROR();
            fprintf(stderr, "Got an error %d (%s) on the listener. "
                    "Shutting down. ", err, evutil_socket_error_to_string(err));

            event_base_loopexit(base, NULL);
    }

    int
    main(int argc, char **argv)
    {
            struct event_base *base;
            struct evconnlistener *listener;
            struct sockaddr_in sin;

            int port = 9876;

            if (argc > 1) {
                    port = atoi(argv[1]);
            }
            if (port<=0 || port>65535) {
                    puts("Invalid port");
                    return 1;
            }

            base = event_base_new();
            if (!base) {
                    puts("Couldn't open event base");
                    return 1;
            }

            /* Clear the sockaddr before using it, in case there are extra         * platform-specific fields that can mess us up. */
            memset(&sin, 0, sizeof(sin));
            /* This is an INET address */
            sin.sin_family = AF_INET;
            /* Listen on 0.0.0.0 */
            sin.sin_addr.s_addr = htonl(0);
            /* Listen on the given port. */
            sin.sin_port = htons(port);

            listener = evconnlistener_new_bind(base, accept_conn_cb, NULL,
                LEV_OPT_CLOSE_ON_FREE|LEV_OPT_REUSEABLE, -1,
                (struct sockaddr*)&sin, sizeof(sin));
            if (!listener) {
                    perror("Couldn't create listener");
                    return 1;
            }
            evconnlistener_set_error_cb(listener, accept_error_cb);

            event_base_dispatch(base);
            return 0;
    }
     
     
     
  • 相关阅读:
    Text Link Ads 注册[赚钱一]
    Linux文件系统中的链接
    C++虚函数和纯虚函数(1)
    Android init reading tips
    Android上GDB的使用
    What is prelink?
    Linux fork哪些被继承,哪些不被继承
    为什么x86 Linux程序起始地址是从0x08048000开始的?
    Android应用开发的插件化 模块化
    C++拷贝构造函数(深拷贝、浅拷贝)
  • 原文地址:https://www.cnblogs.com/manziluo/p/5789641.html
Copyright © 2011-2022 走看看