blob: c89f2140344573552109e4bdcf6413cfc51ed9c9 [file] [log] [blame]
Damien Millerf4684422003-06-02 18:59:08 +10001/* This file has be substantially modified from the original OpenBSD source */
2
Damien Millera97529f2007-10-26 16:16:09 +10003/* $OpenBSD: bindresvport.c,v 1.17 2005/12/21 01:40:22 millert Exp $ */
Damien Miller2a5c1ce2001-01-25 10:32:00 +11004
Damien Miller34132e52000-01-14 15:45:46 +11005/*
Damien Millerf4684422003-06-02 18:59:08 +10006 * Copyright 1996, Jason Downs. All rights reserved.
7 * Copyright 1998, Theo de Raadt. All rights reserved.
8 * Copyright 2000, Damien Miller. All rights reserved.
9 *
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
12 * are met:
13 * 1. Redistributions of source code must retain the above copyright
14 * notice, this list of conditions and the following disclaimer.
15 * 2. Redistributions in binary form must reproduce the above copyright
16 * notice, this list of conditions and the following disclaimer in the
17 * documentation and/or other materials provided with the distribution.
18 *
19 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
20 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
21 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
22 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
23 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
24 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
28 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Damien Miller34132e52000-01-14 15:45:46 +110029 */
30
Darren Tucker6f15c072005-11-10 17:52:08 +110031/* OPENBSD ORIGINAL: lib/libc/rpc/bindresvport.c */
32
Ben Lindstromdd21fe92002-06-27 18:23:20 +000033#include "includes.h"
Damien Miller34132e52000-01-14 15:45:46 +110034
Damien Miller2a5c1ce2001-01-25 10:32:00 +110035#ifndef HAVE_BINDRESVPORT_SA
Damien Miller607aede2006-09-01 15:48:19 +100036#include <sys/types.h>
37#include <sys/socket.h>
Damien Miller34132e52000-01-14 15:45:46 +110038
Damien Miller607aede2006-09-01 15:48:19 +100039#include <netinet/in.h>
Darren Tucker46aa3e02006-09-02 15:32:40 +100040#include <arpa/inet.h>
Damien Miller34132e52000-01-14 15:45:46 +110041
Darren Tucker2c1a02a2006-07-12 22:40:50 +100042#include <errno.h>
Damien Millerb8fe89c2006-07-24 14:51:00 +100043#include <string.h>
Darren Tucker2c1a02a2006-07-12 22:40:50 +100044
Damien Miller34132e52000-01-14 15:45:46 +110045#define STARTPORT 600
46#define ENDPORT (IPPORT_RESERVED - 1)
47#define NPORTS (ENDPORT - STARTPORT + 1)
48
Damien Miller34132e52000-01-14 15:45:46 +110049/*
50 * Bind a socket to a privileged IP port
51 */
52int
Darren Tuckerfe80d7a2005-11-10 17:54:46 +110053bindresvport_sa(int sd, struct sockaddr *sa)
Damien Miller34132e52000-01-14 15:45:46 +110054{
Damien Miller2a5c1ce2001-01-25 10:32:00 +110055 int error, af;
Damien Miller34132e52000-01-14 15:45:46 +110056 struct sockaddr_storage myaddr;
Damien Millerce02e5e2008-07-14 12:02:24 +100057 struct sockaddr_in *in;
58 struct sockaddr_in6 *in6;
Damien Miller34132e52000-01-14 15:45:46 +110059 u_int16_t *portp;
Damien Miller8dbbe6e2000-01-22 18:17:42 +110060 u_int16_t port;
Ben Lindstrom0d5af602001-01-09 00:50:29 +000061 socklen_t salen;
Damien Miller34132e52000-01-14 15:45:46 +110062 int i;
63
64 if (sa == NULL) {
65 memset(&myaddr, 0, sizeof(myaddr));
66 sa = (struct sockaddr *)&myaddr;
Damien Miller2a5c1ce2001-01-25 10:32:00 +110067
68 if (getsockname(sd, sa, &salen) == -1)
69 return -1; /* errno is correctly set */
70
71 af = sa->sa_family;
72 memset(&myaddr, 0, salen);
73 } else
74 af = sa->sa_family;
Damien Miller34132e52000-01-14 15:45:46 +110075
76 if (af == AF_INET) {
Damien Millerce02e5e2008-07-14 12:02:24 +100077 in = (struct sockaddr_in *)sa;
Damien Miller34132e52000-01-14 15:45:46 +110078 salen = sizeof(struct sockaddr_in);
Damien Millerce02e5e2008-07-14 12:02:24 +100079 portp = &in->sin_port;
Damien Miller34132e52000-01-14 15:45:46 +110080 } else if (af == AF_INET6) {
Damien Millerce02e5e2008-07-14 12:02:24 +100081 in6 = (struct sockaddr_in6 *)sa;
Damien Miller34132e52000-01-14 15:45:46 +110082 salen = sizeof(struct sockaddr_in6);
Damien Millerce02e5e2008-07-14 12:02:24 +100083 portp = &in6->sin6_port;
Damien Miller34132e52000-01-14 15:45:46 +110084 } else {
85 errno = EPFNOSUPPORT;
86 return (-1);
87 }
88 sa->sa_family = af;
89
Damien Miller8dbbe6e2000-01-22 18:17:42 +110090 port = ntohs(*portp);
91 if (port == 0)
Damien Millerd89745b2010-12-03 10:50:26 +110092 port = arc4random_uniform(NPORTS) + STARTPORT;
Damien Miller34132e52000-01-14 15:45:46 +110093
Damien Millera64b57a2001-01-17 10:44:13 +110094 /* Avoid warning */
95 error = -1;
96
Damien Miller34132e52000-01-14 15:45:46 +110097 for(i = 0; i < NPORTS; i++) {
Damien Miller8dbbe6e2000-01-22 18:17:42 +110098 *portp = htons(port);
99
Damien Miller34132e52000-01-14 15:45:46 +1100100 error = bind(sd, sa, salen);
Damien Miller2a5c1ce2001-01-25 10:32:00 +1100101
Damien Millerb9b94a72000-01-17 09:52:46 +1100102 /* Terminate on success */
103 if (error == 0)
Damien Miller34132e52000-01-14 15:45:46 +1100104 break;
105
Damien Millerb9b94a72000-01-17 09:52:46 +1100106 /* Terminate on errors, except "address already in use" */
Damien Miller19fe9c72000-01-17 15:23:01 +1100107 if ((error < 0) && !((errno == EADDRINUSE) || (errno == EINVAL)))
Damien Millerb9b94a72000-01-17 09:52:46 +1100108 break;
109
Damien Miller8dbbe6e2000-01-22 18:17:42 +1100110 port++;
111 if (port > ENDPORT)
112 port = STARTPORT;
Damien Miller34132e52000-01-14 15:45:46 +1100113 }
114
115 return (error);
116}
117
Damien Miller2a5c1ce2001-01-25 10:32:00 +1100118#endif /* HAVE_BINDRESVPORT_SA */