Linux TCP服务器无法绑定到close_wait端口

时间:2018-06-26 07:05:24

标签: linux networking tcp server tcp-ip

我试图让tcp服务器绑定到close_wait端口,但是它导致了Errno::EADDRINUSE错误。

我创建了一个监听55555端口的tcp服务器。然后客户端连接到该服务器。进行一些操作后,运行ss -at | grep 55555

# ss -at | grep 55555
LISTEN     0      128                     *:55555                    *:*
FIN-WAIT-2 0      0               127.0.0.1:55555            127.0.0.1:16413
CLOSE-WAIT 0      0               127.0.0.1:16413            127.0.0.1:55555

我试图绑定端口16413,它导致了Errno::EADDRINUSE错误。 但是,如果我连接到ESTAB套接字,则该套接字可以绑定到端口(例如下面的22385)。

# ss -at | grep 55555
LISTEN     0      128                     *:55555                    *:*
ESTAB      0      0               127.0.0.1:22385            127.0.0.1:55555
ESTAB      0      0               127.0.0.1:55555            127.0.0.1:22385 

一些红宝石脚本重现了这个问题。

tcp_server_close_wait.rb

require 'socket'

server = TCPServer.new 55555 # Server bind to port 2000
loop do
  client = server.accept    # Wait for a client to connect
  client.puts "Hello !"
  client.puts "Time is #{Time.now}"
  client.shutdown(Socket::SHUT_WR)
end

tcp_server.rb

require 'socket'

server = TCPServer.new 55555 # Server bind to port 2000
loop do
  client = server.accept    # Wait for a client to connect
  client.puts "Hello !"
  client.puts "Time is #{Time.now}"
  client.close
end

tcp_client.rb

require 'socket'
include Socket::Constants
socket = Socket.new( AF_INET, SOCK_STREAM, 0 )
sockaddr = Socket.pack_sockaddr_in( 55555, '127.0.0.1' )
socket.connect( sockaddr )
res = socket.read
puts res

sleep 10000

** tcp_bind.rb **

require 'socket'

# use Addrinfo
socket = Socket.new(:INET, :STREAM, 0)
socket.bind(Addrinfo.tcp("0.0.0.0", ARGV[0].to_i))

创建close_wait绑定。

  1. 运行ruby tcp_server_close_wait.rb

  2. 运行ruby tcp_client.rb

  3. 运行ss -at | grep 55555来找到客户端端口

  4. 运行ruby tcp_bind.rb $client_port

创建ESTAB绑定。 1.运行ruby tcp_server.rb

  1. 运行ruby tcp_client.rb

  2. 运行ss -at | grep 55555来找到客户端端口

  3. 运行ruby tcp_bind.rb $client_port

1 个答案:

答案 0 :(得分:0)

我使用C程序对Linux 4.4.74-18.20重复了相同的测试。

我得到的结果与OP不同。

禁用SO_REUSEPORT时,两个端口的绑定均失败:

  • 已建立TCP连接(ESTAB)的客户端端口
  • 半开放式TCP连接(CLOSE-WAIT)的客户端端口

为所有套接字启用SO_REUSEPORT时,两个端口的绑定成功。

请参阅socket(7) man page,以获取有关SO_REUSEPORT的更多信息。