1bf6a2064SMaxim Konovalov /*-
2bf6a2064SMaxim Konovalov * Copyright (C) 2005 The FreeBSD Project. All rights reserved.
3bf6a2064SMaxim Konovalov *
4bf6a2064SMaxim Konovalov * Redistribution and use in source and binary forms, with or without
5bf6a2064SMaxim Konovalov * modification, are permitted provided that the following conditions
6bf6a2064SMaxim Konovalov * are met:
7bf6a2064SMaxim Konovalov * 1. Redistributions of source code must retain the above copyright
8bf6a2064SMaxim Konovalov * notice, this list of conditions and the following disclaimer.
9bf6a2064SMaxim Konovalov * 2. Redistributions in binary form must reproduce the above copyright
10bf6a2064SMaxim Konovalov * notice, this list of conditions and the following disclaimer in the
11bf6a2064SMaxim Konovalov * documentation and/or other materials provided with the distribution.
12bf6a2064SMaxim Konovalov *
13bf6a2064SMaxim Konovalov * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
14bf6a2064SMaxim Konovalov * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15bf6a2064SMaxim Konovalov * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16bf6a2064SMaxim Konovalov * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
17bf6a2064SMaxim Konovalov * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18bf6a2064SMaxim Konovalov * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19bf6a2064SMaxim Konovalov * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20bf6a2064SMaxim Konovalov * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21bf6a2064SMaxim Konovalov * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22bf6a2064SMaxim Konovalov * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
23bf6a2064SMaxim Konovalov * SUCH DAMAGE.
24bf6a2064SMaxim Konovalov */
25bf6a2064SMaxim Konovalov
26bf6a2064SMaxim Konovalov #include <sys/types.h>
27bf6a2064SMaxim Konovalov #include <sys/socket.h>
28bf6a2064SMaxim Konovalov
29bf6a2064SMaxim Konovalov #include <netinet/in.h>
30bf6a2064SMaxim Konovalov #include <arpa/inet.h>
31bf6a2064SMaxim Konovalov
32bf6a2064SMaxim Konovalov #include <err.h>
33bf6a2064SMaxim Konovalov #include <errno.h>
34bf6a2064SMaxim Konovalov #include <stdio.h>
35bf6a2064SMaxim Konovalov #include <string.h>
36bf6a2064SMaxim Konovalov #include <unistd.h>
37bf6a2064SMaxim Konovalov
38bf6a2064SMaxim Konovalov int
main(void)39bf6a2064SMaxim Konovalov main(void)
40bf6a2064SMaxim Konovalov {
41bf6a2064SMaxim Konovalov struct sockaddr_in sock;
42bf6a2064SMaxim Konovalov socklen_t len;
43bf6a2064SMaxim Konovalov int listen_sock, connect_sock;
44bf6a2064SMaxim Konovalov u_short port;
45bf6a2064SMaxim Konovalov
46*bcd1483dSEnji Cooper listen_sock = -1;
47*bcd1483dSEnji Cooper
48bf6a2064SMaxim Konovalov /* Shutdown(2) on an invalid file descriptor has to return EBADF. */
49bf6a2064SMaxim Konovalov if ((shutdown(listen_sock, SHUT_RDWR) != -1) && (errno != EBADF))
50bf6a2064SMaxim Konovalov errx(-1, "shutdown() for invalid file descriptor does not "
51bf6a2064SMaxim Konovalov "return EBADF");
52bf6a2064SMaxim Konovalov
53bf6a2064SMaxim Konovalov listen_sock = socket(PF_INET, SOCK_STREAM, 0);
54bf6a2064SMaxim Konovalov if (listen_sock == -1)
55bf6a2064SMaxim Konovalov errx(-1,
56bf6a2064SMaxim Konovalov "socket(PF_INET, SOCK_STREAM, 0) for listen socket: %s",
57bf6a2064SMaxim Konovalov strerror(errno));
58bf6a2064SMaxim Konovalov
59bf6a2064SMaxim Konovalov bzero(&sock, sizeof(sock));
60bf6a2064SMaxim Konovalov sock.sin_len = sizeof(sock);
61bf6a2064SMaxim Konovalov sock.sin_family = AF_INET;
62bf6a2064SMaxim Konovalov sock.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
63bf6a2064SMaxim Konovalov sock.sin_port = 0;
64bf6a2064SMaxim Konovalov
65bf6a2064SMaxim Konovalov if (bind(listen_sock, (struct sockaddr *)&sock, sizeof(sock)) < 0)
66bf6a2064SMaxim Konovalov errx(-1, "bind(%s, %d) for listen socket: %s",
67bf6a2064SMaxim Konovalov inet_ntoa(sock.sin_addr), sock.sin_port, strerror(errno));
68bf6a2064SMaxim Konovalov
69bf6a2064SMaxim Konovalov len = sizeof(sock);
70bf6a2064SMaxim Konovalov if (getsockname(listen_sock, (struct sockaddr *)&sock, &len) < 0)
71bf6a2064SMaxim Konovalov errx(-1, "getsockname() for listen socket: %s",
72bf6a2064SMaxim Konovalov strerror(errno));
73bf6a2064SMaxim Konovalov port = sock.sin_port;
74bf6a2064SMaxim Konovalov
75bf6a2064SMaxim Konovalov if (listen(listen_sock, -1) < 0)
76bf6a2064SMaxim Konovalov errx(-1, "listen() for listen socket: %s", strerror(errno));
77bf6a2064SMaxim Konovalov
78bf6a2064SMaxim Konovalov connect_sock = socket(PF_INET, SOCK_STREAM, 0);
79bf6a2064SMaxim Konovalov if (connect_sock == -1)
80bf6a2064SMaxim Konovalov errx(-1, "socket(PF_INET, SOCK_STREAM, 0) for connect "
81bf6a2064SMaxim Konovalov "socket: %s", strerror(errno));
82bf6a2064SMaxim Konovalov
83bf6a2064SMaxim Konovalov bzero(&sock, sizeof(sock));
84bf6a2064SMaxim Konovalov sock.sin_len = sizeof(sock);
85bf6a2064SMaxim Konovalov sock.sin_family = AF_INET;
86bf6a2064SMaxim Konovalov sock.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
87bf6a2064SMaxim Konovalov sock.sin_port = port;
88bf6a2064SMaxim Konovalov
89bf6a2064SMaxim Konovalov if (connect(connect_sock, (struct sockaddr *)&sock, sizeof(sock)) < 0)
90bf6a2064SMaxim Konovalov errx(-1, "connect() for connect socket: %s", strerror(errno));
91bf6a2064SMaxim Konovalov /* Try to pass an invalid flags. */
92bf6a2064SMaxim Konovalov if ((shutdown(connect_sock, SHUT_RD - 1) != -1) && (errno != EINVAL))
93bf6a2064SMaxim Konovalov errx(-1, "shutdown(SHUT_RD - 1) does not return EINVAL");
94bf6a2064SMaxim Konovalov if ((shutdown(connect_sock, SHUT_RDWR + 1) != -1) && (errno != EINVAL))
95bf6a2064SMaxim Konovalov errx(-1, "shutdown(SHUT_RDWR + 1) does not return EINVAL");
96bf6a2064SMaxim Konovalov
97bf6a2064SMaxim Konovalov if (shutdown(connect_sock, SHUT_RD) < 0)
98bf6a2064SMaxim Konovalov errx(-1, "shutdown(SHUT_RD) for connect socket: %s",
99bf6a2064SMaxim Konovalov strerror(errno));
100bf6a2064SMaxim Konovalov if (shutdown(connect_sock, SHUT_WR) < 0)
101bf6a2064SMaxim Konovalov errx(-1, "shutdown(SHUT_WR) for connect socket: %s",
102bf6a2064SMaxim Konovalov strerror(errno));
103bf6a2064SMaxim Konovalov
104bf6a2064SMaxim Konovalov close(connect_sock);
105bf6a2064SMaxim Konovalov close(listen_sock);
106bf6a2064SMaxim Konovalov
107bf6a2064SMaxim Konovalov return (0);
108bf6a2064SMaxim Konovalov }
109