one-acceptor-multi-workers.c 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. /*
  2. *
  3. * @build make examples
  4. * @server bin/one-acceptor-multi-workers 1234
  5. * @client bin/nc 127.0.0.1 1234
  6. * nc 127.0.0.1 1234
  7. * telnet 127.0.0.1 1234
  8. */
  9. #include "hloop.h"
  10. #include "hsocket.h"
  11. #include "hthread.h"
  12. static const char* host = "0.0.0.0";
  13. static int port = 1234;
  14. static int thread_num = 4;
  15. static hloop_t* accept_loop = NULL;
  16. static hloop_t** worker_loops = NULL;
  17. static hloop_t* get_next_loop() {
  18. static int s_cur_index = 0;
  19. if (++s_cur_index >= thread_num) {
  20. s_cur_index = 0;
  21. }
  22. return worker_loops[s_cur_index % thread_num];
  23. }
  24. static void on_close(hio_t* io) {
  25. printf("on_close fd=%d error=%d\n", hio_fd(io), hio_error(io));
  26. }
  27. static void on_recv(hio_t* io, void* buf, int readbytes) {
  28. // echo
  29. hio_write(io, buf, readbytes);
  30. }
  31. static void new_conn_event(hevent_t* ev) {
  32. hloop_t* loop = ev->loop;
  33. hio_t* io = (hio_t*)hevent_userdata(ev);
  34. hio_attach(loop, io);
  35. char localaddrstr[SOCKADDR_STRLEN] = {0};
  36. char peeraddrstr[SOCKADDR_STRLEN] = {0};
  37. printf("tid=%ld connfd=%d [%s] <= [%s]\n",
  38. (long)hv_gettid(),
  39. (int)hio_fd(io),
  40. SOCKADDR_STR(hio_localaddr(io), localaddrstr),
  41. SOCKADDR_STR(hio_peeraddr(io), peeraddrstr));
  42. hio_setcb_close(io, on_close);
  43. hio_setcb_read(io, on_recv);
  44. hio_read(io);
  45. }
  46. static void on_accept(hio_t* io) {
  47. hio_detach(io);
  48. hloop_t* worker_loop = get_next_loop();
  49. hevent_t ev;
  50. memset(&ev, 0, sizeof(ev));
  51. ev.loop = worker_loop;
  52. ev.cb = new_conn_event;
  53. ev.userdata = io;
  54. hloop_post_event(worker_loop, &ev);
  55. }
  56. static HTHREAD_ROUTINE(worker_thread) {
  57. hloop_t* loop = (hloop_t*)userdata;
  58. hloop_run(loop);
  59. return 0;
  60. }
  61. static HTHREAD_ROUTINE(accept_thread) {
  62. hloop_t* loop = (hloop_t*)userdata;
  63. hio_t* listenio = hloop_create_tcp_server(loop, host, port, on_accept);
  64. if (listenio == NULL) {
  65. exit(1);
  66. }
  67. hloop_run(loop);
  68. return 0;
  69. }
  70. int main(int argc, char** argv) {
  71. if (argc < 2) {
  72. printf("Usage: cmd port\n");
  73. return -10;
  74. }
  75. port = atoi(argv[1]);
  76. worker_loops = (hloop_t**)malloc(sizeof(hloop_t*) * thread_num);
  77. for (int i = 0; i < thread_num; ++i) {
  78. worker_loops[i] = hloop_new(HLOOP_FLAG_AUTO_FREE);
  79. hthread_create(worker_thread, worker_loops[i]);
  80. }
  81. accept_loop = hloop_new(HLOOP_FLAG_AUTO_FREE);
  82. accept_thread(accept_loop);
  83. return 0;
  84. }