1 /* Copyright (C) 1994-2022 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, see
16    <https://www.gnu.org/licenses/>.  */
17 
18 #include <errno.h>
19 #include <sys/socket.h>
20 #include <sys/un.h>
21 #include <hurd.h>
22 #include <hurd/fd.h>
23 #include <hurd/ifsock.h>
24 #include <hurd/socket.h>
25 #include <sysdep-cancel.h>
26 #include "hurd/hurdsocket.h"
27 
28 /* Send N bytes of BUF on socket FD to peer at address ADDR (which is
29    ADDR_LEN bytes long).  Returns the number sent, or -1 for errors.  */
30 ssize_t
__sendto(int fd,const void * buf,size_t n,int flags,const struct sockaddr_un * addr,socklen_t addr_len)31 __sendto (int fd,
32 	  const void *buf,
33 	  size_t n,
34 	  int flags,
35 	  const struct sockaddr_un *addr,
36 	  socklen_t addr_len)
37 {
38   addr_port_t aport = MACH_PORT_NULL;
39   error_t err;
40   size_t wrote;
41 
42   /* Get an address port for the desired destination address.  */
43   error_t create_address_port (io_t port,
44 			       const struct sockaddr_un *addr,
45 			       socklen_t addr_len,
46 			       addr_port_t *aport)
47     {
48       error_t err_port;
49 
50       if (addr->sun_family == AF_LOCAL)
51 	{
52 	  char *name = _hurd_sun_path_dupa (addr, addr_len);
53 	  /* For the local domain, we must look up the name as a file and talk
54 	     to it with the ifsock protocol.  */
55 	  file_t file = __file_name_lookup (name, 0, 0);
56 	  if (file == MACH_PORT_NULL)
57 	    return errno;
58 	  err_port = __ifsock_getsockaddr (file, aport);
59 	  __mach_port_deallocate (__mach_task_self (), file);
60 	  if (err_port == MIG_BAD_ID || err_port == EOPNOTSUPP)
61 	    /* The file did not grok the ifsock protocol.  */
62 	    err_port = ENOTSOCK;
63 	}
64       else
65 	{
66 	  err_port = __socket_create_address (port,
67 					      addr->sun_family,
68 					      (char *) addr,
69 					      addr_len,
70 					      aport);
71 	}
72 
73       return err_port;
74     }
75 
76   err = HURD_DPORT_USE_CANCEL (fd,
77 			({
78 			  if (addr != NULL)
79 			    err = create_address_port (port, addr, addr_len,
80 						       &aport);
81 			  else
82 			    err = 0;
83 			  if (! err)
84 			    {
85 			      /* Send the data.  */
86 			      int cancel_oldtype = LIBC_CANCEL_ASYNC();
87 			      err = __socket_send (port, aport,
88 						   flags, buf, n,
89 						   NULL,
90 						   MACH_MSG_TYPE_COPY_SEND, 0,
91 						   NULL, 0, &wrote);
92 			      LIBC_CANCEL_RESET (cancel_oldtype);
93 			    }
94 			  err;
95 			}));
96 
97   if (aport != MACH_PORT_NULL)
98     __mach_port_deallocate (__mach_task_self (), aport);
99 
100   return err ? __hurd_sockfail (fd, flags, err) : wrote;
101 }
102 
103 weak_alias (__sendto, sendto)
104