03a53256c1f4469ec5a36839672eaa6a5a5f0c28
[nfs-utils.git] / support / nfs / svc_socket.c
1 /* Copyright (C) 2002 Free Software Foundation, Inc.
2    This file is part of the GNU C Library.
3
4    The GNU C Library is free software; you can redistribute it and/or
5    modify it under the terms of the GNU Lesser General Public
6    License as published by the Free Software Foundation; either
7    version 2.1 of the License, or (at your option) any later version.
8
9    The GNU C Library is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12    Lesser General Public License for more details.
13
14    You should have received a copy of the GNU Lesser General Public
15    License along with the GNU C Library; if not, write to the Free
16    Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
17    02111-1307 USA.  */
18
19 #include <stdio.h>
20 #include <string.h>
21 #include <unistd.h>
22 #include <netdb.h>
23 #include <rpc/rpc.h>
24 #include <sys/socket.h>
25 #include <sys/fcntl.h>
26 #include <errno.h>
27
28 #ifdef _LIBC
29 # include <libintl.h>
30 #else
31 # ifndef _
32 #  define _(s)                  (s)
33 # endif
34 # define __socket(d, t, p)      socket ((d), (t), (p))
35 # define __close(f)             close ((f))
36 #endif
37
38 static int
39 svc_socket (u_long number, int type, int protocol, int reuse)
40 {
41   struct sockaddr_in addr;
42   socklen_t len = sizeof (struct sockaddr_in);
43   char rpcdata [1024], servdata [1024];
44   struct rpcent rpcbuf, *rpcp;
45   struct servent servbuf, *servp = NULL;
46   int sock, ret;
47   const char *proto = protocol == IPPROTO_TCP ? "tcp" : "udp";
48
49   if ((sock = __socket (AF_INET, type, protocol)) < 0)
50     {
51       perror (_("svc_socket: socket creation problem"));
52       return sock;
53     }
54
55   if (reuse)
56     {
57       ret = 1;
58       ret = setsockopt (sock, SOL_SOCKET, SO_REUSEADDR, &ret,
59                         sizeof (ret));
60       if (ret < 0)
61         {
62           perror (_("svc_socket: socket reuse problem"));
63           return ret;
64         }
65     }
66
67   memset (&addr, 0, sizeof (addr));
68   addr.sin_family = AF_INET;
69
70   ret = getrpcbynumber_r (number, &rpcbuf, rpcdata, sizeof rpcdata,
71                           &rpcp);
72   if (ret == 0 && rpcp != NULL)
73     {
74       /* First try name.  */
75       ret = getservbyname_r (rpcp->r_name, proto, &servbuf, servdata,
76                              sizeof servdata, &servp);
77       if ((ret != 0 || servp == NULL) && rpcp->r_aliases)
78         {
79           const char **a;
80
81           /* Then we try aliases.  */
82           for (a = (const char **) rpcp->r_aliases; *a != NULL; a++) 
83             {
84               ret = getservbyname_r (*a, proto, &servbuf, servdata,
85                                      sizeof servdata, &servp);
86               if (ret == 0 && servp != NULL)
87                 break;
88             }
89         }
90     }
91
92   if (ret == 0 && servp != NULL)
93     {
94       addr.sin_port = servp->s_port;
95       if (bind (sock, (struct sockaddr *) &addr, len) < 0)
96         {
97           perror (_("svc_socket: bind problem"));
98           (void) __close (sock);
99           sock = -1;
100         }
101     }
102   else
103     {
104           addr.sin_port = 0;
105           if (bind (sock, (struct sockaddr *) &addr, len) < 0)
106             {
107               perror (_("svc_socket: bind problem"));
108               (void) __close (sock);
109               sock = -1;
110             }
111     }
112
113   if (sock >= 0)
114     {
115             /* This socket might be shared among multiple processes
116              * if mountd is run multi-threaded.  So it is safest to
117              * make it non-blocking, else all threads might wake
118              * one will get the data, and the others will block
119              * indefinitely.
120              * In all cases, transaction on this socket are atomic
121              * (accept for TCP, packet-read and packet-write for UDP)
122              * so O_NONBLOCK will not confuse unprepared code causing
123              * it to corrupt messages.
124              * It generally safest to have O_NONBLOCK when doing an accept
125              * as if we get a RST after the SYN and before accept runs,
126              * we can block despite being told there was an acceptable
127              * connection.
128              */
129         int flags;
130         if ((flags = fcntl(sock, F_GETFL)) < 0)
131           {
132               perror (_("svc_socket: can't get socket flags"));
133               (void) __close (sock);
134               sock = -1;
135           }
136         else if (fcntl(sock, F_SETFL, flags|O_NONBLOCK) < 0)
137           {
138               perror (_("svc_socket: can't set socket flags"));
139               (void) __close (sock);
140               sock = -1;
141           }
142     }
143
144   return sock;
145 }
146
147 /*
148  * Create and bind a TCP socket based on program number
149  */
150 int
151 svctcp_socket (u_long number, int reuse)
152 {
153   return svc_socket (number, SOCK_STREAM, IPPROTO_TCP, reuse);
154 }
155
156 /*
157  * Create and bind a UDP socket based on program number
158  */
159 int
160 svcudp_socket (u_long number)
161 {
162   return svc_socket (number, SOCK_DGRAM, IPPROTO_UDP, FALSE);
163 }
164
165 #ifdef TEST
166 static int
167 check (u_long number, u_short port, int protocol, int reuse)
168 {
169   int socket;
170   int result;
171   struct sockaddr_in addr;
172   socklen_t len = sizeof (struct sockaddr_in);
173
174   if (protocol == IPPROTO_TCP)
175     socket = svctcp_socket (number, reuse);
176   else
177     socket = svcudp_socket (number);
178
179   if (socket < 0)
180     return 1;
181
182   result = getsockname (socket, (struct sockaddr *) &addr, &len);
183   if (result == 0)
184     {
185       if (port != 0 && ntohs (addr.sin_port) != port)
186         printf ("Program: %ld, expect port: %d, got: %d\n",
187                 number, port, ntohs (addr.sin_port)); 
188       else
189         printf ("Program: %ld, port: %d\n",
190                 number, ntohs (addr.sin_port)); 
191     }
192
193   close (socket);
194   return result;
195 }
196
197 int
198 main (void)
199 {
200   int result = 0;
201
202   result += check (100001, 0, IPPROTO_TCP, 0);
203   result += check (100001, 0, IPPROTO_UDP, 0);
204   result += check (100003, 2049, IPPROTO_TCP, 1);
205   result += check (100003, 2049, IPPROTO_UDP, 1);
206
207   return result;
208 }
209 #endif