michael@0: /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ michael@0: /* This Source Code Form is subject to the terms of the Mozilla Public michael@0: * License, v. 2.0. If a copy of the MPL was not distributed with this michael@0: * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ michael@0: michael@0: #include "nspr.h" michael@0: #include "prio.h" michael@0: #include "prinit.h" michael@0: #include "prprf.h" michael@0: #include "obsolete/probslet.h" michael@0: michael@0: #include "plerror.h" michael@0: michael@0: #ifdef XP_UNIX michael@0: #include /* SO_REUSEPORT */ michael@0: #endif michael@0: michael@0: static PRFileDesc *err = NULL; michael@0: static PRBool failed = PR_FALSE; michael@0: michael@0: static void Failed(const char *msg1, const char *msg2) michael@0: { michael@0: if (NULL != msg1) PR_fprintf(err, "%s ", msg1); michael@0: PL_FPrintError(err, msg2); michael@0: failed = PR_TRUE; michael@0: } /* Failed */ michael@0: michael@0: static PRSockOption Incr(PRSockOption *option) michael@0: { michael@0: PRIntn val = ((PRIntn)*option) + 1; michael@0: *option = (PRSockOption)val; michael@0: return (PRSockOption)val; michael@0: } /* Incr */ michael@0: michael@0: int main(int argc, char **argv) michael@0: { michael@0: PRStatus rv; michael@0: PRFileDesc *udp = PR_NewUDPSocket(); michael@0: PRFileDesc *tcp = PR_NewTCPSocket(); michael@0: const char *tag[] = michael@0: { michael@0: "PR_SockOpt_Nonblocking", /* nonblocking io */ michael@0: "PR_SockOpt_Linger", /* linger on close if data present */ michael@0: "PR_SockOpt_Reuseaddr", /* allow local address reuse */ michael@0: "PR_SockOpt_Keepalive", /* keep connections alive */ michael@0: "PR_SockOpt_RecvBufferSize", /* send buffer size */ michael@0: "PR_SockOpt_SendBufferSize", /* receive buffer size */ michael@0: michael@0: "PR_SockOpt_IpTimeToLive", /* time to live */ michael@0: "PR_SockOpt_IpTypeOfService", /* type of service and precedence */ michael@0: michael@0: "PR_SockOpt_AddMember", /* add an IP group membership */ michael@0: "PR_SockOpt_DropMember", /* drop an IP group membership */ michael@0: "PR_SockOpt_McastInterface", /* multicast interface address */ michael@0: "PR_SockOpt_McastTimeToLive", /* multicast timetolive */ michael@0: "PR_SockOpt_McastLoopback", /* multicast loopback */ michael@0: michael@0: "PR_SockOpt_NoDelay", /* don't delay send to coalesce packets */ michael@0: "PR_SockOpt_MaxSegment", /* maximum segment size */ michael@0: "PR_SockOpt_Broadcast", /* Enable broadcast */ michael@0: "PR_SockOpt_Reuseport", /* allow local address & port reuse */ michael@0: "PR_SockOpt_Last" michael@0: }; michael@0: michael@0: err = PR_GetSpecialFD(PR_StandardError); michael@0: PR_STDIO_INIT(); michael@0: michael@0: if (NULL == udp) Failed("PR_NewUDPSocket()", NULL); michael@0: else if (NULL == tcp) Failed("PR_NewTCPSocket()", NULL); michael@0: else michael@0: { michael@0: PRSockOption option; michael@0: PRUint32 segment = 1024; michael@0: PRNetAddr addr; michael@0: michael@0: rv = PR_InitializeNetAddr(PR_IpAddrAny, 0, &addr); michael@0: if (PR_FAILURE == rv) Failed("PR_InitializeNetAddr()", NULL); michael@0: rv = PR_Bind(udp, &addr); michael@0: if (PR_FAILURE == rv) Failed("PR_Bind()", NULL); michael@0: for(option = PR_SockOpt_Linger; option < PR_SockOpt_Last; Incr(&option)) michael@0: { michael@0: PRSocketOptionData data; michael@0: PRFileDesc *fd = tcp; michael@0: data.option = option; michael@0: switch (option) michael@0: { michael@0: case PR_SockOpt_Nonblocking: michael@0: data.value.non_blocking = PR_TRUE; michael@0: break; michael@0: #ifndef SYMBIAN michael@0: case PR_SockOpt_Linger: michael@0: data.value.linger.polarity = PR_TRUE; michael@0: data.value.linger.linger = PR_SecondsToInterval(2); michael@0: break; michael@0: #endif michael@0: case PR_SockOpt_Reuseaddr: michael@0: data.value.reuse_addr = PR_TRUE; michael@0: break; michael@0: case PR_SockOpt_Keepalive: michael@0: data.value.keep_alive = PR_TRUE; michael@0: break; michael@0: case PR_SockOpt_RecvBufferSize: michael@0: data.value.recv_buffer_size = segment; michael@0: break; michael@0: case PR_SockOpt_SendBufferSize: michael@0: data.value.send_buffer_size = segment; michael@0: break; michael@0: #ifndef SYMBIAN michael@0: case PR_SockOpt_IpTimeToLive: michael@0: data.value.ip_ttl = 64; michael@0: break; michael@0: case PR_SockOpt_IpTypeOfService: michael@0: data.value.tos = 0; michael@0: break; michael@0: case PR_SockOpt_McastTimeToLive: michael@0: fd = udp; michael@0: data.value.mcast_ttl = 4; michael@0: break; michael@0: case PR_SockOpt_McastLoopback: michael@0: fd = udp; michael@0: data.value.mcast_loopback = PR_TRUE; michael@0: break; michael@0: #endif michael@0: case PR_SockOpt_NoDelay: michael@0: data.value.no_delay = PR_TRUE; michael@0: break; michael@0: #ifndef WIN32 michael@0: case PR_SockOpt_MaxSegment: michael@0: data.value.max_segment = segment; michael@0: break; michael@0: #endif michael@0: #ifndef SYMBIAN michael@0: case PR_SockOpt_Broadcast: michael@0: fd = udp; michael@0: data.value.broadcast = PR_TRUE; michael@0: break; michael@0: #endif michael@0: #ifdef SO_REUSEPORT michael@0: case PR_SockOpt_Reuseport: michael@0: data.value.reuse_port = PR_TRUE; michael@0: break; michael@0: #endif michael@0: default: continue; michael@0: } michael@0: michael@0: /* michael@0: * TCP_MAXSEG can only be read, not set michael@0: */ michael@0: if (option != PR_SockOpt_MaxSegment) { michael@0: #ifdef WIN32 michael@0: if (option != PR_SockOpt_McastLoopback) michael@0: #endif michael@0: { michael@0: rv = PR_SetSocketOption(fd, &data); michael@0: if (PR_FAILURE == rv) michael@0: Failed("PR_SetSocketOption()", tag[option]); michael@0: } michael@0: } michael@0: michael@0: rv = PR_GetSocketOption(fd, &data); michael@0: if (PR_FAILURE == rv) Failed("PR_GetSocketOption()", tag[option]); michael@0: } michael@0: PR_Close(udp); michael@0: PR_Close(tcp); michael@0: } michael@0: PR_fprintf(err, "%s\n", (failed) ? "FAILED" : "PASSED"); michael@0: return (failed) ? 1 : 0; michael@0: } /* main */ michael@0: michael@0: /* sockopt.c */ michael@0: