1 /* Tests for socket address type definitions.
2    Copyright (C) 2016-2022 Free Software Foundation, Inc.
3    This file is part of the GNU C Library.
4 
5    The GNU C Library is free software; you can redistribute it and/or
6    modify it under the terms of the GNU Lesser General Public License as
7    published by the Free Software Foundation; either version 2.1 of the
8    License, or (at your option) any later version.
9 
10    The GNU C Library is distributed in the hope that it will be useful,
11    but WITHOUT ANY WARRANTY; without even the implied warranty of
12    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13    Lesser General Public License for more details.
14 
15    You should have received a copy of the GNU Lesser General Public
16    License along with the GNU C Library; see the file COPYING.LIB.  If
17    not, see <https://www.gnu.org/licenses/>.  */
18 
19 #include <netinet/in.h>
20 #include <stdbool.h>
21 #include <stddef.h>
22 #include <stdio.h>
23 #include <stdlib.h>
24 #include <string.h>
25 #include <sys/socket.h>
26 #include <sys/un.h>
27 
28 /* This is a copy of the previous definition of struct
29    sockaddr_storage.  It is not equal to the old value of _SS_SIZE
30    (128) on all architectures.  We must stay compatible with the old
31    definition.  */
32 
33 #define OLD_REFERENCE_SIZE 128
34 #define OLD_PADSIZE (OLD_REFERENCE_SIZE - (2 * sizeof (__ss_aligntype)))
35 struct sockaddr_storage_old
36   {
37     __SOCKADDR_COMMON (old_);
38     __ss_aligntype old_align;
39     char old_padding[OLD_PADSIZE];
40   };
41 
42 static bool errors;
43 
44 static void
check(bool ok,const char * message)45 check (bool ok, const char *message)
46 {
47   if (!ok)
48     {
49       printf ("error: failed check: %s\n", message);
50       errors = true;
51     }
52 }
53 
54 static int
do_test(void)55 do_test (void)
56 {
57   check (OLD_REFERENCE_SIZE >= _SS_SIZE,
58          "old target size is not smaller than actual size");
59   check (sizeof (struct sockaddr_storage_old)
60          == sizeof (struct sockaddr_storage),
61          "old and new sizes match");
62   check (__alignof (struct sockaddr_storage_old)
63          == __alignof (struct sockaddr_storage),
64          "old and new alignment matches");
65   check (offsetof (struct sockaddr_storage_old, old_family)
66          == offsetof (struct sockaddr_storage, ss_family),
67          "old and new family offsets match");
68   check (sizeof (struct sockaddr_storage) == _SS_SIZE,
69          "struct sockaddr_storage size");
70 
71   /* Check for lack of holes in the struct definition.   */
72   check (offsetof (struct sockaddr_storage, __ss_padding)
73          == __SOCKADDR_COMMON_SIZE,
74          "implicit padding before explicit padding");
75   check (offsetof (struct sockaddr_storage, __ss_align)
76          == __SOCKADDR_COMMON_SIZE
77            + sizeof (((struct sockaddr_storage) {}).__ss_padding),
78          "implicit padding before explicit padding");
79 
80   /* Check for POSIX compatibility requirements between struct
81      sockaddr_storage and struct sockaddr_un.  */
82   check (sizeof (struct sockaddr_storage) >= sizeof (struct sockaddr_un),
83          "sockaddr_storage is at least as large as sockaddr_un");
84   check (__alignof (struct sockaddr_storage)
85          >= __alignof (struct sockaddr_un),
86          "sockaddr_storage is at least as aligned as sockaddr_un");
87   check (offsetof (struct sockaddr_storage, ss_family)
88          == offsetof (struct sockaddr_un, sun_family),
89          "family offsets match");
90 
91   /* Check that the compiler preserves bit patterns in aggregate
92      copies.  Based on <https://gcc.gnu.org/PR71120>.  */
93   check (sizeof (struct sockaddr_storage) >= sizeof (struct sockaddr_in),
94          "sockaddr_storage is at least as large as sockaddr_in");
95   {
96     struct sockaddr_storage addr;
97     memset (&addr, 0, sizeof (addr));
98     {
99       struct sockaddr_in *sinp = (struct sockaddr_in *)&addr;
100       sinp->sin_family = AF_INET;
101       sinp->sin_addr.s_addr = htonl (INADDR_LOOPBACK);
102       sinp->sin_port = htons (80);
103     }
104     struct sockaddr_storage copy;
105     copy = addr;
106 
107     struct sockaddr_storage *p = malloc (sizeof (*p));
108     if (p == NULL)
109       {
110         printf ("error: malloc: %m\n");
111         return 1;
112       }
113     *p = copy;
114     const struct sockaddr_in *sinp = (const struct sockaddr_in *)p;
115     check (sinp->sin_family == AF_INET, "sin_family");
116     check (sinp->sin_addr.s_addr == htonl (INADDR_LOOPBACK), "sin_addr");
117     check (sinp->sin_port == htons (80), "sin_port");
118     free (p);
119   }
120 
121   return errors;
122 }
123 
124 #define TEST_FUNCTION do_test ()
125 #include "../test-skeleton.c"
126