2017-04-21 10:43:26 +00:00
|
|
|
/*
|
2021-09-18 08:05:45 +00:00
|
|
|
* Copyright (C) 2017-2021 THL A29 Limited, a Tencent company.
|
2017-04-21 10:43:26 +00:00
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
|
|
|
* list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
|
|
* and/or other materials provided with the distribution.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
|
|
|
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
|
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
|
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _FSTACK_API_H
|
|
|
|
#define _FSTACK_API_H
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/select.h>
|
|
|
|
#include <sys/poll.h>
|
2017-06-14 07:25:35 +00:00
|
|
|
#include <netinet/in.h>
|
2017-08-04 09:40:50 +00:00
|
|
|
#include <sys/time.h>
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
#include "ff_event.h"
|
|
|
|
#include "ff_errno.h"
|
|
|
|
|
|
|
|
struct linux_sockaddr {
|
|
|
|
short sa_family;
|
2021-01-29 08:29:43 +00:00
|
|
|
char sa_data[14];
|
2017-04-21 10:43:26 +00:00
|
|
|
};
|
|
|
|
|
2019-07-12 12:56:01 +00:00
|
|
|
#define AF_INET6_LINUX 10
|
2019-07-15 09:22:15 +00:00
|
|
|
#define PF_INET6_LINUX AF_INET6_LINUX
|
2021-01-22 09:12:05 +00:00
|
|
|
#define AF_INET6_FREEBSD 28
|
|
|
|
#define PF_INET6_FREEBSD AF_INET6_FREEBSD
|
2019-07-12 12:56:01 +00:00
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
typedef int (*loop_func_t)(void *arg);
|
|
|
|
|
2017-08-08 14:36:49 +00:00
|
|
|
int ff_init(int argc, char * const argv[]);
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
void ff_run(loop_func_t loop, void *arg);
|
|
|
|
|
2017-11-09 13:22:13 +00:00
|
|
|
/* POSIX-LIKE api begin */
|
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
int ff_fcntl(int fd, int cmd, ...);
|
|
|
|
|
|
|
|
int ff_sysctl(const int *name, u_int namelen, void *oldp, size_t *oldlenp,
|
|
|
|
const void *newp, size_t newlen);
|
|
|
|
|
|
|
|
int ff_ioctl(int fd, unsigned long request, ...);
|
|
|
|
|
2023-02-17 11:10:41 +00:00
|
|
|
/*
|
|
|
|
* While get sockfd from this API, and then need set it to non-blocking mode like this,
|
|
|
|
* Otherwise, sometimes the socket interface will not work properly, such as `ff_write()`
|
|
|
|
*
|
|
|
|
* int on = 1;
|
|
|
|
* ff_ioctl(sockfd, FIONBIO, &on);
|
|
|
|
*
|
|
|
|
* See also `example/main.c`
|
|
|
|
*/
|
2017-04-21 10:43:26 +00:00
|
|
|
int ff_socket(int domain, int type, int protocol);
|
|
|
|
|
|
|
|
int ff_setsockopt(int s, int level, int optname, const void *optval,
|
|
|
|
socklen_t optlen);
|
|
|
|
|
|
|
|
int ff_getsockopt(int s, int level, int optname, void *optval,
|
|
|
|
socklen_t *optlen);
|
|
|
|
|
|
|
|
int ff_listen(int s, int backlog);
|
|
|
|
int ff_bind(int s, const struct linux_sockaddr *addr, socklen_t addrlen);
|
|
|
|
int ff_accept(int s, struct linux_sockaddr *addr, socklen_t *addrlen);
|
|
|
|
int ff_connect(int s, const struct linux_sockaddr *name, socklen_t namelen);
|
|
|
|
int ff_close(int fd);
|
|
|
|
int ff_shutdown(int s, int how);
|
|
|
|
|
|
|
|
int ff_getpeername(int s, struct linux_sockaddr *name,
|
|
|
|
socklen_t *namelen);
|
|
|
|
int ff_getsockname(int s, struct linux_sockaddr *name,
|
|
|
|
socklen_t *namelen);
|
|
|
|
|
|
|
|
ssize_t ff_read(int d, void *buf, size_t nbytes);
|
|
|
|
ssize_t ff_readv(int fd, const struct iovec *iov, int iovcnt);
|
|
|
|
|
2023-02-17 11:10:41 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Write data to the socket sendspace buf.
|
|
|
|
*
|
|
|
|
* Note:
|
|
|
|
* The `fd` parameter need set non-blocking mode in advance if F-Stack's APP.
|
|
|
|
* Otherwise if the `nbytes` parameter is greater than
|
|
|
|
* `net.inet.tcp.sendspace + net.inet.tcp.sendbuf_inc`,
|
|
|
|
* the API will return -1, but not the length that has been sent.
|
|
|
|
*
|
|
|
|
* You also can modify the value of `net.inet.tcp.sendspace`(default 16384 bytes)
|
|
|
|
* and `net.inet.tcp.sendbuf_inc`(default 16384 bytes) with `config.ini`.
|
|
|
|
* But it should be noted that not all parameters can take effect, such as 32768 and 32768.
|
|
|
|
* `ff_sysctl` can see there values while APP is running.
|
|
|
|
*/
|
2017-04-21 10:43:26 +00:00
|
|
|
ssize_t ff_write(int fd, const void *buf, size_t nbytes);
|
|
|
|
ssize_t ff_writev(int fd, const struct iovec *iov, int iovcnt);
|
|
|
|
|
|
|
|
ssize_t ff_send(int s, const void *buf, size_t len, int flags);
|
|
|
|
ssize_t ff_sendto(int s, const void *buf, size_t len, int flags,
|
|
|
|
const struct linux_sockaddr *to, socklen_t tolen);
|
|
|
|
ssize_t ff_sendmsg(int s, const struct msghdr *msg, int flags);
|
|
|
|
|
|
|
|
ssize_t ff_recv(int s, void *buf, size_t len, int flags);
|
|
|
|
ssize_t ff_recvfrom(int s, void *buf, size_t len, int flags,
|
|
|
|
struct linux_sockaddr *from, socklen_t *fromlen);
|
|
|
|
ssize_t ff_recvmsg(int s, struct msghdr *msg, int flags);
|
|
|
|
|
|
|
|
int ff_select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds,
|
|
|
|
struct timeval *timeout);
|
|
|
|
|
|
|
|
int ff_poll(struct pollfd fds[], nfds_t nfds, int timeout);
|
|
|
|
|
|
|
|
int ff_kqueue(void);
|
2022-09-02 08:51:22 +00:00
|
|
|
int ff_kevent(int kq, const struct kevent *changelist, int nchanges,
|
2017-04-21 10:43:26 +00:00
|
|
|
struct kevent *eventlist, int nevents, const struct timespec *timeout);
|
2022-09-02 08:51:22 +00:00
|
|
|
int ff_kevent_do_each(int kq, const struct kevent *changelist, int nchanges,
|
|
|
|
void *eventlist, int nevents, const struct timespec *timeout,
|
2017-06-13 09:50:51 +00:00
|
|
|
void (*do_each)(void **, struct kevent *));
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2023-10-19 08:48:44 +00:00
|
|
|
#if __GLIBC__ > 2 || (__GLIBC__ == 2 && __GLIBC_MINOR__ >= 31)
|
|
|
|
int ff_gettimeofday(struct timeval *tv, void *tz);
|
|
|
|
#else
|
2017-08-04 09:40:50 +00:00
|
|
|
int ff_gettimeofday(struct timeval *tv, struct timezone *tz);
|
2023-10-19 08:48:44 +00:00
|
|
|
#endif
|
2017-08-04 09:40:50 +00:00
|
|
|
|
2018-06-10 12:37:34 +00:00
|
|
|
int ff_dup(int oldfd);
|
|
|
|
int ff_dup2(int oldfd, int newfd);
|
|
|
|
|
2017-11-09 13:22:13 +00:00
|
|
|
/* POSIX-LIKE api end */
|
|
|
|
|
|
|
|
|
|
|
|
/* Tests if fd is used by F-Stack */
|
2017-08-08 14:36:49 +00:00
|
|
|
extern int ff_fdisused(int fd);
|
|
|
|
|
2017-12-08 10:32:08 +00:00
|
|
|
extern int ff_getmaxfd(void);
|
2017-11-09 13:22:13 +00:00
|
|
|
|
2017-06-14 07:25:35 +00:00
|
|
|
/* route api begin */
|
|
|
|
enum FF_ROUTE_CTL {
|
|
|
|
FF_ROUTE_ADD,
|
|
|
|
FF_ROUTE_DEL,
|
|
|
|
FF_ROUTE_CHANGE,
|
|
|
|
};
|
|
|
|
|
|
|
|
enum FF_ROUTE_FLAG {
|
|
|
|
FF_RTF_HOST,
|
|
|
|
FF_RTF_GATEWAY,
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* On success, 0 is returned.
|
|
|
|
* On error, -1 is returned, and errno is set appropriately.
|
|
|
|
*/
|
|
|
|
int ff_route_ctl(enum FF_ROUTE_CTL req, enum FF_ROUTE_FLAG flag,
|
|
|
|
struct linux_sockaddr *dst, struct linux_sockaddr *gw,
|
|
|
|
struct linux_sockaddr *netmask);
|
|
|
|
|
2017-11-01 09:38:22 +00:00
|
|
|
/* route api end */
|
|
|
|
|
2017-11-09 13:22:13 +00:00
|
|
|
|
|
|
|
/* dispatch api begin */
|
2019-03-08 07:12:57 +00:00
|
|
|
#define FF_DISPATCH_ERROR (-1)
|
|
|
|
#define FF_DISPATCH_RESPONSE (-2)
|
2017-11-09 13:22:13 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Packet dispatch callback function.
|
|
|
|
* Implemented by user.
|
|
|
|
*
|
|
|
|
* @param data
|
2017-11-14 08:34:53 +00:00
|
|
|
* The data pointer of this packet.
|
2017-11-09 13:22:13 +00:00
|
|
|
* @param len
|
2017-11-14 08:34:53 +00:00
|
|
|
* The length of this packet.
|
|
|
|
* @param queue_id
|
|
|
|
* Current queue of this packet.
|
2017-11-09 13:22:13 +00:00
|
|
|
* @param nb_queues
|
|
|
|
* Number of queues to be dispatched.
|
|
|
|
*
|
|
|
|
* @return 0 to (nb_queues - 1)
|
|
|
|
* The queue id that the packet will be dispatched to.
|
2019-06-25 04:38:19 +00:00
|
|
|
* @return FF_DISPATCH_ERROR (-1)
|
2017-11-09 13:22:13 +00:00
|
|
|
* Error occurs or packet is handled by user, packet will be freed.
|
2019-06-25 04:38:19 +00:00
|
|
|
* @return FF_DISPATCH_RESPONSE (-2)
|
|
|
|
* Packet is handled by user, packet will be responsed.
|
2017-11-09 13:22:13 +00:00
|
|
|
*
|
|
|
|
*/
|
2019-03-08 07:12:57 +00:00
|
|
|
typedef int (*dispatch_func_t)(void *data, uint16_t *len,
|
2017-11-14 08:34:53 +00:00
|
|
|
uint16_t queue_id, uint16_t nb_queues);
|
2017-11-09 13:22:13 +00:00
|
|
|
|
|
|
|
/* regist a packet dispath function */
|
|
|
|
void ff_regist_packet_dispatcher(dispatch_func_t func);
|
|
|
|
|
|
|
|
/* dispatch api end */
|
|
|
|
|
2021-01-22 09:12:05 +00:00
|
|
|
/* pcb lddr api begin */
|
|
|
|
/*
|
|
|
|
* pcb lddr callback function.
|
|
|
|
* Implemented by user.
|
|
|
|
*
|
|
|
|
* @param family
|
|
|
|
* The remote server addr, should be AF_INET or AF_INET6.
|
|
|
|
* @param dst_addr
|
|
|
|
* The remote server addr, should be (in_addr *) or (in6_addr *).
|
|
|
|
* @param dst_port
|
|
|
|
* The remote server port.
|
|
|
|
* @param src_addr
|
|
|
|
* Return parameter.
|
|
|
|
* The local addr, should be (in_addr *) or (in6_addr *).
|
|
|
|
* If set (INADDR_ANY) or (in6addr_any), the app then will
|
|
|
|
* call `in_pcbladdr()` to get laddr.
|
|
|
|
*
|
|
|
|
* @return error_no
|
|
|
|
* 0 means success.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
typedef int (*pcblddr_func_t)(uint16_t family, void *dst_addr,
|
|
|
|
uint16_t dst_port, void *src_addr);
|
|
|
|
|
|
|
|
/* regist a pcb lddr function */
|
|
|
|
void ff_regist_pcblddr_fun(pcblddr_func_t func);
|
|
|
|
|
|
|
|
/* pcb lddr api end */
|
2017-11-09 13:22:13 +00:00
|
|
|
|
2017-11-01 09:38:22 +00:00
|
|
|
/* internal api begin */
|
|
|
|
|
2018-06-10 12:37:34 +00:00
|
|
|
/* FreeBSD style calls. Used for tools. */
|
|
|
|
int ff_ioctl_freebsd(int fd, unsigned long request, ...);
|
|
|
|
int ff_setsockopt_freebsd(int s, int level, int optname,
|
|
|
|
const void *optval, socklen_t optlen);
|
|
|
|
int ff_getsockopt_freebsd(int s, int level, int optname,
|
|
|
|
void *optval, socklen_t *optlen);
|
|
|
|
|
2017-06-14 07:25:35 +00:00
|
|
|
/*
|
2017-11-01 09:38:22 +00:00
|
|
|
* Handle rtctl.
|
2017-06-14 07:25:35 +00:00
|
|
|
* The data is a pointer to struct rt_msghdr.
|
|
|
|
*/
|
|
|
|
int ff_rtioctl(int fib, void *data, unsigned *plen, unsigned maxlen);
|
|
|
|
|
2017-11-01 09:38:22 +00:00
|
|
|
/*
|
|
|
|
* Handle ngctl.
|
|
|
|
*/
|
|
|
|
enum FF_NGCTL_CMD {
|
|
|
|
NGCTL_SOCKET,
|
|
|
|
NGCTL_BIND,
|
|
|
|
NGCTL_CONNECT,
|
|
|
|
NGCTL_SEND,
|
|
|
|
NGCTL_RECV,
|
|
|
|
NGCTL_CLOSE,
|
|
|
|
};
|
|
|
|
|
|
|
|
int ff_ngctl(int cmd, void *data);
|
|
|
|
|
|
|
|
/* internal api end */
|
2017-06-14 07:25:35 +00:00
|
|
|
|
2022-04-15 10:00:49 +00:00
|
|
|
/* zero ccopy API begin */
|
|
|
|
struct ff_zc_mbuf {
|
|
|
|
void *bsd_mbuf; /* point to the head mbuf */
|
|
|
|
void *bsd_mbuf_off; /* ponit to the current mbuf in the mbuf chain with offset */
|
|
|
|
int off; /* the offset of total mbuf, APP shouldn't modify it */
|
|
|
|
int len; /* the total len of the mbuf chain */
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Get the ff zero copy mbuf.
|
|
|
|
*
|
|
|
|
* @param m
|
|
|
|
* The ponitor of 'sturct ff_zc_mbuf', and can't be NULL.
|
|
|
|
* Can used by 'ff_zc_mbuf_write' and 'ff_zc_mbuf_read'.
|
|
|
|
* @param len
|
|
|
|
* The total buf len of mbuf chain that you want to alloc.
|
|
|
|
*
|
|
|
|
* @return error_no
|
|
|
|
* 0 means success.
|
|
|
|
* -1 means error.
|
|
|
|
*/
|
|
|
|
int ff_zc_mbuf_get(struct ff_zc_mbuf *m, int len);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Write data to the mbuf chain in 'sturct ff_zc_mbuf'.
|
|
|
|
* APP can call this function multiple times, need pay attion to the offset of data.
|
|
|
|
* but the total len can't be larger than m->len.
|
|
|
|
* After this fuction return success,
|
2022-04-17 13:05:32 +00:00
|
|
|
*
|
|
|
|
* the struct 'ff_zc_mbuf *m' can be reused in `ff_zc_mbuf_get` and then Use normally.
|
|
|
|
* Nerver directly reused in `ff_zc_mbuf_write` before recall `ff_zc_mbuf_get`.
|
2022-04-15 10:00:49 +00:00
|
|
|
*
|
|
|
|
* APP nedd call 'ff_write' to send data actually after finish write data to mbuf,
|
|
|
|
* And use 'bsd_mbuf' of 'struct ff_zc_mbuf' as the 'buf' argument.
|
|
|
|
*
|
|
|
|
* See 'example/main_zc.c'
|
|
|
|
*
|
|
|
|
* @param m
|
|
|
|
* The ponitor of 'sturct ff_zc_mbuf', must be call 'ff_zc_mbuf_get' first.
|
|
|
|
* @param data
|
|
|
|
* The pointer of data that want to write to socket, need pay attion to the offset.
|
|
|
|
* @param len
|
|
|
|
* The len that APP want to write to mbuf chain this time.
|
|
|
|
*
|
|
|
|
* @return error_no
|
|
|
|
* 0 means success.
|
|
|
|
* -1 means error.
|
|
|
|
*/
|
|
|
|
int ff_zc_mbuf_write(struct ff_zc_mbuf *m, const char *data, int len);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Read data to the mbuf chain in 'sturct ff_zc_mbuf'.
|
|
|
|
* not implemented now.
|
|
|
|
*/
|
|
|
|
int ff_zc_mbuf_read(struct ff_zc_mbuf *m, const char *data, int len);
|
|
|
|
|
|
|
|
/* ZERO COPY API end */
|
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|