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

📄 sendrecv.c

📁 log4cxx 0.10 unix下编译包
💻 C
字号:
/* Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements.  See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You under the Apache License, Version 2.0 * (the "License"); you may not use this file except in compliance with * the License.  You may obtain a copy of the License at * *     http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */#include "apr_arch_networkio.h"#include "apr_errno.h"#include "apr_general.h"#include "apr_network_io.h"#include "apr_lib.h"#include "apr_arch_file_io.h"#if APR_HAVE_TIME_H#include <time.h>#endif/* MAX_SEGMENT_SIZE is the maximum amount of data that will be sent to a client * in one call of TransmitFile. This number must be small enough to give the  * slowest client time to receive the data before the socket timeout triggers. * The same problem can exist with apr_socket_send(). In that case, we rely on * the application to adjust socket timeouts and max send segment  * sizes appropriately. * For example, Apache will in most cases call apr_socket_send() with less * than 8193 bytes. */#define MAX_SEGMENT_SIZE 65536#define WSABUF_ON_STACK 50APR_DECLARE(apr_status_t) apr_socket_send(apr_socket_t *sock, const char *buf,                                          apr_size_t *len){    apr_ssize_t rv;    WSABUF wsaData;    int lasterror;    DWORD dwBytes = 0;    wsaData.len = (u_long)*len;    wsaData.buf = (char*) buf;#ifndef _WIN32_WCE    rv = WSASend(sock->socketdes, &wsaData, 1, &dwBytes, 0, NULL, NULL);#else    rv = send(sock->socketdes, wsaData.buf, wsaData.len, 0);    dwBytes = rv;#endif    if (rv == SOCKET_ERROR) {        lasterror = apr_get_netos_error();        *len = 0;        return lasterror;    }    *len = dwBytes;    return APR_SUCCESS;}APR_DECLARE(apr_status_t) apr_socket_recv(apr_socket_t *sock, char *buf,                                          apr_size_t *len) {    apr_ssize_t rv;    WSABUF wsaData;    int lasterror;    DWORD dwBytes = 0;    DWORD flags = 0;    wsaData.len = (u_long)*len;    wsaData.buf = (char*) buf;#ifndef _WIN32_WCE    rv = WSARecv(sock->socketdes, &wsaData, 1, &dwBytes, &flags, NULL, NULL);#else    rv = recv(sock->socketdes, wsaData.buf, wsaData.len, 0);    dwBytes = rv;#endif    if (rv == SOCKET_ERROR) {        lasterror = apr_get_netos_error();        *len = 0;        return lasterror;    }    *len = dwBytes;    return dwBytes == 0 ? APR_EOF : APR_SUCCESS;}APR_DECLARE(apr_status_t) apr_socket_sendv(apr_socket_t *sock,                                           const struct iovec *vec,                                           apr_int32_t in_vec, apr_size_t *nbytes){    apr_status_t rc = APR_SUCCESS;    apr_ssize_t rv;    apr_size_t cur_len;    apr_int32_t nvec = 0;    int i, j = 0;    DWORD dwBytes = 0;    WSABUF *pWsaBuf;    for (i = 0; i < in_vec; i++) {        cur_len = vec[i].iov_len;        nvec++;        while (cur_len > APR_DWORD_MAX) {            nvec++;            cur_len -= APR_DWORD_MAX;        }     }    pWsaBuf = (nvec <= WSABUF_ON_STACK) ? _alloca(sizeof(WSABUF) * (nvec))                                         : malloc(sizeof(WSABUF) * (nvec));    if (!pWsaBuf)        return APR_ENOMEM;    for (i = 0; i < in_vec; i++) {        char * base = vec[i].iov_base;        cur_len = vec[i].iov_len;                do {            if (cur_len > APR_DWORD_MAX) {                pWsaBuf[j].buf = base;                pWsaBuf[j].len = APR_DWORD_MAX;                cur_len -= APR_DWORD_MAX;                base += APR_DWORD_MAX;            }            else {                pWsaBuf[j].buf = base;                pWsaBuf[j].len = (DWORD)cur_len;                cur_len = 0;            }            j++;        } while (cur_len > 0);    }#ifndef _WIN32_WCE    rv = WSASend(sock->socketdes, pWsaBuf, nvec, &dwBytes, 0, NULL, NULL);    if (rv == SOCKET_ERROR) {        rc = apr_get_netos_error();    }#else    for (i = 0; i < nvec; i++) {        rv = send(sock->socketdes, pWsaBuf[i].buf, pWsaBuf[i].len, 0);        if (rv == SOCKET_ERROR) {            rc = apr_get_netos_error();            break;        }        dwBytes += rv;    }#endif    if (nvec > WSABUF_ON_STACK)         free(pWsaBuf);    *nbytes = dwBytes;    return rc;}APR_DECLARE(apr_status_t) apr_socket_sendto(apr_socket_t *sock,                                            apr_sockaddr_t *where,                                            apr_int32_t flags, const char *buf,                                             apr_size_t *len){    apr_ssize_t rv;    rv = sendto(sock->socketdes, buf, (int)*len, flags,                 (const struct sockaddr*)&where->sa,                 where->salen);    if (rv == SOCKET_ERROR) {        *len = 0;        return apr_get_netos_error();    }    *len = rv;    return APR_SUCCESS;}APR_DECLARE(apr_status_t) apr_socket_recvfrom(apr_sockaddr_t *from,                                               apr_socket_t *sock,                                              apr_int32_t flags,                                               char *buf, apr_size_t *len){    apr_ssize_t rv;    rv = recvfrom(sock->socketdes, buf, (int)*len, flags,                   (struct sockaddr*)&from->sa, &from->salen);    if (rv == SOCKET_ERROR) {        (*len) = 0;        return apr_get_netos_error();    }    (*len) = rv;    if (rv == 0 && sock->type == SOCK_STREAM)        return APR_EOF;    return APR_SUCCESS;}static apr_status_t collapse_iovec(char **off, apr_size_t *len,                                    struct iovec *iovec, int numvec,                                    char *buf, apr_size_t buflen){    if (numvec == 1) {        *off = iovec[0].iov_base;        *len = iovec[0].iov_len;    }    else {        int i;        for (i = 0; i < numvec; i++) {            *len += iovec[i].iov_len;        }        if (*len > buflen) {            *len = 0;            return APR_INCOMPLETE;        }        *off = buf;        for (i = 0; i < numvec; i++) {            memcpy(buf, iovec[i].iov_base, iovec[i].iov_len);            buf += iovec[i].iov_len;        }    }    return APR_SUCCESS;}#if APR_HAS_SENDFILE/* * apr_status_t apr_socket_sendfile(apr_socket_t *, apr_file_t *, apr_hdtr_t *,  *                                 apr_off_t *, apr_size_t *, apr_int32_t flags) *    Send a file from an open file descriptor to a socket, along with  *    optional headers and trailers * arg 1) The socket to which we're writing * arg 2) The open file from which to read * arg 3) A structure containing the headers and trailers to send * arg 4) Offset into the file where we should begin writing * arg 5) Number of bytes to send out of the file * arg 6) APR flags that are mapped to OS specific flags */APR_DECLARE(apr_status_t) apr_socket_sendfile(apr_socket_t *sock,                                               apr_file_t *file,                                              apr_hdtr_t *hdtr,                                              apr_off_t *offset,                                              apr_size_t *len,                                              apr_int32_t flags) {    apr_status_t status = APR_SUCCESS;    apr_status_t rv;    apr_off_t curoff = *offset;    DWORD dwFlags = 0;    apr_size_t nbytes;    TRANSMIT_FILE_BUFFERS tfb, *ptfb = NULL;    int ptr = 0;    apr_size_t bytes_to_send;   /* Bytes to send out of the file (not including headers) */    int disconnected = 0;    int sendv_trailers = 0;    char hdtrbuf[4096];    if (apr_os_level < APR_WIN_NT) {        return APR_ENOTIMPL;    }    /* Use len to keep track of number of total bytes sent (including headers) */    bytes_to_send = *len;    *len = 0;    /* Handle the goofy case of sending headers/trailers and a zero byte file */    if (!bytes_to_send && hdtr) {        if (hdtr->numheaders) {            rv = apr_socket_sendv(sock, hdtr->headers, hdtr->numheaders,                                   &nbytes);            if (rv != APR_SUCCESS)                return rv;            *len += nbytes;        }        if (hdtr->numtrailers) {            rv = apr_socket_sendv(sock, hdtr->trailers, hdtr->numtrailers,                                  &nbytes);            if (rv != APR_SUCCESS)                return rv;            *len += nbytes;        }        return APR_SUCCESS;    }    memset(&tfb, '\0', sizeof (tfb));    /* Collapse the headers into a single buffer */    if (hdtr && hdtr->numheaders) {        apr_size_t head_length = tfb.HeadLength;        ptfb = &tfb;        nbytes = 0;        rv = collapse_iovec((char **)&ptfb->Head, &head_length,                             hdtr->headers, hdtr->numheaders,                             hdtrbuf, sizeof(hdtrbuf));        tfb.HeadLength = (DWORD)head_length;        /* If not enough buffer, punt to sendv */        if (rv == APR_INCOMPLETE) {            rv = apr_socket_sendv(sock, hdtr->headers, hdtr->numheaders, &nbytes);            if (rv != APR_SUCCESS)                return rv;            *len += nbytes;            ptfb = NULL;        }    }    /* Initialize the overlapped structure used on TransmitFile     */    if (!sock->overlapped) {        sock->overlapped = apr_pcalloc(sock->pool, sizeof(OVERLAPPED));        sock->overlapped->hEvent = CreateEvent(NULL, FALSE, FALSE, NULL);    }    while (bytes_to_send) {        DWORD xmitbytes;        if (bytes_to_send > MAX_SEGMENT_SIZE) {            xmitbytes = MAX_SEGMENT_SIZE;        }        else {            /* Last call to TransmitFile() */            xmitbytes = (DWORD)bytes_to_send;            /* Collapse the trailers into a single buffer */            if (hdtr && hdtr->numtrailers) {                apr_size_t tail_length = tfb.TailLength;                ptfb = &tfb;                rv = collapse_iovec((char**) &ptfb->Tail, &tail_length,                                    hdtr->trailers, hdtr->numtrailers,                                    hdtrbuf + ptfb->HeadLength,                                    sizeof(hdtrbuf) - ptfb->HeadLength);                tfb.TailLength = (DWORD)tail_length;                if (rv == APR_INCOMPLETE) {                    /* If not enough buffer, punt to sendv, later */                    sendv_trailers = 1;                }            }            /* Disconnect the socket after last send */            if ((flags & APR_SENDFILE_DISCONNECT_SOCKET)                    && !sendv_trailers) {                dwFlags |= TF_REUSE_SOCKET;                dwFlags |= TF_DISCONNECT;                disconnected = 1;            }        }        sock->overlapped->Offset = (DWORD)(curoff);#if APR_HAS_LARGE_FILES        sock->overlapped->OffsetHigh = (DWORD)(curoff >> 32);#endif          /* XXX BoundsChecker claims dwFlags must not be zero. */        rv = TransmitFile(sock->socketdes,  /* socket */                          file->filehand, /* open file descriptor of the file to be sent */                          xmitbytes,      /* number of bytes to send. 0=send all */                          0,              /* Number of bytes per send. 0=use default */                          sock->overlapped,    /* OVERLAPPED structure */                          ptfb,           /* header and trailer buffers */                          dwFlags);       /* flags to control various aspects of TransmitFile */        if (!rv) {            status = apr_get_netos_error();            if ((status == APR_FROM_OS_ERROR(ERROR_IO_PENDING)) ||                (status == APR_FROM_OS_ERROR(WSA_IO_PENDING)))             {                rv = WaitForSingleObject(sock->overlapped->hEvent,                                          (DWORD)(sock->timeout >= 0                                                  ? sock->timeout_ms : INFINITE));                if (rv == WAIT_OBJECT_0) {                    status = APR_SUCCESS;                    if (!disconnected) {                        if (!WSAGetOverlappedResult(sock->socketdes,                                                    sock->overlapped,                                                    &xmitbytes,                                                    FALSE,                                                    &dwFlags)) {                            status = apr_get_netos_error();                        }                        /* Ugly code alert: WSAGetOverlappedResult returns                         * a count of all bytes sent. This loop only                         * tracks bytes sent out of the file.                         */                        else if (ptfb) {                            xmitbytes -= (ptfb->HeadLength + ptfb->TailLength);                        }                    }                }                else if (rv == WAIT_TIMEOUT) {                    status = APR_FROM_OS_ERROR(WAIT_TIMEOUT);                }                else if (rv == WAIT_ABANDONED) {                    /* Hummm... WAIT_ABANDONDED is not an error code. It is                     * a return specific to the Win32 WAIT functions that                     * indicates that a thread exited while holding a                     * mutex. Should consider triggering an assert                     * to detect the condition...                     */                    status = APR_FROM_OS_ERROR(WAIT_TIMEOUT);                }                else                    status = apr_get_os_error();            }        }        if (status != APR_SUCCESS)            break;        bytes_to_send -= xmitbytes;        curoff += xmitbytes;        *len += xmitbytes;        /* Adjust len for any headers/trailers sent */        if (ptfb) {            *len += (ptfb->HeadLength + ptfb->TailLength);            memset(&tfb, '\0', sizeof (tfb));            ptfb = NULL;        }    }    if (status == APR_SUCCESS) {        if (sendv_trailers) {            rv = apr_socket_sendv(sock, hdtr->trailers, hdtr->numtrailers, &nbytes);            if (rv != APR_SUCCESS)                return rv;            *len += nbytes;        }            /* Mark the socket as disconnected, but do not close it.         * Note: The application must have stored the socket prior to making         * the call to apr_socket_sendfile in order to either reuse it          * or close it.         */        if (disconnected) {            sock->disconnected = 1;            sock->socketdes = INVALID_SOCKET;        }    }    return status;}#endif

⌨️ 快捷键说明

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