1*7c478bd9Sstevel@tonic-gate /* 2*7c478bd9Sstevel@tonic-gate * CDDL HEADER START 3*7c478bd9Sstevel@tonic-gate * 4*7c478bd9Sstevel@tonic-gate * The contents of this file are subject to the terms of the 5*7c478bd9Sstevel@tonic-gate * Common Development and Distribution License, Version 1.0 only 6*7c478bd9Sstevel@tonic-gate * (the "License"). You may not use this file except in compliance 7*7c478bd9Sstevel@tonic-gate * with the License. 8*7c478bd9Sstevel@tonic-gate * 9*7c478bd9Sstevel@tonic-gate * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 10*7c478bd9Sstevel@tonic-gate * or http://www.opensolaris.org/os/licensing. 11*7c478bd9Sstevel@tonic-gate * See the License for the specific language governing permissions 12*7c478bd9Sstevel@tonic-gate * and limitations under the License. 13*7c478bd9Sstevel@tonic-gate * 14*7c478bd9Sstevel@tonic-gate * When distributing Covered Code, include this CDDL HEADER in each 15*7c478bd9Sstevel@tonic-gate * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 16*7c478bd9Sstevel@tonic-gate * If applicable, add the following below this CDDL HEADER, with the 17*7c478bd9Sstevel@tonic-gate * fields enclosed by brackets "[]" replaced with your own identifying 18*7c478bd9Sstevel@tonic-gate * information: Portions Copyright [yyyy] [name of copyright owner] 19*7c478bd9Sstevel@tonic-gate * 20*7c478bd9Sstevel@tonic-gate * CDDL HEADER END 21*7c478bd9Sstevel@tonic-gate */ 22*7c478bd9Sstevel@tonic-gate /* 23*7c478bd9Sstevel@tonic-gate * Copyright 2004 Sun Microsystems, Inc. All rights reserved. 24*7c478bd9Sstevel@tonic-gate * Use is subject to license terms. 25*7c478bd9Sstevel@tonic-gate */ 26*7c478bd9Sstevel@tonic-gate 27*7c478bd9Sstevel@tonic-gate /* Copyright (c) 1983, 1984, 1985, 1986, 1987, 1988, 1989 AT&T */ 28*7c478bd9Sstevel@tonic-gate /* All Rights Reserved */ 29*7c478bd9Sstevel@tonic-gate 30*7c478bd9Sstevel@tonic-gate /* 31*7c478bd9Sstevel@tonic-gate * Portions of this source code were derived from Berkeley 4.3 BSD 32*7c478bd9Sstevel@tonic-gate * under license from the Regents of the University of California. 33*7c478bd9Sstevel@tonic-gate */ 34*7c478bd9Sstevel@tonic-gate 35*7c478bd9Sstevel@tonic-gate #pragma ident "%Z%%M% %I% %E% SMI" 36*7c478bd9Sstevel@tonic-gate 37*7c478bd9Sstevel@tonic-gate #include <sys/types.h> 38*7c478bd9Sstevel@tonic-gate #include <errno.h> 39*7c478bd9Sstevel@tonic-gate #include <sys/socket.h> 40*7c478bd9Sstevel@tonic-gate #include <netinet/in.h> 41*7c478bd9Sstevel@tonic-gate #include <netinet/tcp.h> 42*7c478bd9Sstevel@tonic-gate #include <netinet/udp.h> 43*7c478bd9Sstevel@tonic-gate #include <string.h> 44*7c478bd9Sstevel@tonic-gate #include <unistd.h> 45*7c478bd9Sstevel@tonic-gate 46*7c478bd9Sstevel@tonic-gate #ifdef SYSV 47*7c478bd9Sstevel@tonic-gate #define bzero(s, len) (void) memset((s), 0, (len)) 48*7c478bd9Sstevel@tonic-gate #endif 49*7c478bd9Sstevel@tonic-gate 50*7c478bd9Sstevel@tonic-gate 51*7c478bd9Sstevel@tonic-gate /* 52*7c478bd9Sstevel@tonic-gate * Bind a socket to a privileged IP port 53*7c478bd9Sstevel@tonic-gate */ 54*7c478bd9Sstevel@tonic-gate int 55*7c478bd9Sstevel@tonic-gate bindresvport(int sd, struct sockaddr_in *sin) 56*7c478bd9Sstevel@tonic-gate { 57*7c478bd9Sstevel@tonic-gate struct sockaddr_in myaddr; 58*7c478bd9Sstevel@tonic-gate struct sockaddr_in *bindaddr; 59*7c478bd9Sstevel@tonic-gate int level, optname; 60*7c478bd9Sstevel@tonic-gate int optval, len; 61*7c478bd9Sstevel@tonic-gate int ret; 62*7c478bd9Sstevel@tonic-gate 63*7c478bd9Sstevel@tonic-gate bindaddr = sin; 64*7c478bd9Sstevel@tonic-gate if (bindaddr == (struct sockaddr_in *)0) { 65*7c478bd9Sstevel@tonic-gate bindaddr = &myaddr; 66*7c478bd9Sstevel@tonic-gate bzero(bindaddr, sizeof (*bindaddr)); 67*7c478bd9Sstevel@tonic-gate bindaddr->sin_family = AF_INET; 68*7c478bd9Sstevel@tonic-gate } else if (bindaddr->sin_family != AF_INET) { 69*7c478bd9Sstevel@tonic-gate errno = EPFNOSUPPORT; 70*7c478bd9Sstevel@tonic-gate return (-1); 71*7c478bd9Sstevel@tonic-gate } 72*7c478bd9Sstevel@tonic-gate 73*7c478bd9Sstevel@tonic-gate len = sizeof (optval); 74*7c478bd9Sstevel@tonic-gate if (getsockopt(sd, SOL_SOCKET, SO_TYPE, &optval, &len) < 0) { 75*7c478bd9Sstevel@tonic-gate return (-1); 76*7c478bd9Sstevel@tonic-gate } 77*7c478bd9Sstevel@tonic-gate /* 78*7c478bd9Sstevel@tonic-gate * Use *_ANONPRIVBIND to ask the kernel to pick a port in the 79*7c478bd9Sstevel@tonic-gate * priviledged range for us. 80*7c478bd9Sstevel@tonic-gate */ 81*7c478bd9Sstevel@tonic-gate if (optval == SOCK_STREAM) { 82*7c478bd9Sstevel@tonic-gate level = IPPROTO_TCP; 83*7c478bd9Sstevel@tonic-gate optname = TCP_ANONPRIVBIND; 84*7c478bd9Sstevel@tonic-gate } else if (optval == SOCK_DGRAM) { 85*7c478bd9Sstevel@tonic-gate level = IPPROTO_UDP; 86*7c478bd9Sstevel@tonic-gate optname = UDP_ANONPRIVBIND; 87*7c478bd9Sstevel@tonic-gate } else { 88*7c478bd9Sstevel@tonic-gate errno = EPROTONOSUPPORT; 89*7c478bd9Sstevel@tonic-gate return (-1); 90*7c478bd9Sstevel@tonic-gate } 91*7c478bd9Sstevel@tonic-gate 92*7c478bd9Sstevel@tonic-gate optval = 1; 93*7c478bd9Sstevel@tonic-gate if (setsockopt(sd, level, optname, &optval, sizeof (optval)) < 0) { 94*7c478bd9Sstevel@tonic-gate return (-1); 95*7c478bd9Sstevel@tonic-gate } 96*7c478bd9Sstevel@tonic-gate 97*7c478bd9Sstevel@tonic-gate bindaddr->sin_port = 0; 98*7c478bd9Sstevel@tonic-gate ret = bind(sd, (struct sockaddr *)bindaddr, 99*7c478bd9Sstevel@tonic-gate sizeof (struct sockaddr_in)); 100*7c478bd9Sstevel@tonic-gate 101*7c478bd9Sstevel@tonic-gate /* 102*7c478bd9Sstevel@tonic-gate * Always turn off the option when we are done. Note that by doing 103*7c478bd9Sstevel@tonic-gate * this, if the caller has set this option before calling 104*7c478bd9Sstevel@tonic-gate * bindresvport(), it will be unset. But this should never happen... 105*7c478bd9Sstevel@tonic-gate */ 106*7c478bd9Sstevel@tonic-gate optval = 0; 107*7c478bd9Sstevel@tonic-gate (void) setsockopt(sd, level, optname, &optval, sizeof (optval)); 108*7c478bd9Sstevel@tonic-gate 109*7c478bd9Sstevel@tonic-gate if (ret >= 0 && sin != NULL) { 110*7c478bd9Sstevel@tonic-gate /* 111*7c478bd9Sstevel@tonic-gate * Historical note: 112*7c478bd9Sstevel@tonic-gate * 113*7c478bd9Sstevel@tonic-gate * Past versions of this bindresvport() code have 114*7c478bd9Sstevel@tonic-gate * returned with the reserved port number bound 115*7c478bd9Sstevel@tonic-gate * filled in its "sin" parameter (if passed in), perhaps 116*7c478bd9Sstevel@tonic-gate * "accidently" because of the structure of historical code. 117*7c478bd9Sstevel@tonic-gate * 118*7c478bd9Sstevel@tonic-gate * This is not documented but the behavior is 119*7c478bd9Sstevel@tonic-gate * explicitly retained here for compatibility to minimize 120*7c478bd9Sstevel@tonic-gate * risk to applications, even though it is not clear if this 121*7c478bd9Sstevel@tonic-gate * was a design intent. 122*7c478bd9Sstevel@tonic-gate */ 123*7c478bd9Sstevel@tonic-gate len = sizeof (struct sockaddr_in); 124*7c478bd9Sstevel@tonic-gate (void) getsockname(sd, (struct sockaddr *)bindaddr, &len); 125*7c478bd9Sstevel@tonic-gate } 126*7c478bd9Sstevel@tonic-gate return (ret); 127*7c478bd9Sstevel@tonic-gate } 128