101 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			101 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  *
 | |
|  * @build   make examples
 | |
|  * @server  bin/one-acceptor-multi-workers 1234
 | |
|  * @client  bin/nc 127.0.0.1 1234
 | |
|  *          nc     127.0.0.1 1234
 | |
|  *          telnet 127.0.0.1 1234
 | |
|  */
 | |
| 
 | |
| #include "hloop.h"
 | |
| #include "hsocket.h"
 | |
| #include "hthread.h"
 | |
| 
 | |
| static const char* host = "0.0.0.0";
 | |
| static int port = 1234;
 | |
| static int thread_num = 4;
 | |
| static hloop_t*  accept_loop = NULL;
 | |
| static hloop_t** worker_loops = NULL;
 | |
| 
 | |
| static hloop_t* get_next_loop() {
 | |
|     static int s_cur_index = 0;
 | |
|     if (++s_cur_index >= thread_num) {
 | |
|         s_cur_index = 0;
 | |
|     }
 | |
|     return worker_loops[s_cur_index % thread_num];
 | |
| }
 | |
| 
 | |
| static void on_close(hio_t* io) {
 | |
|     printf("on_close fd=%d error=%d\n", hio_fd(io), hio_error(io));
 | |
| }
 | |
| 
 | |
| static void on_recv(hio_t* io, void* buf, int readbytes) {
 | |
|     // echo
 | |
|     hio_write(io, buf, readbytes);
 | |
| }
 | |
| 
 | |
| static void new_conn_event(hevent_t* ev) {
 | |
|     hloop_t* loop = ev->loop;
 | |
|     hio_t* io = (hio_t*)hevent_userdata(ev);
 | |
|     hio_attach(loop, io);
 | |
| 
 | |
|     char localaddrstr[SOCKADDR_STRLEN] = {0};
 | |
|     char peeraddrstr[SOCKADDR_STRLEN] = {0};
 | |
|     printf("tid=%ld connfd=%d [%s] <= [%s]\n",
 | |
|             (long)hv_gettid(),
 | |
|             (int)hio_fd(io),
 | |
|             SOCKADDR_STR(hio_localaddr(io), localaddrstr),
 | |
|             SOCKADDR_STR(hio_peeraddr(io), peeraddrstr));
 | |
| 
 | |
|     hio_setcb_close(io, on_close);
 | |
|     hio_setcb_read(io, on_recv);
 | |
|     hio_read(io);
 | |
| }
 | |
| 
 | |
| static void on_accept(hio_t* io) {
 | |
|     hio_detach(io);
 | |
| 
 | |
|     hloop_t* worker_loop = get_next_loop();
 | |
|     hevent_t ev;
 | |
|     memset(&ev, 0, sizeof(ev));
 | |
|     ev.loop = worker_loop;
 | |
|     ev.cb = new_conn_event;
 | |
|     ev.userdata = io;
 | |
|     hloop_post_event(worker_loop, &ev);
 | |
| }
 | |
| 
 | |
| static HTHREAD_ROUTINE(worker_thread) {
 | |
|     hloop_t* loop = (hloop_t*)userdata;
 | |
|     hloop_run(loop);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static HTHREAD_ROUTINE(accept_thread) {
 | |
|     hloop_t* loop = (hloop_t*)userdata;
 | |
|     hio_t* listenio = hloop_create_tcp_server(loop, host, port, on_accept);
 | |
|     if (listenio == NULL) {
 | |
|         exit(1);
 | |
|     }
 | |
|     hloop_run(loop);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| int main(int argc, char** argv) {
 | |
|     if (argc < 2) {
 | |
|         printf("Usage: cmd port\n");
 | |
|         return -10;
 | |
|     }
 | |
|     port = atoi(argv[1]);
 | |
| 
 | |
|     worker_loops = (hloop_t**)malloc(sizeof(hloop_t*) * thread_num);
 | |
|     for (int i = 0; i < thread_num; ++i) {
 | |
|         worker_loops[i] = hloop_new(HLOOP_FLAG_AUTO_FREE);
 | |
|         hthread_create(worker_thread, worker_loops[i]);
 | |
|     }
 | |
| 
 | |
|     accept_loop = hloop_new(HLOOP_FLAG_AUTO_FREE);
 | |
|     accept_thread(accept_loop);
 | |
| 
 | |
|     return 0;
 | |
| }
 | 
