xref: /f-stack/app/micro_thread/mt_sys_hook.cpp (revision 35a81399)
1a9643ea8Slogwang 
2a9643ea8Slogwang /**
3a9643ea8Slogwang  * Tencent is pleased to support the open source community by making MSEC available.
4a9643ea8Slogwang  *
5a9643ea8Slogwang  * Copyright (C) 2016 THL A29 Limited, a Tencent company. All rights reserved.
6a9643ea8Slogwang  *
7a9643ea8Slogwang  * Licensed under the GNU General Public License, Version 2.0 (the "License");
8a9643ea8Slogwang  * you may not use this file except in compliance with the License. You may
9a9643ea8Slogwang  * obtain a copy of the License at
10a9643ea8Slogwang  *
11a9643ea8Slogwang  *     https://opensource.org/licenses/GPL-2.0
12a9643ea8Slogwang  *
13a9643ea8Slogwang  * Unless required by applicable law or agreed to in writing, software distributed under the
14a9643ea8Slogwang  * License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
15a9643ea8Slogwang  * either express or implied. See the License for the specific language governing permissions
16a9643ea8Slogwang  * and limitations under the License.
17a9643ea8Slogwang  */
18a9643ea8Slogwang 
19a9643ea8Slogwang 
20a9643ea8Slogwang /**
21a9643ea8Slogwang  *  @filename mt_sys_hook.cpp
22a9643ea8Slogwang  */
23a9643ea8Slogwang 
24a9643ea8Slogwang #include <stdio.h>
25a9643ea8Slogwang #include <stdlib.h>
26a9643ea8Slogwang #include <stdint.h>
27a9643ea8Slogwang #include <unistd.h>
28a9643ea8Slogwang #include <errno.h>
29a9643ea8Slogwang #include <assert.h>
30a9643ea8Slogwang #include <sys/types.h>
31a9643ea8Slogwang #include <sys/stat.h>
32a9643ea8Slogwang #include <stdarg.h>
33a9643ea8Slogwang 
34a9643ea8Slogwang #include "micro_thread.h"
35a9643ea8Slogwang #include "mt_sys_hook.h"
36a9643ea8Slogwang #include "ff_hook.h"
37a9643ea8Slogwang 
38a9643ea8Slogwang using namespace NS_MICRO_THREAD;
39a9643ea8Slogwang 
40a9643ea8Slogwang 
41a9643ea8Slogwang #define MT_HOOK_MAX_FD      65535*2
42a9643ea8Slogwang #define MT_FD_FLG_INUSE     0x1
43a9643ea8Slogwang #define MT_FD_FLG_UNBLOCK   0x2
44a9643ea8Slogwang 
45a9643ea8Slogwang typedef struct socket_hook_info
46a9643ea8Slogwang {
47*35a81399Slogwang     int     sock_flag;
48*35a81399Slogwang     int     read_timeout;
49*35a81399Slogwang     int     write_timeout;
50a9643ea8Slogwang }MtHookFd;
51a9643ea8Slogwang 
52*35a81399Slogwang MtSyscallFuncTab       g_mt_syscall_tab;
53*35a81399Slogwang int                    g_mt_hook_flag;
54*35a81399Slogwang int                    g_ff_hook_flag;
55*35a81399Slogwang static MtHookFd        g_mt_hook_fd_tab[MT_HOOK_MAX_FD];
56a9643ea8Slogwang 
mt_hook_find_fd(int fd)57a9643ea8Slogwang MtHookFd* mt_hook_find_fd(int fd)
58a9643ea8Slogwang {
59a9643ea8Slogwang     if ((fd < 0) || (fd >= MT_HOOK_MAX_FD)) {
60a9643ea8Slogwang         return NULL;
61a9643ea8Slogwang     }
62a9643ea8Slogwang 
63a9643ea8Slogwang     MtHookFd* fd_info =  &g_mt_hook_fd_tab[fd];
64a9643ea8Slogwang     if (!(fd_info->sock_flag & MT_FD_FLG_INUSE)) {
65a9643ea8Slogwang         return NULL;
66a9643ea8Slogwang     } else {
67a9643ea8Slogwang         return fd_info;
68a9643ea8Slogwang     }
69a9643ea8Slogwang }
70a9643ea8Slogwang 
mt_hook_new_fd(int fd)71a9643ea8Slogwang void mt_hook_new_fd(int fd)
72a9643ea8Slogwang {
73a9643ea8Slogwang     if ((fd < 0) || (fd >= MT_HOOK_MAX_FD)) {
74a9643ea8Slogwang         return;
75a9643ea8Slogwang     }
76a9643ea8Slogwang 
77a9643ea8Slogwang     MtHookFd* fd_info       = &g_mt_hook_fd_tab[fd];
78a9643ea8Slogwang     fd_info->sock_flag      = MT_FD_FLG_INUSE;
79a9643ea8Slogwang     fd_info->read_timeout   = 500;
80a9643ea8Slogwang     fd_info->write_timeout  = 500;
81a9643ea8Slogwang }
82a9643ea8Slogwang 
mt_hook_free_fd(int fd)83a9643ea8Slogwang void mt_hook_free_fd(int fd)
84a9643ea8Slogwang {
85a9643ea8Slogwang     if ((fd < 0) || (fd >= MT_HOOK_MAX_FD)) {
86a9643ea8Slogwang         return;
87a9643ea8Slogwang     }
88a9643ea8Slogwang 
89a9643ea8Slogwang     MtHookFd* fd_info       = &g_mt_hook_fd_tab[fd];
90a9643ea8Slogwang     fd_info->sock_flag      = 0;
91a9643ea8Slogwang     fd_info->read_timeout   = 0;
92a9643ea8Slogwang     fd_info->write_timeout  = 0;
93a9643ea8Slogwang }
94a9643ea8Slogwang 
95a9643ea8Slogwang #ifdef __cplusplus
96a9643ea8Slogwang extern "C" {
97a9643ea8Slogwang #endif
ioctl(int fd,unsigned long cmd,...)98a9643ea8Slogwang int ioctl(int fd, unsigned long cmd, ...)
99a9643ea8Slogwang {
100a9643ea8Slogwang     va_list ap;
101a9643ea8Slogwang     va_start(ap, cmd);
102a9643ea8Slogwang     void* arg = va_arg(ap, void *);
103a9643ea8Slogwang     va_end(ap);
104a9643ea8Slogwang 
105a9643ea8Slogwang     mt_hook_syscall(ioctl);
106a9643ea8Slogwang     MtHookFd* hook_fd = mt_hook_find_fd(fd);
107a02c88d6Slogwang     if (!mt_hook_active() || !hook_fd || !ff_hook_active())
108a9643ea8Slogwang     {
109a9643ea8Slogwang         return ff_hook_ioctl(fd, cmd, arg);
110a9643ea8Slogwang     }
111a9643ea8Slogwang 
112a9643ea8Slogwang     if (cmd == FIONBIO)
113a9643ea8Slogwang     {
114a9643ea8Slogwang         int flags =  (arg != NULL) ? *((int*)arg) : 0;
115a9643ea8Slogwang         if (flags != 0) {
116a9643ea8Slogwang             hook_fd->sock_flag |= MT_FD_FLG_UNBLOCK;
117a9643ea8Slogwang         }
118a9643ea8Slogwang     }
119a9643ea8Slogwang 
120a9643ea8Slogwang     return ff_hook_ioctl(fd, cmd, arg);
121a9643ea8Slogwang }
122a9643ea8Slogwang 
socket(int domain,int type,int protocol)123a9643ea8Slogwang int socket(int domain, int type, int protocol)
124a9643ea8Slogwang {
125a9643ea8Slogwang     mt_hook_syscall(socket);
126a9643ea8Slogwang 
127a02c88d6Slogwang     if (!ff_hook_active())
128a9643ea8Slogwang     {
129a02c88d6Slogwang         return mt_real_func(socket)(domain, type, protocol);
130a9643ea8Slogwang     }
131a9643ea8Slogwang 
132a9643ea8Slogwang     int fd = ff_hook_socket(domain, type, protocol);
133a9643ea8Slogwang     if (fd < 0)
134a9643ea8Slogwang     {
135a9643ea8Slogwang         return fd;
136a9643ea8Slogwang     }
137a9643ea8Slogwang 
138a9643ea8Slogwang     mt_hook_new_fd(fd);
139a9643ea8Slogwang 
140a9643ea8Slogwang     mt_hook_syscall(ioctl);
141a9643ea8Slogwang     int nb = 1;
142a9643ea8Slogwang     ff_hook_ioctl(fd, FIONBIO, &nb);
143a9643ea8Slogwang 
144a9643ea8Slogwang     return fd;
145a9643ea8Slogwang }
146a9643ea8Slogwang 
close(int fd)147a9643ea8Slogwang int close(int fd)
148a9643ea8Slogwang {
149a9643ea8Slogwang     mt_hook_syscall(close);
150a02c88d6Slogwang     if (!ff_hook_active())
151a9643ea8Slogwang     {
152a02c88d6Slogwang         return mt_real_func(close)(fd);
153a9643ea8Slogwang     }
154a9643ea8Slogwang 
155a9643ea8Slogwang     mt_hook_free_fd(fd);
156a9643ea8Slogwang     return ff_hook_close(fd);
157a9643ea8Slogwang }
158a9643ea8Slogwang 
connect(int fd,const struct sockaddr * address,socklen_t address_len)159a9643ea8Slogwang int connect(int fd, const struct sockaddr *address, socklen_t address_len)
160a9643ea8Slogwang {
161a9643ea8Slogwang     mt_hook_syscall(connect);
162a9643ea8Slogwang     MtHookFd* hook_fd = mt_hook_find_fd(fd);
163a02c88d6Slogwang     if (!mt_hook_active() || !hook_fd || !ff_hook_active())
164a9643ea8Slogwang     {
165a02c88d6Slogwang         return mt_real_func(connect)(fd, address, address_len);
166a9643ea8Slogwang     }
167a9643ea8Slogwang 
168a9643ea8Slogwang     if (hook_fd->sock_flag & MT_FD_FLG_UNBLOCK)
169a9643ea8Slogwang     {
170a9643ea8Slogwang         return ff_hook_connect(fd, address, address_len);
171a9643ea8Slogwang     }
172a9643ea8Slogwang 
173a9643ea8Slogwang     return MtFrame::connect(fd, address, (int)address_len, hook_fd->write_timeout);
174a9643ea8Slogwang }
175a9643ea8Slogwang 
read(int fd,void * buf,size_t nbyte)176a9643ea8Slogwang ssize_t read(int fd, void *buf, size_t nbyte)
177a9643ea8Slogwang {
178a9643ea8Slogwang     mt_hook_syscall(read);
179a9643ea8Slogwang     MtHookFd* hook_fd = mt_hook_find_fd(fd);
180a02c88d6Slogwang     if (!mt_hook_active() || !hook_fd || !ff_hook_active())
181a9643ea8Slogwang     {
182a02c88d6Slogwang         return mt_real_func(read)(fd, buf, nbyte);
183a9643ea8Slogwang     }
184a9643ea8Slogwang 
185a9643ea8Slogwang     if (hook_fd->sock_flag & MT_FD_FLG_UNBLOCK)
186a9643ea8Slogwang     {
187a9643ea8Slogwang         return ff_hook_read(fd, buf, nbyte);
188a9643ea8Slogwang     }
189a9643ea8Slogwang 
190a9643ea8Slogwang     return MtFrame::read(fd, buf, nbyte, hook_fd->read_timeout);
191a9643ea8Slogwang }
192a9643ea8Slogwang 
write(int fd,const void * buf,size_t nbyte)193a9643ea8Slogwang ssize_t write(int fd, const void *buf, size_t nbyte)
194a9643ea8Slogwang {
195a9643ea8Slogwang     mt_hook_syscall(write);
196a9643ea8Slogwang     MtHookFd* hook_fd = mt_hook_find_fd(fd);
197a02c88d6Slogwang     if (!mt_hook_active() || !hook_fd || !ff_hook_active())
198a9643ea8Slogwang     {
199a02c88d6Slogwang         return mt_real_func(write)(fd, buf, nbyte);
200a9643ea8Slogwang     }
201a9643ea8Slogwang 
202a9643ea8Slogwang     if (hook_fd->sock_flag & MT_FD_FLG_UNBLOCK)
203a9643ea8Slogwang     {
204a9643ea8Slogwang         return ff_hook_write(fd, buf, nbyte);
205a9643ea8Slogwang     }
206a9643ea8Slogwang 
207a9643ea8Slogwang     return MtFrame::write(fd, buf, nbyte, hook_fd->write_timeout);
208a9643ea8Slogwang }
209a9643ea8Slogwang 
sendto(int fd,const void * message,size_t length,int flags,const struct sockaddr * dest_addr,socklen_t dest_len)210a9643ea8Slogwang ssize_t sendto(int fd, const void *message, size_t length, int flags,
211a9643ea8Slogwang                const struct sockaddr *dest_addr, socklen_t dest_len)
212a9643ea8Slogwang {
213a9643ea8Slogwang     mt_hook_syscall(sendto);
214a9643ea8Slogwang     MtHookFd* hook_fd = mt_hook_find_fd(fd);
215a02c88d6Slogwang     if (!mt_hook_active() || !hook_fd || !ff_hook_active())
216a9643ea8Slogwang     {
217a02c88d6Slogwang         return mt_real_func(sendto)(fd, message, length, flags, dest_addr, dest_len);
218a9643ea8Slogwang     }
219a9643ea8Slogwang 
220a9643ea8Slogwang     if (hook_fd->sock_flag & MT_FD_FLG_UNBLOCK)
221a9643ea8Slogwang     {
222a9643ea8Slogwang         return ff_hook_sendto(fd, message, length, flags, dest_addr, dest_len);
223a9643ea8Slogwang     }
224a9643ea8Slogwang 
225a9643ea8Slogwang     return MtFrame::sendto(fd, message, (int)length, flags,
226a9643ea8Slogwang                            dest_addr, dest_len, hook_fd->write_timeout);
227a9643ea8Slogwang }
228a9643ea8Slogwang 
recvfrom(int fd,void * buffer,size_t length,int flags,struct sockaddr * address,socklen_t * address_len)229a9643ea8Slogwang ssize_t recvfrom(int fd, void *buffer, size_t length, int flags,
230a9643ea8Slogwang                   struct sockaddr *address, socklen_t *address_len)
231a9643ea8Slogwang {
232a9643ea8Slogwang     mt_hook_syscall(recvfrom);
233a9643ea8Slogwang     MtHookFd* hook_fd = mt_hook_find_fd(fd);
234a02c88d6Slogwang     if (!mt_hook_active() || !hook_fd || !ff_hook_active())
235a9643ea8Slogwang     {
236a02c88d6Slogwang         return mt_real_func(recvfrom)(fd, buffer, length, flags, address, address_len);
237a9643ea8Slogwang     }
238a9643ea8Slogwang 
239a9643ea8Slogwang     if (hook_fd->sock_flag & MT_FD_FLG_UNBLOCK)
240a9643ea8Slogwang     {
241a9643ea8Slogwang         return ff_hook_recvfrom(fd, buffer, length, flags, address, address_len);
242a9643ea8Slogwang     }
243a9643ea8Slogwang 
244a9643ea8Slogwang     return MtFrame::recvfrom(fd, buffer, length, flags, address, address_len, hook_fd->read_timeout);
245a9643ea8Slogwang 
246a9643ea8Slogwang }
247a9643ea8Slogwang 
recv(int fd,void * buffer,size_t length,int flags)248a9643ea8Slogwang ssize_t recv(int fd, void *buffer, size_t length, int flags)
249a9643ea8Slogwang {
250a9643ea8Slogwang     mt_hook_syscall(recv);
251a9643ea8Slogwang     MtHookFd* hook_fd = mt_hook_find_fd(fd);
252a02c88d6Slogwang     if (!mt_hook_active() || !hook_fd || !ff_hook_active())
253a9643ea8Slogwang     {
254a02c88d6Slogwang         return mt_real_func(recv)(fd, buffer, length, flags);
255a9643ea8Slogwang     }
256a9643ea8Slogwang 
257a9643ea8Slogwang     if (hook_fd->sock_flag & MT_FD_FLG_UNBLOCK)
258a9643ea8Slogwang     {
259a9643ea8Slogwang         return ff_hook_recv(fd, buffer, length, flags);
260a9643ea8Slogwang     }
261a9643ea8Slogwang 
262a9643ea8Slogwang     return MtFrame::recv(fd, buffer, length, flags, hook_fd->read_timeout);
263a9643ea8Slogwang }
264a9643ea8Slogwang 
send(int fd,const void * buf,size_t nbyte,int flags)265a9643ea8Slogwang ssize_t send(int fd, const void *buf, size_t nbyte, int flags)
266a9643ea8Slogwang {
267a9643ea8Slogwang     mt_hook_syscall(send);
268a9643ea8Slogwang     MtHookFd* hook_fd = mt_hook_find_fd(fd);
269a02c88d6Slogwang     if (!mt_hook_active() || !hook_fd || !ff_hook_active())
270a9643ea8Slogwang     {
271a02c88d6Slogwang         return mt_real_func(send)(fd, buf, nbyte, flags);
272a9643ea8Slogwang     }
273a9643ea8Slogwang 
274a9643ea8Slogwang     if (hook_fd->sock_flag & MT_FD_FLG_UNBLOCK)
275a9643ea8Slogwang     {
276a9643ea8Slogwang         return ff_hook_send(fd, buf, nbyte, flags);
277a9643ea8Slogwang     }
278a9643ea8Slogwang 
279a9643ea8Slogwang     return MtFrame::send(fd, buf, nbyte, flags, hook_fd->write_timeout);
280a9643ea8Slogwang }
281a9643ea8Slogwang 
setsockopt(int fd,int level,int option_name,const void * option_value,socklen_t option_len)282a9643ea8Slogwang int setsockopt(int fd, int level, int option_name, const void *option_value, socklen_t option_len)
283a9643ea8Slogwang {
284a9643ea8Slogwang     mt_hook_syscall(setsockopt);
285a9643ea8Slogwang     MtHookFd* hook_fd = mt_hook_find_fd(fd);
286a02c88d6Slogwang     if (!mt_hook_active() || !hook_fd || !ff_hook_active())
287a9643ea8Slogwang     {
288a02c88d6Slogwang         return mt_real_func(setsockopt)(fd, level, option_name, option_value, option_len);
289a9643ea8Slogwang     }
290a9643ea8Slogwang 
291a9643ea8Slogwang     if (SOL_SOCKET == level)
292a9643ea8Slogwang     {
293a9643ea8Slogwang         struct timeval *val = (struct timeval*)option_value;
294a9643ea8Slogwang         if (SO_RCVTIMEO == option_name)
295a9643ea8Slogwang         {
296a9643ea8Slogwang             hook_fd->read_timeout = val->tv_sec * 1000 + val->tv_usec / 1000;
297a9643ea8Slogwang         }
298a9643ea8Slogwang         else if (SO_SNDTIMEO == option_name)
299a9643ea8Slogwang         {
300a9643ea8Slogwang             hook_fd->write_timeout = val->tv_sec * 1000 + val->tv_usec / 1000;
301a9643ea8Slogwang         }
302a9643ea8Slogwang     }
303a9643ea8Slogwang 
304a9643ea8Slogwang     return ff_hook_setsockopt(fd, level, option_name, option_value, option_len);
305a9643ea8Slogwang }
306a9643ea8Slogwang 
fcntl(int fd,int cmd,...)307a9643ea8Slogwang int fcntl(int fd, int cmd, ...)
308a9643ea8Slogwang {
309a9643ea8Slogwang     va_list ap;
310a9643ea8Slogwang     va_start(ap, cmd);
311a9643ea8Slogwang     void* arg = va_arg(ap, void *);
312a9643ea8Slogwang     va_end(ap);
313a9643ea8Slogwang 
314a9643ea8Slogwang     mt_hook_syscall(fcntl);
315a9643ea8Slogwang     MtHookFd* hook_fd = mt_hook_find_fd(fd);
316a02c88d6Slogwang     if (!mt_hook_active() || !hook_fd || !ff_hook_active())
317a9643ea8Slogwang     {
318a02c88d6Slogwang         return mt_real_func(fcntl)(fd, cmd, arg);
319a9643ea8Slogwang     }
320a9643ea8Slogwang 
321a9643ea8Slogwang     if (cmd == F_SETFL)
322a9643ea8Slogwang     {
323a9643ea8Slogwang         va_start(ap, cmd);
324a9643ea8Slogwang         int flags = va_arg(ap, int);
325a9643ea8Slogwang         va_end(ap);
326a9643ea8Slogwang 
327a9643ea8Slogwang         if (flags & O_NONBLOCK)
328a9643ea8Slogwang         {
329a9643ea8Slogwang             hook_fd->sock_flag |= MT_FD_FLG_UNBLOCK;
330a9643ea8Slogwang         }
331a9643ea8Slogwang     }
332a9643ea8Slogwang 
333a9643ea8Slogwang     return ff_hook_fcntl(fd, cmd, arg);
334a9643ea8Slogwang }
335a9643ea8Slogwang 
listen(int sockfd,int backlog)336a9643ea8Slogwang int listen(int sockfd, int backlog)
337a9643ea8Slogwang {
338a9643ea8Slogwang     mt_hook_syscall(listen);
339a02c88d6Slogwang     if (!ff_hook_active())
340a02c88d6Slogwang     {
341a02c88d6Slogwang         return mt_real_func(listen)(sockfd, backlog);
342a02c88d6Slogwang     }
343a02c88d6Slogwang 
344a9643ea8Slogwang     return ff_hook_listen(sockfd, backlog);
345a9643ea8Slogwang }
346a9643ea8Slogwang 
bind(int sockfd,const struct sockaddr * addr,socklen_t addrlen)347a9643ea8Slogwang int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen)
348a9643ea8Slogwang {
349a9643ea8Slogwang     mt_hook_syscall(bind);
350a02c88d6Slogwang     if (!ff_hook_active())
351a02c88d6Slogwang     {
352a02c88d6Slogwang         return mt_real_func(bind)(sockfd, addr, addrlen);
353a02c88d6Slogwang     }
354a02c88d6Slogwang 
355a9643ea8Slogwang     return ff_hook_bind(sockfd, addr, addrlen);
356a9643ea8Slogwang }
357a9643ea8Slogwang 
accept(int fd,struct sockaddr * addr,socklen_t * addrlen)358a9643ea8Slogwang int accept(int fd, struct sockaddr *addr, socklen_t *addrlen)
359a9643ea8Slogwang {
360a9643ea8Slogwang     mt_hook_syscall(accept);
361a02c88d6Slogwang     if (!ff_hook_active())
362a02c88d6Slogwang     {
363a02c88d6Slogwang         return mt_real_func(accept)(fd, addr, addrlen);
364a02c88d6Slogwang     }
365a02c88d6Slogwang 
366a9643ea8Slogwang     return ff_hook_accept(fd, addr, addrlen);
367a9643ea8Slogwang }
368a9643ea8Slogwang 
369a9643ea8Slogwang #ifdef __cplusplus
370a9643ea8Slogwang }
371a9643ea8Slogwang #endif
372