/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License (the "License"). * You may not use this file except in compliance with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright 2011 Nexenta Systems, Inc. All rights reserved. * Copyright (c) 2008, 2010, Oracle and/or its affiliates. All rights reserved. * Copyright 2015, Joyent, Inc. */ #ifndef _SYS_KSOCKET_H_ #define _SYS_KSOCKET_H_ #ifdef __cplusplus extern "C" { #endif /* Opaque kernel socket type */ typedef struct __ksocket *ksocket_t; struct nmsghdr; struct msgb; /* avoiding sys/stream.h here */ /* flag bit for each Callback Event */ #define KSOCKET_CB_CONNECTED 0x00000001 #define KSOCKET_CB_CONNECTFAILED 0x00000002 #define KSOCKET_CB_DISCONNECTED 0x00000004 #define KSOCKET_CB_NEWDATA 0x00000008 #define KSOCKET_CB_NEWCONN 0x00000010 #define KSOCKET_CB_CANSEND 0x00000020 #define KSOCKET_CB_OOBDATA 0x00000040 #define KSOCKET_CB_CANTSENDMORE 0x00000080 #define KSOCKET_CB_CANTRECVMORE 0x00000100 #define KSOCKET_CB_ERROR 0x00000200 /* * Kernel Socket Callback Events */ typedef enum ksocket_event { KSOCKET_EV_CONNECTED, KSOCKET_EV_CONNECTFAILED, KSOCKET_EV_DISCONNECTED, KSOCKET_EV_OOBDATA, KSOCKET_EV_NEWDATA, KSOCKET_EV_NEWCONN, KSOCKET_EV_CANSEND, KSOCKET_EV_CANTSENDMORE, KSOCKET_EV_CANTRECVMORE, KSOCKET_EV_ERROR } ksocket_callback_event_t; typedef void (*ksocket_callback_t)(ksocket_t, ksocket_callback_event_t, void *, uintptr_t); typedef struct ksocket_callbacks { uint32_t ksock_cb_flags; ksocket_callback_t ksock_cb_connected; ksocket_callback_t ksock_cb_connectfailed; ksocket_callback_t ksock_cb_disconnected; ksocket_callback_t ksock_cb_newdata; ksocket_callback_t ksock_cb_newconn; ksocket_callback_t ksock_cb_cansend; ksocket_callback_t ksock_cb_oobdata; ksocket_callback_t ksock_cb_cantsendmore; ksocket_callback_t ksock_cb_cantrecvmore; ksocket_callback_t ksock_cb_error; } ksocket_callbacks_t; #define KSOCKET_SLEEP SOCKET_SLEEP #define KSOCKET_NOSLEEP SOCKET_NOSLEEP extern int ksocket_socket(ksocket_t *, int, int, int, int, struct cred *); extern int ksocket_bind(ksocket_t, struct sockaddr *, socklen_t, struct cred *); extern int ksocket_listen(ksocket_t, int, struct cred *); extern int ksocket_accept(ksocket_t, struct sockaddr *, socklen_t *, ksocket_t *, struct cred *); extern int ksocket_connect(ksocket_t, struct sockaddr *, socklen_t, struct cred *); extern int ksocket_send(ksocket_t, void *, size_t, int, size_t *, struct cred *); extern int ksocket_sendto(ksocket_t, void *, size_t, int, struct sockaddr *, socklen_t, size_t *, struct cred *); extern int ksocket_sendmsg(ksocket_t, struct nmsghdr *, int, size_t *, struct cred *); extern int ksocket_sendmblk(ksocket_t, struct nmsghdr *, int, struct msgb **, struct cred *); extern int ksocket_recv(ksocket_t, void *, size_t, int, size_t *, struct cred *); extern int ksocket_recvfrom(ksocket_t, void *, size_t, int, struct sockaddr *, socklen_t *, size_t *, struct cred *); extern int ksocket_recvmsg(ksocket_t, struct nmsghdr *, int, size_t *, struct cred *); extern int ksocket_shutdown(ksocket_t, int, struct cred *); extern int ksocket_setsockopt(ksocket_t, int, int, const void *, int, struct cred *); extern int ksocket_getsockopt(ksocket_t, int, int, void *, int *, struct cred *); extern int ksocket_getpeername(ksocket_t, struct sockaddr *, socklen_t *, struct cred *); extern int ksocket_getsockname(ksocket_t, struct sockaddr *, socklen_t *, struct cred *); extern int ksocket_ioctl(ksocket_t, int, intptr_t, int *, struct cred *); extern int ksocket_spoll(ksocket_t, int, short, short *, struct cred *); extern int ksocket_setcallbacks(ksocket_t, ksocket_callbacks_t *, void *, struct cred *); extern int ksocket_close(ksocket_t, struct cred *); extern void ksocket_hold(ksocket_t); extern void ksocket_rele(ksocket_t); /* * These functions allow an alternative way for a ksocket to directly * receive data when it arrives in sockfs rather than having it queued * in a socket buffer that it must separately poll. The use of this * results in no data being queued in sockfs. * * When the receive function receives data, it is responsible for always * consuming all of the data. The return value of the callback function * is used to indicate flow control and backpressure (similar to * mc_tx(9E)). If, after processing the data, additional data can be * received and processed, then the callback function should return * B_TRUE. Otherwise it should return B_FALSE. This will result in the * lower level socket interfaces (e.g. TCP) understanding that * backpressure has been asserted (as though the sockfs buffer is full). * * Once whatever conditions that caused the callback function to assert * that it needed to assert flow control are done, then it must call * ksocket_krecv_unblock() to allow the flow to continue. If the receive * callback ever returns B_FALSE there will generally be no additional * data received until this is called. */ typedef boolean_t (*ksocket_krecv_f)(ksocket_t, struct msgb *, size_t, int, void *); extern int ksocket_krecv_set(ksocket_t, ksocket_krecv_f, void *); extern void ksocket_krecv_unblock(ksocket_t); #ifdef __cplusplus } #endif #endif /* _SYS_KSOCKET_H_ */