/* ==================================================================== * 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 * . */ #include "networkio.h" #include "inherit.h" #include "apr_network_io.h" #include "apr_general.h" #include "apr_portable.h" #include "apr_lib.h" #include #include #include #include #include #include #include #include "os2calls.h" static apr_status_t socket_cleanup(void *sock) { apr_socket_t *thesocket = sock; if (thesocket->socketdes < 0) { return APR_EINVALSOCK; } if (soclose(thesocket->socketdes) == 0) { thesocket->socketdes = -1; return APR_SUCCESS; } else { return APR_OS2_STATUS(sock_errno()); } } static void set_socket_vars(apr_socket_t *sock, int family, int type) { sock->type = type; sock->local_addr->family = family; sock->local_addr->sa.sin.sin_family = family; sock->remote_addr->family = family; sock->remote_addr->sa.sin.sin_family = family; if (family == AF_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 == AF_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 } 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_DECLARE(apr_status_t) apr_socket_create(apr_socket_t **new, int family, int type, apr_pool_t *cont) { int downgrade = (family == AF_UNSPEC); if (family == AF_UNSPEC) { #if APR_HAVE_IPV6 family = AF_INET6; #else family = AF_INET; #endif } alloc_socket(new, cont); if ((*new) == NULL) { return APR_ENOMEM; } 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 && downgrade) { family = AF_INET; (*new)->socketdes = socket(family, type, 0); } #endif if ((*new)->socketdes < 0) { return APR_OS2_STATUS(sock_errno()); } set_socket_vars(*new, family, type); (*new)->timeout = -1; (*new)->nonblock = FALSE; apr_pool_cleanup_register((*new)->cntxt, (void *)(*new), socket_cleanup, apr_pool_cleanup_null); return APR_SUCCESS; } APR_DECLARE(apr_status_t) apr_shutdown(apr_socket_t *thesocket, apr_shutdown_how_e how) { if (shutdown(thesocket->socketdes, how) == 0) { return APR_SUCCESS; } else { return APR_OS2_STATUS(sock_errno()); } } APR_DECLARE(apr_status_t) apr_socket_close(apr_socket_t *thesocket) { apr_pool_cleanup_kill(thesocket->cntxt, thesocket, socket_cleanup); return socket_cleanup(thesocket); } APR_DECLARE(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 APR_OS2_STATUS(sock_errno()); else { sock->local_addr = sa; return APR_SUCCESS; } } APR_DECLARE(apr_status_t) apr_listen(apr_socket_t *sock, apr_int32_t backlog) { if (listen(sock->socketdes, backlog) == -1) return APR_OS2_STATUS(sock_errno()); else return APR_SUCCESS; } APR_DECLARE(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, SOCK_STREAM); (*new)->timeout = -1; (*new)->nonblock = FALSE; (*new)->socketdes = accept(sock->socketdes, (struct sockaddr *)&(*new)->remote_addr->sa, &(*new)->remote_addr->salen); if ((*new)->socketdes < 0) { return APR_OS2_STATUS(sock_errno()); } (*new)->remote_addr->port = ntohs((*new)->remote_addr->sa.sin.sin_port); apr_pool_cleanup_register((*new)->cntxt, (void *)(*new), socket_cleanup, apr_pool_cleanup_null); return APR_SUCCESS; } APR_DECLARE(apr_status_t) apr_connect(apr_socket_t *sock, apr_sockaddr_t *sa) { if ((connect(sock->socketdes, (struct sockaddr *)&sa->sa.sin, sa->salen) < 0) && (sock_errno() != SOCEINPROGRESS)) { return APR_OS2_STATUS(sock_errno()); } else { int namelen = sizeof(sock->local_addr->sa.sin); getsockname(sock->socketdes, (struct sockaddr *)&sock->local_addr->sa.sin, &namelen); sock->remote_addr = sa; return APR_SUCCESS; } } APR_DECLARE(apr_status_t) apr_socket_data_get(void **data, const char *key, apr_socket_t *socket) { return apr_pool_userdata_get(data, key, socket->cntxt); } APR_DECLARE(apr_status_t) apr_socket_data_set(apr_socket_t *socket, void *data, const char *key, apr_status_t (*cleanup) (void *)) { return apr_pool_userdata_set(data, key, cleanup, socket->cntxt); } APR_DECLARE(apr_status_t) apr_os_sock_get(apr_os_sock_t *thesock, apr_socket_t *sock) { *thesock = sock->socketdes; return APR_SUCCESS; } APR_DECLARE(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, os_sock_info->type); (*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) { 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_DECLARE(apr_status_t) apr_os_sock_put(apr_socket_t **sock, apr_os_sock_t *thesock, apr_pool_t *cont) { if (cont == NULL) { return APR_ENOPOOL; } if ((*sock) == NULL) { alloc_socket(sock, cont); set_socket_vars(*sock, AF_INET, SOCK_STREAM); } (*sock)->socketdes = *thesock; return APR_SUCCESS; } APR_IMPLEMENT_SET_INHERIT(socket, inherit, cntxt, socket_cleanup) APR_IMPLEMENT_UNSET_INHERIT(socket, inherit, cntxt, socket_cleanup)