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

📄 sockets.c

📁 Apache V2.0.15 Alpha For Linuxhttpd-2_0_15-alpha.tar.Z
💻 C
字号:
/* ==================================================================== * The Apache Software License, Version 1.1 * * Copyright (c) 2000-2001 The Apache Software Foundation.  All rights * reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright *    notice, this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright *    notice, this list of conditions and the following disclaimer in *    the documentation and/or other materials provided with the *    distribution. * * 3. The end-user documentation included with the redistribution, *    if any, must include the following acknowledgment: *       "This product includes software developed by the *        Apache Software Foundation (http://www.apache.org/)." *    Alternately, this acknowledgment may appear in the software itself, *    if and wherever such third-party acknowledgments normally appear. * * 4. The names "Apache" and "Apache Software Foundation" must *    not be used to endorse or promote products derived from this *    software without prior written permission. For written *    permission, please contact apache@apache.org. * * 5. Products derived from this software may not be called "Apache", *    nor may "Apache" appear in their name, without prior written *    permission of the Apache Software Foundation. * * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED.  IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * ==================================================================== * * This software consists of voluntary contributions made by many * individuals on behalf of the Apache Software Foundation.  For more * information on the Apache Software Foundation, please see * <http://www.apache.org/>. */#include "networkio.h"#include "apr_portable.h"static apr_status_t socket_cleanup(void *sock){    apr_socket_t *thesocket = sock;#if defined(BEOS) && !defined(BEOS_BONE)    if (closesocket(thesocket->socketdes) == 0) {#else    if (close(thesocket->socketdes) == 0) {#endif        thesocket->socketdes = -1;        return APR_SUCCESS;    }    else {        return errno;    }}static void set_socket_vars(apr_socket_t *sock, int family){    sock->local_addr->sa.sin.sin_family = family;    sock->remote_addr->sa.sin.sin_family = family;    if (family == APR_INET) {        sock->local_addr->salen = sizeof(struct sockaddr_in);        sock->local_addr->addr_str_len = 16;        sock->local_addr->ipaddr_ptr = &(sock->local_addr->sa.sin.sin_addr);        sock->local_addr->ipaddr_len = sizeof(struct in_addr);        sock->remote_addr->salen = sizeof(struct sockaddr_in);        sock->remote_addr->addr_str_len = 16;        sock->remote_addr->ipaddr_ptr = &(sock->remote_addr->sa.sin.sin_addr);        sock->remote_addr->ipaddr_len = sizeof(struct in_addr);    }#if APR_HAVE_IPV6    else if (family == APR_INET6) {        sock->local_addr->salen = sizeof(struct sockaddr_in6);        sock->local_addr->addr_str_len = 46;        sock->local_addr->ipaddr_ptr = &(sock->local_addr->sa.sin6.sin6_addr);        sock->local_addr->ipaddr_len = sizeof(struct in6_addr);        sock->remote_addr->salen = sizeof(struct sockaddr_in6);        sock->remote_addr->addr_str_len = 46;        sock->remote_addr->ipaddr_ptr = &(sock->remote_addr->sa.sin6.sin6_addr);        sock->remote_addr->ipaddr_len = sizeof(struct in6_addr);    }#endif    sock->netmask = 0;#if defined(BEOS) && !defined(BEOS_BONE)    /* BeOS pre-BONE has TCP_NODELAY on by default and it can't be     * switched off!     */    sock->netmask |= APR_TCP_NODELAY;#endif}                                                                                                  static void alloc_socket(apr_socket_t **new, apr_pool_t *p){    *new = (apr_socket_t *)apr_pcalloc(p, sizeof(apr_socket_t));    (*new)->cntxt = p;    (*new)->local_addr = (apr_sockaddr_t *)apr_pcalloc((*new)->cntxt,                                                       sizeof(apr_sockaddr_t));    (*new)->local_addr->pool = p;    (*new)->remote_addr = (apr_sockaddr_t *)apr_pcalloc((*new)->cntxt,                                                        sizeof(apr_sockaddr_t));    (*new)->remote_addr->pool = p;}apr_status_t apr_socket_create(apr_socket_t **new, int ofamily, int type,                               apr_pool_t *cont){    int family = ofamily;    if (family == APR_UNSPEC) {#if APR_HAVE_IPV6        family = APR_INET6;#else        family = APR_INET;#endif    }    alloc_socket(new, cont);    if ((*new)->local_addr == NULL || (*new)->remote_addr == NULL) {        return APR_ENOMEM;    }    (*new)->socketdes = socket(family, type, 0);#if APR_HAVE_IPV6    if ((*new)->socketdes < 0 && ofamily == APR_UNSPEC) {        family = APR_INET;        (*new)->socketdes = socket(family, type, 0);    }#endif    if ((*new)->socketdes < 0) {        return errno;    }    set_socket_vars(*new, family);    (*new)->timeout = -1;    apr_pool_cleanup_register((*new)->cntxt, (void *)(*new),                         socket_cleanup, apr_pool_cleanup_null);    return APR_SUCCESS;} apr_status_t apr_shutdown(apr_socket_t *thesocket, apr_shutdown_how_e how){    return (shutdown(thesocket->socketdes, how) == -1) ? errno : APR_SUCCESS;}apr_status_t apr_socket_close(apr_socket_t *thesocket){    apr_pool_cleanup_kill(thesocket->cntxt, thesocket, socket_cleanup);    return socket_cleanup(thesocket);}apr_status_t apr_bind(apr_socket_t *sock, apr_sockaddr_t *sa){    if (bind(sock->socketdes,              (struct sockaddr *)&sa->sa, sa->salen) == -1) {        return errno;    }    else {        sock->local_addr = sa;        /* XXX IPv6 - this assumes sin_port and sin6_port at same offset */        if (sock->local_addr->sa.sin.sin_port == 0) { /* no need for ntohs() when comparing w/ 0 */            sock->local_port_unknown = 1; /* kernel got us an ephemeral port */        }        return APR_SUCCESS;    }}apr_status_t apr_listen(apr_socket_t *sock, apr_int32_t backlog){    if (listen(sock->socketdes, backlog) == -1)        return errno;    else        return APR_SUCCESS;}apr_status_t apr_accept(apr_socket_t **new, apr_socket_t *sock, apr_pool_t *connection_context){    alloc_socket(new, connection_context);    set_socket_vars(*new, sock->local_addr->sa.sin.sin_family);#ifndef HAVE_POLL    (*new)->connected = 1;#endif    (*new)->timeout = -1;        (*new)->remote_addr->salen = sizeof((*new)->remote_addr->sa);    (*new)->socketdes = accept(sock->socketdes,                                (struct sockaddr *)&(*new)->remote_addr->sa,                               &(*new)->remote_addr->salen);    if ((*new)->socketdes < 0) {        return errno;    }    *(*new)->local_addr = *sock->local_addr;    /* fix up any pointers which are no longer valid */    if (sock->local_addr->sa.sin.sin_family == AF_INET) {        (*new)->local_addr->ipaddr_ptr = &(*new)->local_addr->sa.sin.sin_addr;    }#if APR_HAVE_IPV6    else if (sock->local_addr->sa.sin.sin_family == AF_INET6) {        (*new)->local_addr->ipaddr_ptr = &(*new)->local_addr->sa.sin6.sin6_addr;    }#endif    if (sock->local_port_unknown) {        /* not likely for a listening socket, but theoretically possible :) */        (*new)->local_port_unknown = 1;    }    if (sock->local_interface_unknown ||        /* XXX IPv6 issue */        sock->local_addr->sa.sin.sin_addr.s_addr == 0) {        /* If the interface address inside the listening socket's local_addr wasn't          * up-to-date, we don't know local interface of the connected socket either.         *         * If the listening socket was not bound to a specific interface, we         * don't know the local_addr of the connected socket.         */        (*new)->local_interface_unknown = 1;    }    apr_pool_cleanup_register((*new)->cntxt, (void *)(*new),                         socket_cleanup, apr_pool_cleanup_null);    return APR_SUCCESS;}apr_status_t apr_connect(apr_socket_t *sock, apr_sockaddr_t *sa){    if ((sock->socketdes < 0) || (!sock->remote_addr)) {        return APR_ENOTSOCK;    }    if ((connect(sock->socketdes,                 (const struct sockaddr *)&sa->sa.sin,                 sa->salen) < 0) &&        (errno != EINPROGRESS)) {        return errno;    }    else {        sock->remote_addr = sa;        /* XXX IPv6 assumes sin_port and sin6_port at same offset */        if (sock->local_addr->sa.sin.sin_port == 0) {            /* connect() got us an ephemeral port */            sock->local_port_unknown = 1;        }        /* XXX IPv6 to be handled better later... */        if (#if APR_HAVE_IPV6            sock->local_addr->sa.sin.sin_family == APR_INET6 ||#endif            sock->local_addr->sa.sin.sin_addr.s_addr == 0) {            /* not bound to specific local interface; connect() had to assign             * one for the socket             */            sock->local_interface_unknown = 1;        }#ifndef HAVE_POLL        sock->connected=1;#endif        return APR_SUCCESS;    }}apr_status_t apr_socket_data_get(void **data, const char *key, apr_socket_t *sock){    return apr_pool_userdata_get(data, key, sock->cntxt);}apr_status_t apr_socket_data_set(apr_socket_t *sock, void *data, const char *key,                              apr_status_t (*cleanup) (void *)){    return apr_pool_userdata_set(data, key, cleanup, sock->cntxt);}apr_status_t apr_os_sock_get(apr_os_sock_t *thesock, apr_socket_t *sock){    *thesock = sock->socketdes;    return APR_SUCCESS;}apr_status_t apr_os_sock_make(apr_socket_t **apr_sock,                               apr_os_sock_info_t *os_sock_info,                               apr_pool_t *cont){    alloc_socket(apr_sock, cont);    set_socket_vars(*apr_sock, os_sock_info->family);    (*apr_sock)->timeout = -1;    (*apr_sock)->socketdes = *os_sock_info->os_sock;    if (os_sock_info->local) {        memcpy(&(*apr_sock)->local_addr->sa.sin,                os_sock_info->local,                (*apr_sock)->local_addr->salen);    }    else {        (*apr_sock)->local_port_unknown = (*apr_sock)->local_interface_unknown = 1;    }    if (os_sock_info->remote) {#ifndef HAVE_POLL        (*apr_sock)->connected = 1;#endif        memcpy(&(*apr_sock)->remote_addr->sa.sin,                os_sock_info->remote,               (*apr_sock)->remote_addr->salen);    }            apr_pool_cleanup_register((*apr_sock)->cntxt, (void *)(*apr_sock),                         socket_cleanup, apr_pool_cleanup_null);    return APR_SUCCESS;}apr_status_t apr_os_sock_put(apr_socket_t **sock, apr_os_sock_t *thesock,                            apr_pool_t *cont){    if ((*sock) == NULL) {        alloc_socket(sock, cont);        /* XXX IPv6 figure out the family here! */        set_socket_vars(*sock, APR_INET);        (*sock)->timeout = -1;    }    (*sock)->local_port_unknown = (*sock)->local_interface_unknown = 1;    (*sock)->socketdes = *thesock;    return APR_SUCCESS;}

⌨️ 快捷键说明

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