Unix下基于SCTP socket的通信:关闭Association

1. 使能Notification的接收

为了查看SCTP的通信过程Association的行为,开启了Association change的通知,代码如下,

evnts.sctp_association_event = 1;

2. 建立新的association

Server端创建新的socket,并绑定到地址一个给定的地址加端口号,这里是127.0.0.1: 18002,

[Server]: Server Start and bind to addr(127.0.0.1:18002)

然后Client端创建新的socket,并向Server端地址(127.0.0.1: 18002)发送第一条消息,
Server端会首先收到一条notification通知有新的association的建立,

[Server]: Assoc change(ID=0x10), COMMUNICATION UP!
[Server]: New ClientAddr{ IP/Port(127.0.0.1:50574) }

同时Client端也会收到新的association建立的通知,这个时候握手完成,连接建立,

[Client]: Assoc change(ID=0xf), COMMUNICATION UP!
[Client]: New ClientAddr{ IP/Port(127.0.0.1:18002) }

NOTE:

  1. 这里可以看到通信的两端都可以在sctp_recvmsg的时候获取对端的IP地址和端口号,可以看出来,Server端的端口号就是配置的18002;而Client端的端口号是动态分配的;
  2. Server端的association ID和Client的ID不是一样的,都是动态分配的,各自有一个association ID。可以用来给各自管理自己应用程序有关的association;

然后Server端才会收到Client发送的消息,

[Server]: SCTP message('hello') received from IP/Port(127.0.0.1:50574) on assoc(0x10) / stream(1)

3. Client主动关闭socket

Client端主动调用close关闭client这一端的socket的时候,Server端会收到通知:

[Server]: Notification received!
[Server]: Assoc change(ID=0x8), SHUTDOWN COMPLETE

依据这个通知,server端可以知道association(ID=0x8)已经不再可用了,association已经断开;

4. Client不主动关闭Socket

这种情况下可以通过发送ABORT消息来关闭association,代码如下,

sctp_sendmsg(sock_op->socket_fd(), message, MAX_BUFFER, 
                    (sockaddr *)&servaddr, sizeof(servaddr), 0, SCTP_ABORT, stream, 0, 0);

client发送完ABORT后,server端会收到association变动的通知,提示连接丢失,

[Server]: Assoc change(ID=0xa), COMMUNICATION LOST

client端同样也会收到asoociation变动的通知,

[Client]: Assoc change(ID=0xd), COMMUNICATION LOST

4.1 Association关闭了,Client继续往server发送消息

association关闭以后,两端的socket还都在,如果Client再次发送数据给Server端,association会再次建立。同时server端也可以接收到消息。

4.1 Association关闭了,Server继续往Client发送消息

这种情况下会报错,

[Server]: Assoc change(ID=0x0), CAN'T START ASSOCIATION

所以实际在应用程序里面,当收到association断开的通知后,就要清除server端保存的这个association的相关信息。

  1. 示例代码
    terminate_association

你可能感兴趣的:(Unix下基于SCTP socket的通信:关闭Association)