On Mon, Feb 06, 2023 at 03:11:10PM +0100, Alejandro Colomar wrote: > Hi Rich, > > On 2/6/23 14:38, Rich Felker wrote: > >There is absolutely not any need to declare the union for application > >code calling the socket APIs. You declare whatever type you will be > >using. For binding or connecting a unix socket, sockaddr_un. For IPv6, > >sockaddr_in6. Etc. Then you cast the pointer to struct sockaddr * and > >pass it to the function. > > Except that you may be using generic code that may use either of > AF_UNIX, AF_INET, and AF_INET6. A web server may very well use all > the three. If you have generic code, the generic code is not what's creating the object. It got the object from the calling code or from some callback that allocated it, in which case it doesn't have to care about any of this. > >But normally you don't use declared-type objects for this anyway. You > >use the struct sockaddr * obtained from getaddrinfo as an abstract > >pointer and never dereference it at all. > > That's right. Most of the time, we should be using getaddrinfo(3), > which already provides the storage. I don't know for sure if there > are any cases that can't be rewritten to work that way. > > However, there are some APIs that require you to allocate an object. > For example recvfrom(2). How would you recommend using recvfrom(2), > or is it some API to avoid? Example of usage: > <Mhttps://man7.org/tlpi/code/online/dist/sockets/id_echo_sv.c.html>. If using allocated storage, there's nothing special you have to do. But if you want to avoid malloc and use a declared object, you have a few options: If it's your socket and you know what address family it's associated with, you just pass an object of that type. recvfrom will always produce output of the same AF. If you're in a situation like having been invoked from inetd, you can use getsockname into a sockaddr_storage object or union or whatever, then read out the family field (with memcpy if needed). Then you know the AF to use. Or, you can just recvfrom into a sockaddr_storage object, determine the family at that time, then memcpy the the appropriate object type. Alternatively, for the common case where you want a printable name for the address, you just pass it to getnameinfo as-is and let getnameinfo deal with how to read it. Rich