⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 pollset-sample.c

📁 跨平台windowsunixlinux的c语言编程解决方案
💻 C
字号:
/** * apr tutorial sample code * http://dev.ariel-networks.com/apr/ */#ifdef HAVE_CONFIG_H#include <config.h>#endif#include <stdio.h>#include <assert.h>#include <apr_general.h>#include <apr_file_io.h>#include <apr_strings.h>#include <apr_network_io.h>#include <apr_poll.h>/* default listen port number */#define DEF_LISTEN_PORT		8081/* default socket backlog number. SOMAXCONN is a system default value */#define DEF_SOCKET_BACKLOG	SOMAXCONN#define DEF_POLLSET_NUM		32/* default socket timeout */#define DEF_POLL_TIMEOUT	(APR_USEC_PER_SEC * 30)/* default buffer size */#define BUFSIZE			4096/* useful macro */#define CRLF_STR		"\r\n"typedef struct _serv_ctx_t serv_ctx_t;/** * network event callback function type */typedef int (*socket_callback_t)(serv_ctx_t *serv_ctx, apr_pollset_t *pollset, apr_socket_t *sock);/** * network server context  */struct _serv_ctx_t {    enum {        SERV_RECV_REQUEST,        SERV_SEND_RESPONSE,    } status;    socket_callback_t cb_func;    apr_pool_t *mp;    /* recv ctx */    struct {        const char *filepath;        int is_firstline;    } recv;    /* send ctx */    struct {        char *send_buf;        apr_off_t offset;    } send;};static apr_socket_t* create_listen_sock(apr_pool_t *mp);static int do_accept(apr_pollset_t *pollset, apr_socket_t *lsock, apr_pool_t *mp);static int recv_req_cb(serv_ctx_t *serv_ctx, apr_pollset_t *pollset, apr_socket_t *sock);static int send_resp_cb(serv_ctx_t *serv_ctx, apr_pollset_t *pollset, apr_socket_t *sock);/** * @remark Error checks omitted */int main(int argc, const char *argv[]){    apr_status_t rv;    apr_pool_t *mp;    apr_socket_t *lsock;/* listening socket */    apr_pollset_t *pollset;    apr_int32_t num;    const apr_pollfd_t *ret_pfd;    apr_initialize();    apr_pool_create(&mp, NULL);    lsock = create_listen_sock(mp);    assert(lsock);        apr_pollset_create(&pollset, DEF_POLLSET_NUM, mp, 0);    {        apr_pollfd_t pfd = { mp, APR_POLL_SOCKET, APR_POLLIN, 0, { NULL }, NULL };        pfd.desc.s = lsock;        apr_pollset_add(pollset, &pfd);    }    while (1) {        rv = apr_pollset_poll(pollset, DEF_POLL_TIMEOUT, &num, &ret_pfd);        if (rv == APR_SUCCESS) {            int i;            assert(num > 0);            /* scan the active sockets */            for (i = 0; i < num; i++) {                if (ret_pfd[i].desc.s == lsock) {                    /* the listen socket is readable. that indicates we accepted a new connection */                    do_accept(pollset, lsock, mp);                } else {                    serv_ctx_t *serv_ctx = ret_pfd[i].client_data;                    socket_callback_t cb_func = serv_ctx->cb_func;                    cb_func(serv_ctx, pollset, ret_pfd[i].desc.s);                }            }        }    }        return 0;}static apr_socket_t* create_listen_sock(apr_pool_t *mp){    apr_status_t rv;    apr_socket_t *s;    apr_sockaddr_t *sa;        rv = apr_sockaddr_info_get(&sa, NULL, APR_INET, DEF_LISTEN_PORT, 0, mp);    if (rv != APR_SUCCESS) {	goto error;    }        rv = apr_socket_create(&s, sa->family, SOCK_STREAM, APR_PROTO_TCP, mp);    if (rv != APR_SUCCESS) {	goto error;    }    /* non-blocking socket */    apr_socket_opt_set(s, APR_SO_NONBLOCK, 1);    apr_socket_timeout_set(s, 0);    apr_socket_opt_set(s, APR_SO_REUSEADDR, 1);/* this is useful for a server(socket listening) process */    rv = apr_socket_bind(s, sa);    if (rv != APR_SUCCESS) {	goto error;    }    rv = apr_socket_listen(s, DEF_SOCKET_BACKLOG);    if (rv != APR_SUCCESS) {	goto error;    }    return s; error:    return NULL;}static int do_accept(apr_pollset_t *pollset, apr_socket_t *lsock, apr_pool_t *mp){    apr_socket_t *ns;/* accepted socket */    apr_status_t rv;                rv = apr_socket_accept(&ns, lsock, mp);    if (rv == APR_SUCCESS) {        serv_ctx_t *serv_ctx = apr_palloc(mp, sizeof(serv_ctx_t));        apr_pollfd_t pfd = { mp, APR_POLL_SOCKET, APR_POLLIN, 0, { NULL }, serv_ctx };        pfd.desc.s = ns;        /* at first, we expect requests, so we poll APR_POLLIN event */        serv_ctx->status = SERV_RECV_REQUEST;        serv_ctx->cb_func = recv_req_cb;        serv_ctx->recv.is_firstline = TRUE;        serv_ctx->mp = mp;        /* non-blocking socket. We can't expect that @ns inherits non-blocking mode from @lsock */        apr_socket_opt_set(ns, APR_SO_NONBLOCK, 1);        apr_socket_timeout_set(ns, 0);                apr_pollset_add(pollset, &pfd);    }    return TRUE;}static int recv_req_cb(serv_ctx_t *serv_ctx, apr_pollset_t *pollset, apr_socket_t *sock){    while (1) {	char buf[BUFSIZE];	apr_size_t len = sizeof(buf) - 1;/* -1 for a null-terminated */	apr_status_t rv = apr_socket_recv(sock, buf, &len);	if (rv == APR_EOF || len == 0) {	    break;        }        buf[len] = '\0';/* apr_socket_recv() doesn't return a null-terminated string */                if (serv_ctx->recv.is_firstline && strstr(buf, CRLF_STR)) {            char **tokens;            apr_tokenize_to_argv(buf, &tokens, serv_ctx->mp);            if (tokens[0] && tokens[1] && strcmp(tokens[0], "GET") == 0) {                serv_ctx->recv.filepath = apr_pstrdup(serv_ctx->mp, tokens[1]);            }            serv_ctx->recv.is_firstline = FALSE;        }        if (strstr(buf, CRLF_STR CRLF_STR)) {/* expect the end of the request. no guarantee */            /* status change (from read to write) */            apr_pollfd_t pfd = { serv_ctx->mp, APR_POLL_SOCKET, APR_POLLIN, 0, { NULL }, serv_ctx };            pfd.desc.s = sock;            apr_pollset_remove(pollset, &pfd);            pfd.reqevents = APR_POLLOUT;            apr_pollset_add(pollset, &pfd);            serv_ctx->status = SERV_SEND_RESPONSE;            serv_ctx->cb_func = send_resp_cb;            break;        }    }    return TRUE;}/** * Generate a response string. * @remark This would consume many memories for a large file. */static void gen_response(serv_ctx_t *serv_ctx){    apr_status_t rv;    apr_file_t *fp;    const char *filepath = serv_ctx->recv.filepath;        if ((rv = apr_file_open(&fp, filepath, APR_READ, APR_OS_DEFAULT, serv_ctx->mp)) == APR_SUCCESS) {        const char *resp_hdr;        apr_size_t len;        const char *resp_body;                apr_finfo_t finfo;        apr_file_info_get(&finfo, APR_FINFO_SIZE, fp);                resp_hdr = apr_psprintf(serv_ctx->mp, "HTTP/1.0 200 OK" CRLF_STR "Content-Length: %" APR_OFF_T_FMT CRLF_STR CRLF_STR, finfo.size);        resp_body = apr_pcalloc(serv_ctx->mp, finfo.size + 1);        len = finfo.size;        apr_file_read(fp, (void*)resp_body, &len);        serv_ctx->send.send_buf = apr_pstrcat(serv_ctx->mp, resp_hdr, resp_body, NULL);    } else {        serv_ctx->send.send_buf = "HTTP/1.0 404 Not Found" CRLF_STR CRLF_STR;    }}/** * Send a response to the client. */static int send_resp_cb(serv_ctx_t *serv_ctx, apr_pollset_t *pollset, apr_socket_t *sock){    if (!serv_ctx->send.send_buf) {        gen_response(serv_ctx);    }    {        apr_off_t offset = serv_ctx->send.offset;        const char *send_pt = &serv_ctx->send.send_buf[offset];        apr_size_t len = strlen(send_pt);        apr_size_t wlen = len;        apr_status_t rv;                rv = apr_socket_send(sock, send_pt, &wlen);        if (wlen == len) {            apr_pollfd_t pfd = { serv_ctx->mp, APR_POLL_SOCKET, APR_POLLOUT, 0, { NULL }, serv_ctx };            pfd.desc.s = sock;            apr_pollset_remove(pollset, &pfd);            apr_socket_close(sock);        } else {            serv_ctx->send.offset += wlen;        }    }        return TRUE;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -