mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
socket: avoid arg parsing in rsock_s_accept_nonblock
* ext/socket/init.c (rsock_s_accept_nonblock): avoid parsing args [ruby-core:71439] [Feature #11339] * ext/socket/rubysocket.h: adjust prototype * ext/socket/socket.c (sock_accept_nonblock): make private * ext/socket/tcpserver.c (tcp_accept_nonblock): ditto * ext/socket/unixserver.c (unix_accept_nonblock): ditto * ext/socket/lib/socket.rb (Socket#accept_nonblock): implement as wrapper, move RDoc (TCPServer#accept_nonblock): ditto (UNIXServer#accept_nonblock): ditto target 0: a (ruby 2.3.0dev (2015-11-12 trunk 52550) [x86_64-linux]) target 1: b (ruby 2.3.0dev (2015-11-12 avoid-kwarg-capi 52550) [x86_64-linux] ----------------------------------------------------------- accept_nonblock require 'tempfile' require 'socket' require 'io/wait' nr = 500000 Tempfile.create(%w(accept_nonblock .sock)) do |tmp| path = tmp.path File.unlink(path) s = UNIXServer.new(path) addr = Socket.sockaddr_un(path).freeze nr.times do s.accept_nonblock(exception: false) c = UNIXSocket.new(path) s.wait_readable s.accept_nonblock(exception: false).close c.close end end ----------------------------------------------------------- raw data: [["accept_nonblock", [[4.807877402752638, 4.930681671947241, 4.738454818725586, 4.69268161803484, 4.684675686061382], [4.253904823213816, 4.255124930292368, 4.295955188572407, 4.248479191213846, 4.213303029537201]]]] Elapsed time: 45.123040065 (sec) ----------------------------------------------------------- benchmark results: minimum results in each 5 measurements. Execution time (sec) name a b accept_nonblock 4.685 4.213 Speedup ratio: compare with the result of `a' (greater is better) name b accept_nonblock 1.112 git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@52601 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
bb6dfab2a8
commit
bee5b49aec
7 changed files with 179 additions and 148 deletions
|
|
@ -64,53 +64,16 @@ tcp_accept(VALUE sock)
|
|||
return rsock_s_accept(rb_cTCPSocket, fptr->fd, &from.addr, &fromlen);
|
||||
}
|
||||
|
||||
/*
|
||||
* call-seq:
|
||||
* tcpserver.accept_nonblock([options]) => tcpsocket
|
||||
*
|
||||
* Accepts an incoming connection using accept(2) after
|
||||
* O_NONBLOCK is set for the underlying file descriptor.
|
||||
* It returns an accepted TCPSocket for the incoming connection.
|
||||
*
|
||||
* === Example
|
||||
* require 'socket'
|
||||
* serv = TCPServer.new(2202)
|
||||
* begin # emulate blocking accept
|
||||
* sock = serv.accept_nonblock
|
||||
* rescue IO::WaitReadable, Errno::EINTR
|
||||
* IO.select([serv])
|
||||
* retry
|
||||
* end
|
||||
* # sock is an accepted socket.
|
||||
*
|
||||
* Refer to Socket#accept for the exceptions that may be thrown if the call
|
||||
* to TCPServer#accept_nonblock fails.
|
||||
*
|
||||
* TCPServer#accept_nonblock may raise any error corresponding to accept(2) failure,
|
||||
* including Errno::EWOULDBLOCK.
|
||||
*
|
||||
* If the exception is Errno::EWOULDBLOCK, Errno::EAGAIN, Errno::ECONNABORTED, Errno::EPROTO,
|
||||
* it is extended by IO::WaitReadable.
|
||||
* So IO::WaitReadable can be used to rescue the exceptions for retrying accept_nonblock.
|
||||
*
|
||||
* By specifying `exception: false`, the options hash allows you to indicate
|
||||
* that accept_nonblock should not raise an IO::WaitReadable exception, but
|
||||
* return the symbol :wait_readable instead.
|
||||
*
|
||||
* === See
|
||||
* * TCPServer#accept
|
||||
* * Socket#accept
|
||||
*/
|
||||
/* :nodoc: */
|
||||
static VALUE
|
||||
tcp_accept_nonblock(int argc, VALUE *argv, VALUE sock)
|
||||
tcp_accept_nonblock(VALUE sock, VALUE ex)
|
||||
{
|
||||
rb_io_t *fptr;
|
||||
union_sockaddr from;
|
||||
socklen_t fromlen;
|
||||
socklen_t len = (socklen_t)sizeof(from);
|
||||
|
||||
GetOpenFile(sock, fptr);
|
||||
fromlen = (socklen_t)sizeof(from);
|
||||
return rsock_s_accept_nonblock(argc, argv, rb_cTCPSocket, fptr, &from.addr, &fromlen);
|
||||
return rsock_s_accept_nonblock(rb_cTCPSocket, ex, fptr, &from.addr, &len);
|
||||
}
|
||||
|
||||
/*
|
||||
|
|
@ -175,7 +138,8 @@ rsock_init_tcpserver(void)
|
|||
*/
|
||||
rb_cTCPServer = rb_define_class("TCPServer", rb_cTCPSocket);
|
||||
rb_define_method(rb_cTCPServer, "accept", tcp_accept, 0);
|
||||
rb_define_method(rb_cTCPServer, "accept_nonblock", tcp_accept_nonblock, -1);
|
||||
rb_define_private_method(rb_cTCPServer,
|
||||
"__accept_nonblock", tcp_accept_nonblock, 1);
|
||||
rb_define_method(rb_cTCPServer, "sysaccept", tcp_sysaccept, 0);
|
||||
rb_define_method(rb_cTCPServer, "initialize", tcp_svr_init, -1);
|
||||
rb_define_method(rb_cTCPServer, "listen", rsock_sock_listen, 1); /* in socket.c */
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue