1 /*- 2 * Copyright (c) 1982, 1986, 1990, 1993 3 * The Regents of the University of California. All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * 2. Redistributions in binary form must reproduce the above copyright 11 * notice, this list of conditions and the following disclaimer in the 12 * documentation and/or other materials provided with the distribution. 13 * 4. Neither the name of the University nor the names of its contributors 14 * may be used to endorse or promote products derived from this software 15 * without specific prior written permission. 16 * 17 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 18 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 20 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 21 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 22 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 23 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 24 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 25 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 26 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 27 * SUCH DAMAGE. 28 * 29 * @(#)sys_socket.c 8.1 (Berkeley) 6/10/93 30 */ 31 32 #include <sys/cdefs.h> 33 __FBSDID("$FreeBSD$"); 34 35 #include <sys/param.h> 36 #include <sys/systm.h> 37 #include <sys/domain.h> 38 #include <sys/file.h> 39 #include <sys/filedesc.h> 40 #include <sys/malloc.h> 41 #include <sys/proc.h> 42 #include <sys/protosw.h> 43 #include <sys/sigio.h> 44 #include <sys/signal.h> 45 #include <sys/signalvar.h> 46 #include <sys/socket.h> 47 #include <sys/socketvar.h> 48 #include <sys/filio.h> /* XXX */ 49 #include <sys/sockio.h> 50 #include <sys/stat.h> 51 #include <sys/uio.h> 52 #include <sys/ucred.h> 53 #include <sys/un.h> 54 #include <sys/unpcb.h> 55 #include <sys/user.h> 56 57 #include <net/if.h> 58 #include <net/if_var.h> 59 #include <net/route.h> 60 #include <net/vnet.h> 61 62 #include <netinet/in.h> 63 #include <netinet/in_pcb.h> 64 65 #include <security/mac/mac_framework.h> 66 67 static fo_rdwr_t soo_read; 68 static fo_rdwr_t soo_write; 69 static fo_ioctl_t soo_ioctl; 70 static fo_poll_t soo_poll; 71 extern fo_kqfilter_t soo_kqfilter; 72 static fo_stat_t soo_stat; 73 static fo_close_t soo_close; 74 static fo_fill_kinfo_t soo_fill_kinfo; 75 76 struct fileops socketops = { 77 .fo_read = soo_read, 78 .fo_write = soo_write, 79 .fo_truncate = invfo_truncate, 80 .fo_ioctl = soo_ioctl, 81 .fo_poll = soo_poll, 82 .fo_kqfilter = soo_kqfilter, 83 .fo_stat = soo_stat, 84 .fo_close = soo_close, 85 .fo_chmod = invfo_chmod, 86 .fo_chown = invfo_chown, 87 .fo_sendfile = invfo_sendfile, 88 .fo_fill_kinfo = soo_fill_kinfo, 89 .fo_flags = DFLAG_PASSABLE 90 }; 91 92 static int 93 soo_read(struct file *fp, struct uio *uio, struct ucred *active_cred, 94 int flags, struct thread *td) 95 { 96 struct socket *so = fp->f_data; 97 int error; 98 99 #ifdef MAC 100 error = mac_socket_check_receive(active_cred, so); 101 if (error) 102 return (error); 103 #endif 104 error = soreceive(so, 0, uio, 0, 0, 0); 105 return (error); 106 } 107 108 static int 109 soo_write(struct file *fp, struct uio *uio, struct ucred *active_cred, 110 int flags, struct thread *td) 111 { 112 struct socket *so = fp->f_data; 113 int error; 114 115 #ifdef MAC 116 error = mac_socket_check_send(active_cred, so); 117 if (error) 118 return (error); 119 #endif 120 error = sosend(so, 0, uio, 0, 0, 0, uio->uio_td); 121 if (error == EPIPE && (so->so_options & SO_NOSIGPIPE) == 0) { 122 PROC_LOCK(uio->uio_td->td_proc); 123 tdsignal(uio->uio_td, SIGPIPE); 124 PROC_UNLOCK(uio->uio_td->td_proc); 125 } 126 return (error); 127 } 128 129 static int 130 soo_ioctl(struct file *fp, u_long cmd, void *data, struct ucred *active_cred, 131 struct thread *td) 132 { 133 struct socket *so = fp->f_data; 134 int error = 0; 135 136 switch (cmd) { 137 case FIONBIO: 138 SOCK_LOCK(so); 139 if (*(int *)data) 140 so->so_state |= SS_NBIO; 141 else 142 so->so_state &= ~SS_NBIO; 143 SOCK_UNLOCK(so); 144 break; 145 146 case FIOASYNC: 147 /* 148 * XXXRW: This code separately acquires SOCK_LOCK(so) and 149 * SOCKBUF_LOCK(&so->so_rcv) even though they are the same 150 * mutex to avoid introducing the assumption that they are 151 * the same. 152 */ 153 if (*(int *)data) { 154 SOCK_LOCK(so); 155 so->so_state |= SS_ASYNC; 156 SOCK_UNLOCK(so); 157 SOCKBUF_LOCK(&so->so_rcv); 158 so->so_rcv.sb_flags |= SB_ASYNC; 159 SOCKBUF_UNLOCK(&so->so_rcv); 160 SOCKBUF_LOCK(&so->so_snd); 161 so->so_snd.sb_flags |= SB_ASYNC; 162 SOCKBUF_UNLOCK(&so->so_snd); 163 } else { 164 SOCK_LOCK(so); 165 so->so_state &= ~SS_ASYNC; 166 SOCK_UNLOCK(so); 167 SOCKBUF_LOCK(&so->so_rcv); 168 so->so_rcv.sb_flags &= ~SB_ASYNC; 169 SOCKBUF_UNLOCK(&so->so_rcv); 170 SOCKBUF_LOCK(&so->so_snd); 171 so->so_snd.sb_flags &= ~SB_ASYNC; 172 SOCKBUF_UNLOCK(&so->so_snd); 173 } 174 break; 175 176 case FIONREAD: 177 /* Unlocked read. */ 178 *(int *)data = so->so_rcv.sb_cc; 179 break; 180 181 case FIONWRITE: 182 /* Unlocked read. */ 183 *(int *)data = so->so_snd.sb_cc; 184 break; 185 186 case FIONSPACE: 187 if ((so->so_snd.sb_hiwat < so->so_snd.sb_cc) || 188 (so->so_snd.sb_mbmax < so->so_snd.sb_mbcnt)) 189 *(int *)data = 0; 190 else 191 *(int *)data = sbspace(&so->so_snd); 192 break; 193 194 case FIOSETOWN: 195 error = fsetown(*(int *)data, &so->so_sigio); 196 break; 197 198 case FIOGETOWN: 199 *(int *)data = fgetown(&so->so_sigio); 200 break; 201 202 case SIOCSPGRP: 203 error = fsetown(-(*(int *)data), &so->so_sigio); 204 break; 205 206 case SIOCGPGRP: 207 *(int *)data = -fgetown(&so->so_sigio); 208 break; 209 210 case SIOCATMARK: 211 /* Unlocked read. */ 212 *(int *)data = (so->so_rcv.sb_state & SBS_RCVATMARK) != 0; 213 break; 214 default: 215 /* 216 * Interface/routing/protocol specific ioctls: interface and 217 * routing ioctls should have a different entry since a 218 * socket is unnecessary. 219 */ 220 if (IOCGROUP(cmd) == 'i') 221 error = ifioctl(so, cmd, data, td); 222 else if (IOCGROUP(cmd) == 'r') { 223 CURVNET_SET(so->so_vnet); 224 error = rtioctl_fib(cmd, data, so->so_fibnum); 225 CURVNET_RESTORE(); 226 } else { 227 CURVNET_SET(so->so_vnet); 228 error = ((*so->so_proto->pr_usrreqs->pru_control) 229 (so, cmd, data, 0, td)); 230 CURVNET_RESTORE(); 231 } 232 break; 233 } 234 return (error); 235 } 236 237 static int 238 soo_poll(struct file *fp, int events, struct ucred *active_cred, 239 struct thread *td) 240 { 241 struct socket *so = fp->f_data; 242 #ifdef MAC 243 int error; 244 245 error = mac_socket_check_poll(active_cred, so); 246 if (error) 247 return (error); 248 #endif 249 return (sopoll(so, events, fp->f_cred, td)); 250 } 251 252 static int 253 soo_stat(struct file *fp, struct stat *ub, struct ucred *active_cred, 254 struct thread *td) 255 { 256 struct socket *so = fp->f_data; 257 #ifdef MAC 258 int error; 259 #endif 260 261 bzero((caddr_t)ub, sizeof (*ub)); 262 ub->st_mode = S_IFSOCK; 263 #ifdef MAC 264 error = mac_socket_check_stat(active_cred, so); 265 if (error) 266 return (error); 267 #endif 268 /* 269 * If SBS_CANTRCVMORE is set, but there's still data left in the 270 * receive buffer, the socket is still readable. 271 */ 272 SOCKBUF_LOCK(&so->so_rcv); 273 if ((so->so_rcv.sb_state & SBS_CANTRCVMORE) == 0 || 274 so->so_rcv.sb_cc != 0) 275 ub->st_mode |= S_IRUSR | S_IRGRP | S_IROTH; 276 ub->st_size = so->so_rcv.sb_cc - so->so_rcv.sb_ctl; 277 SOCKBUF_UNLOCK(&so->so_rcv); 278 /* Unlocked read. */ 279 if ((so->so_snd.sb_state & SBS_CANTSENDMORE) == 0) 280 ub->st_mode |= S_IWUSR | S_IWGRP | S_IWOTH; 281 ub->st_uid = so->so_cred->cr_uid; 282 ub->st_gid = so->so_cred->cr_gid; 283 return (*so->so_proto->pr_usrreqs->pru_sense)(so, ub); 284 } 285 286 /* 287 * API socket close on file pointer. We call soclose() to close the socket 288 * (including initiating closing protocols). soclose() will sorele() the 289 * file reference but the actual socket will not go away until the socket's 290 * ref count hits 0. 291 */ 292 static int 293 soo_close(struct file *fp, struct thread *td) 294 { 295 int error = 0; 296 struct socket *so; 297 298 so = fp->f_data; 299 fp->f_ops = &badfileops; 300 fp->f_data = NULL; 301 302 if (so) 303 error = soclose(so); 304 return (error); 305 } 306 307 static int 308 soo_fill_kinfo(struct file *fp, struct kinfo_file *kif, struct filedesc *fdp) 309 { 310 struct sockaddr *sa; 311 struct inpcb *inpcb; 312 struct unpcb *unpcb; 313 struct socket *so; 314 int error; 315 316 kif->kf_type = KF_TYPE_SOCKET; 317 so = fp->f_data; 318 kif->kf_sock_domain = so->so_proto->pr_domain->dom_family; 319 kif->kf_sock_type = so->so_type; 320 kif->kf_sock_protocol = so->so_proto->pr_protocol; 321 kif->kf_un.kf_sock.kf_sock_pcb = (uintptr_t)so->so_pcb; 322 switch (kif->kf_sock_domain) { 323 case AF_INET: 324 case AF_INET6: 325 if (kif->kf_sock_protocol == IPPROTO_TCP) { 326 if (so->so_pcb != NULL) { 327 inpcb = (struct inpcb *)(so->so_pcb); 328 kif->kf_un.kf_sock.kf_sock_inpcb = 329 (uintptr_t)inpcb->inp_ppcb; 330 } 331 } 332 break; 333 case AF_UNIX: 334 if (so->so_pcb != NULL) { 335 unpcb = (struct unpcb *)(so->so_pcb); 336 if (unpcb->unp_conn) { 337 kif->kf_un.kf_sock.kf_sock_unpconn = 338 (uintptr_t)unpcb->unp_conn; 339 kif->kf_un.kf_sock.kf_sock_rcv_sb_state = 340 so->so_rcv.sb_state; 341 kif->kf_un.kf_sock.kf_sock_snd_sb_state = 342 so->so_snd.sb_state; 343 } 344 } 345 break; 346 } 347 error = so->so_proto->pr_usrreqs->pru_sockaddr(so, &sa); 348 if (error == 0 && sa->sa_len <= sizeof(kif->kf_sa_local)) { 349 bcopy(sa, &kif->kf_sa_local, sa->sa_len); 350 free(sa, M_SONAME); 351 } 352 error = so->so_proto->pr_usrreqs->pru_peeraddr(so, &sa); 353 if (error == 0 && sa->sa_len <= sizeof(kif->kf_sa_peer)) { 354 bcopy(sa, &kif->kf_sa_peer, sa->sa_len); 355 free(sa, M_SONAME); 356 } 357 strncpy(kif->kf_path, so->so_proto->pr_domain->dom_name, 358 sizeof(kif->kf_path)); 359 return (0); 360 } 361