Edit

kc3-lang/libevent/include/event2/util.h

Branch :

  • Show log

    Commit

  • Author : Nick Mathewson
    Date : 2009-02-11 17:23:32
    Hash : acaf65c3
    Message : Make evutil_parse_sockaddr_port give a useful socket-length output. svn:r1118

  • include/event2/util.h
  • /*
     * Copyright (c) 2007-2009 Niels Provos and Nick Mathewson
     *
     * 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 name of the author may not be used to endorse or promote products
     *    derived from this software without specific prior written permission.
     *
     * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS 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 AUTHOR 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.
     */
    #ifndef _EVENT2_UTIL_H_
    #define _EVENT2_UTIL_H_
    
    /** @file event2/util.h
    
      Common convenience functions for cross-platform portability and
      related socket manipulations.
    
     */
    
    #ifdef __cplusplus
    extern "C" {
    #endif
    
    #include <event-config.h>
    #ifdef _EVENT_HAVE_SYS_TIME_H
    #include <sys/time.h>
    #endif
    #ifdef _EVENT_HAVE_STDINT_H
    #include <stdint.h>
    #elif defined(_EVENT_HAVE_INTTYPES_H)
    #include <inttypes.h>
    #endif
    #ifdef _EVENT_HAVE_SYS_TYPES_H
    #include <sys/types.h>
    #endif
    #ifdef _EVENT_HAVE_STDDEF_H
    #include <stddef.h>
    #endif
    #include <stdarg.h>
    
    #ifdef _EVENT_HAVE_UINT64_T
    #define ev_uint64_t uint64_t
    #define ev_int64_t int64_t
    #elif defined(WIN32)
    #define ev_uint64_t unsigned __int64
    #define ev_int64_t signed __int64
    #elif _EVENT_SIZEOF_LONG_LONG == 8
    #define ev_uint64_t unsigned long long
    #define ev_int64_t long long
    #elif _EVENT_SIZEOF_LONG == 8
    #define ev_uint64_t unsigned long
    #define ev_int64_t long
    #else
    #error "No way to define ev_uint64_t"
    #endif
    
    #ifdef _EVENT_HAVE_UINT32_T
    #define ev_uint32_t uint32_t
    #elif defined(WIN32)
    #define ev_uint32_t unsigned int
    #elif _EVENT_SIZEOF_LONG == 4
    #define ev_uint32_t unsigned long
    #elif _EVENT_SIZEOF_INT == 4
    #define ev_uint32_t unsigned int
    #else
    #error "No way to define ev_uint32_t"
    #endif
    
    #ifdef _EVENT_HAVE_UINT16_T
    #define ev_uint16_t uint16_t
    #elif defined(WIN32)
    #define ev_uint16_t unsigned short
    #elif _EVENT_SIZEOF_INT == 2
    #define ev_uint16_t unsigned int
    #elif _EVENT_SIZEOF_SHORT == 2
    #define ev_uint16_t unsigned short
    #else
    #error "No way to define ev_uint16_t"
    #endif
    
    #ifdef _EVENT_HAVE_UINT8_T
    #define ev_uint8_t uint8_t
    #else
    #define ev_uint8_t unsigned char
    #endif
    
    #ifdef WIN32
    /** A type wide enough to hold the output of "socket()" or "accept()".  On
     * Windows, this is an intptr_t; elsewhere, it is an int. */
    #define evutil_socket_t intptr_t
    #else
    #define evutil_socket_t int
    #endif
    
    /** Create two new sockets that are connected to each other.  On Unix, this
        simply calls socketpair().  On Windows, it uses the loopback network
        interface on 127.0.0.1, and only AF_INET,SOCK_STREAM are supported.
    
        Parameters and return values are as for socketpair()
    */
    int evutil_socketpair(int d, int type, int protocol, evutil_socket_t sv[2]);
    /** Do platform-specific operations as needed to make a socket nonblocking.
    
        @param sock The socket to make nonblocking
        @return 0 on success, -1 on failure
     */
    int evutil_make_socket_nonblocking(evutil_socket_t sock);
    
    /** Do platform-specific operations on a listener socket to make sure that
        another program will be able to bind this address right after we've
        closed the listener
    
        @param sock The socket to make reuseabla
        @return 0 on success, -1 on failure
     */
    int evutil_make_listen_socket_reuseable(evutil_socket_t);
    
    #ifdef WIN32
    /** Do the platform-specific call needed to close a socket returned from
        socket() or accept(). */
    #define EVUTIL_CLOSESOCKET(s) closesocket(s)
    #else
    /** Do the platform-specific call needed to close a socket returned from
        socket() or accept(). */
    #define EVUTIL_CLOSESOCKET(s) close(s)
    #endif
    
    /* Winsock handles socket errors differently from the rest of the world.
     * Elsewhere, a socket error is like any other error and is stored in errno.
     * But winsock functions require you to retrieve the error with a special
     * function, and don't let you use strerror for the error codes.  And handling
     * EWOULD block is ... different. */
    
    #ifdef WIN32
    /** Return the most recent socket error.  Not idempotent on all platforms. */
    #define EVUTIL_SOCKET_ERROR() WSAGetLastError()
    /** Replace the most recent socket error with errcode */
    #define EVUTIL_SET_SOCKET_ERROR(errcode)		\
    	do { WSASetLastError(errcode); } while (0)
    /** Return the most recent socket error to occur on sock. */
    int evutil_socket_geterror(evutil_socket_t sock);
    /** Convert a socket error to a string. */
    const char *evutil_socket_error_to_string(int errcode);
    #else
    #define EVUTIL_SOCKET_ERROR() (errno)
    #define EVUTIL_SET_SOCKET_ERROR(errcode)		\
    		do { errno = (errcode); } while (0)
    #define evutil_socket_geterror(sock) (errno)
    #define evutil_socket_error_to_string(errcode) (strerror(errcode))
    #endif
    
    /*
     * Manipulation macros for struct timeval
     */
    #ifdef _EVENT_HAVE_TIMERADD
    #define evutil_timeradd(tvp, uvp, vvp) timeradd((tvp), (uvp), (vvp))
    #define evutil_timersub(tvp, uvp, vvp) timersub((tvp), (uvp), (vvp))
    #else
    #define evutil_timeradd(tvp, uvp, vvp)					\
    	do {								\
    		(vvp)->tv_sec = (tvp)->tv_sec + (uvp)->tv_sec;		\
    		(vvp)->tv_usec = (tvp)->tv_usec + (uvp)->tv_usec;       \
    		if ((vvp)->tv_usec >= 1000000) {			\
    			(vvp)->tv_sec++;				\
    			(vvp)->tv_usec -= 1000000;			\
    		}							\
    	} while (0)
    #define	evutil_timersub(tvp, uvp, vvp)					\
    	do {								\
    		(vvp)->tv_sec = (tvp)->tv_sec - (uvp)->tv_sec;		\
    		(vvp)->tv_usec = (tvp)->tv_usec - (uvp)->tv_usec;	\
    		if ((vvp)->tv_usec < 0) {				\
    			(vvp)->tv_sec--;				\
    			(vvp)->tv_usec += 1000000;			\
    		}							\
    	} while (0)
    #endif /* !_EVENT_HAVE_HAVE_TIMERADD */
    
    #ifdef _EVENT_HAVE_TIMERCLEAR
    #define evutil_timerclear(tvp) timerclear(tvp)
    #else
    #define	evutil_timerclear(tvp)	(tvp)->tv_sec = (tvp)->tv_usec = 0
    #endif
    
    #define	evutil_timercmp(tvp, uvp, cmp)					\
    	(((tvp)->tv_sec == (uvp)->tv_sec) ?				\
    	 ((tvp)->tv_usec cmp (uvp)->tv_usec) :				\
    	 ((tvp)->tv_sec cmp (uvp)->tv_sec))
    
    #ifdef _EVENT_HAVE_TIMERISSET
    #define evutil_timerisset(tvp) timerisset(tvp)
    #else
    #define	evutil_timerisset(tvp)	((tvp)->tv_sec || (tvp)->tv_usec)
    #endif
    
    #ifdef offsetof
    #define evutil_offsetof(type, field) offsetof(type, field)
    #else
    #define evutil_offsetof(type, field) ((off_t)(&((type *)0)->field))
    #endif
    
    /* big-int related functions */
    /** Parse a 64-bit value from a string.  Arguments are as for strtol. */
    ev_int64_t evutil_strtoll(const char *s, char **endptr, int base);
    
    /* Replacement for gettimeofday on platforms that lack it. */
    #ifdef _EVENT_HAVE_GETTIMEOFDAY
    #define evutil_gettimeofday(tv, tz) gettimeofday((tv), (tz))
    #else
    int evutil_gettimeofday(struct timeval *tv, struct timezone *tz);
    #endif
    
    #ifdef __GNUC__
    /** Helper macro; used to tell the compiler that a given function takes a
     * printf-like format string as argument number 'a', and a set of printf-like
     * arguments starting in argument 'b'. */
    #define EVUTIL_CHECK_FMT(a,b) __attribute__((format(printf, a, b)))
    #else
    #define EVUTIL_CHECK_FMT(a,b)
    #endif
    
    /** Replacement for snprintf to get consistent behavior on platforms for
        which the return value of snprintf does not conform to C99.
     */
    int evutil_snprintf(char *buf, size_t buflen, const char *format, ...)
    	EVUTIL_CHECK_FMT(3,4);
    int evutil_vsnprintf(char *buf, size_t buflen, const char *format, va_list ap);
    
    /** Replacement for inet_ntop for platforms which lack it. */
    const char *evutil_inet_ntop(int af, const void *src, char *dst, size_t len);
    /** Replacement for inet_pton for platforms which lack it. */
    int evutil_inet_pton(int af, const char *src, void *dst);
    struct sockaddr;
    
    /** Parse an IPv4 or IPv6 address, with optional port, from a string.
    
        Recognized formats are:
        - [IPv6Address]:port
        - [IPv6Address]
        - IPv6Address
        - IPv4Address:port
        - IPv4Address
    
        If no port is specified, the port in the output is set to 0.
    
        @param str The string to parse.
        @param out A struct sockaddr to hold the result.  This should probably be
           a struct sockaddr_storage.
        @param outlen A pointer to the number of bytes that that 'out' can safely
           hold.  Set to the number of bytes used in 'out' on sucess.
        @return -1 if the address is not well-formed, if the port is out of range,
           or if out is not large enough to hold the result.  Otherwise returns
           0 on success.
    */
    int evutil_parse_sockaddr_port(const char *str, struct sockaddr *out, int *outlen);
    
    
    #ifdef __cplusplus
    }
    #endif
    
    #endif /* _EVUTIL_H_ */