Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Linux, sockets, non-blocking connect

Tags:

I want to create a non-blocking connect. Like this:

socket.connect(); // returns immediately 

For this, I use another thread, an infinite loop and Linux epoll. Like this(pseudocode):

// in another thread {   create_non_block_socket();   connect();    epoll_create();   epoll_ctl(); // subscribe socket to all events   while (true)   {     epoll_wait(); // wait a small time(~100 ms)     check_socket(); // check on EPOLLOUT event   } } 

If I run a server and then a client, all it works. If I first run a client, wait a some small time, run a server, then the client doesn't connect.

What am I doing wrong? Maybe it can be done differently?

like image 642
herolover Avatar asked Jul 21 '13 07:07

herolover


1 Answers

You should use the following steps for an async connect:

  • create socket with socket(..., SOCK_NONBLOCK, ...)
  • start connection with connect(fd, ...)
  • if return value is neither 0 nor EINPROGRESS, then abort with error
  • wait until fd is signalled as ready for output
  • check status of socket with getsockopt(fd, SOL_SOCKET, SO_ERROR, ...)
  • done

No loops - unless you want to handle EINTR.

If the client is started first, you should see the error ECONNREFUSED in the last step. If this happens, close the socket and start from the beginning.

It is difficult to tell what's wrong with your code, without seeing more details. I suppose, that you do not abort on errors in your check_socket operation.

like image 74
nosid Avatar answered Oct 19 '22 07:10

nosid