-
Notifications
You must be signed in to change notification settings - Fork 277
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
a performance issue for epoll idle #4
Comments
I find the code in Line 1281 in d04dcb5
When there are some epoll events(the nfd >0), it will iterate the _ST_IOQ, a loop link list. The _ST_IOQ is the waiting queue, fd will push in queue in Line 81 in d04dcb5
I don't know why need to iterate all blocked fds? How to fix this? |
It seems to find the thread which is waiting on fd, and switch to the thread. |
Yes it does, but I don't think it's necessary. |
I believe we can store thread info in _st_epoll_data[], and resume the threads without iterating the IOQ. |
I do think there aren't many users now, so nobody cares. |
I'm a heavy user of ST. Please make sure there is a problem, and I'll arrange a patch. |
Yes. It is a issue. But I am afraid that your way doesn't work. There maybe more threads waiting on a fd, and there isn't any reference to the monitoring fds. I think you'd better extend _epoll_fd_data, and record _st_pollq in _epoll_fd_data when inserting it into IOQ. BTW: Maybe EPOLLONESHOT and EPOLLET can be used to get more better performance. |
For performance issue, please write a test example and do some benchmarks, we SHOULD NEVER guess about it. |
FYI: I just cooked a patch for this issue, and will submit a PR after testing. @winlinvip Is there any unit testing case? |
See #5 @lihuiba @winlinvip |
@xiaosuo Thanks for your work and PR, that's great. And can you write a example code, which illustrate the problem? For example, we can write a program, it use 90% CPU or there is a performance report indicates the bottleneck. Then we patch ST with your PR, it use less CPU or the performance report finger it out. Please write a benchmark to prove the PR really fixed the problem. |
@lihuiba reported this issue. I think he will test the PR with his workload. I tested the server in the example directory, and ab reported slight improvements on both QPS and latency. |
@winlinvip @xiaosuo I found the issue in my production server, where there were lots of idle connections. The server consumed several times more CPU resource than expected. I believe the situation could be easily reproduced, and I'll try the patch in a test environment, before launching it to my production server. |
@lihuiba Please test it, thanks~ If you can write a simple test example, it will be better, because it's necessary for any performance issue. |
@lihuiba what is the result? |
@xiaosuo I'm on sth else these days, and I think I'll do the tests next week. |
there were 10,000 idle connections (from a custom tool) in the test, and only 1 active connection from ab. |
@lihuiba Are many coroutines monitoring a FD? Is a coroutine monitoring lots of FDs? |
@xiaosuo No. One coroutine is responsible for a single connection. |
@lihuiba Would you provide the testing codes? |
1. diff of server.c:
2. idle connections:#include <sys/types.h>
#include <sys/socket.h>
#include <fcntl.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <sys/time.h>
#include <string.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
int Socket(const char *host, int clientPort)
{
int sock;
unsigned long inaddr;
struct sockaddr_in ad;
struct hostent *hp;
memset(&ad, 0, sizeof(ad));
ad.sin_family = AF_INET;
inaddr = inet_addr(host);
if (inaddr != INADDR_NONE)
memcpy(&ad.sin_addr, &inaddr, sizeof(inaddr));
else
{
hp = gethostbyname(host);
if (hp == NULL)
return -1;
memcpy(&ad.sin_addr, hp->h_addr, hp->h_length);
}
ad.sin_port = htons(clientPort);
sock = socket(AF_INET, SOCK_STREAM, 0);
if (sock < 0)
return sock;
if (connect(sock, (struct sockaddr *)&ad, sizeof(ad)) < 0)
return -1;
return sock;
}
int main(int argc, char **argv)
{
int idleTime = atoi(argv[4]);
int numberOfConnection = atoi(argv[3]);
int clientPort = atoi(argv[2]);
for (int i = 0; i < numberOfConnection; ++i) {
int s = Socket(argv[1], clientPort);
if (s < 0)
printf("error connect to %d\n", clientPort);
}
sleep(idleTime);
exit(0);
} 3. ab:
|
It is strange. I ran your code, and got an opposite conclusion. ./server -t 1 -p 1 -b 127.0.0.1:8888 -l . -i Could you run your code again? |
I found it quite slow to create large number of connections to the examples/server.c, about 1 second per connection. So I wrote a minimal server myself: #include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <time.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/socket.h>
#include <sys/wait.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <fcntl.h>
#include <signal.h>
#include <pwd.h>
#include "st.h"
void* handle_connection(void* s_)
{
st_netfd_t s = s_;
char buf[512];
st_read(s, buf, sizeof(buf), ST_UTIME_NO_TIMEOUT);
static char resp[] =
"HTTP/1.0 200 OK\r\n"
"Content-type: text/html\r\n"
"Connection: close\r\n"
"\r\n"
"<H2>It worked!</H2>\n";
st_write(s, resp, sizeof(resp) - 1, ST_UTIME_NO_TIMEOUT);
st_netfd_close(s);
}
int main()
{
st_set_eventsys(ST_EVENTSYS_ALT);
st_init();
st_randomize_stacks(1);
int ret, sock = socket(PF_INET, SOCK_STREAM, 0);
struct sockaddr_in serv_addr;
memset(&serv_addr, 0, sizeof(serv_addr));
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = INADDR_ANY;
serv_addr.sin_port = htons(8086);
ret = bind(sock, (struct sockaddr *)&serv_addr, sizeof(serv_addr));
ret = listen(sock, 16);
st_netfd_t server = st_netfd_open_socket(sock);
while(1)
{
st_netfd_t c = st_accept(server, NULL, 0, ST_UTIME_NO_TIMEOUT);
st_thread_create(handle_connection, c, 0, 0);
}
} I created 10,000 idle connections in the test. and I run ab as:
And I got a very promising result:
and
The result showed that the server with pathed ST was 8.3 times (4.679/0.561) faster than before, and consumed much less CPU resource. Greate job! |
Cheers! |
👍 So the patch works well when there are lots of idle connections, right? Could you please write more use scenarios, such as:
Maybe you should let ST to allocate memory in heap instead of stack for large amount of corotines:
|
Have you merged the PR? |
I'm still waiting for the test for active connections 😃 |
I have merge the PR to https://github.com/ossrs/state-threads/tree/features/xiaosuo/epoll |
最近在做perf分析时,发现这个切换会比较占CPU,这个问题值得解决。 |
需要补全utest,覆盖到正常和异常逻辑,才能合并,否则上线后出现偶现的问题,会对质量造成非常大的影响。 utest框架已经完成了:https://github.com/ossrs/state-threads#utest-and-coverage |
I came across a performance issue in epoll mode, when there were thousands concurrent connections. Profiling shows that _st_epoll_dispatch() consumed a lot of CPU.
After reviewing the function, I think I've found the reason: there's a loop that enumerates ALL threads in the I/O queue.
As I'm using one thread per connection model, I believe this loop make epoll mode degraded effectively to select mode.
The text was updated successfully, but these errors were encountered: