-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathrfc6458.xml
7103 lines (5884 loc) · 234 KB
/
rfc6458.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
<?xml version='1.0' encoding='utf-8'?>
<!DOCTYPE rfc SYSTEM "rfc2629.dtd" [
<!ENTITY RFC3493 SYSTEM "https://xml2rfc.ietf.org/public/rfc/bibxml/reference.RFC.3493.xml">
<!ENTITY RFC3542 SYSTEM "https://xml2rfc.ietf.org/public/rfc/bibxml/reference.RFC.3542.xml">
<!ENTITY RFC3758 SYSTEM "https://xml2rfc.ietf.org/public/rfc/bibxml/reference.RFC.3758.xml">
<!ENTITY RFC4895 SYSTEM "https://xml2rfc.ietf.org/public/rfc/bibxml/reference.RFC.4895.xml">
<!ENTITY RFC4960 SYSTEM "https://xml2rfc.ietf.org/public/rfc/bibxml/reference.RFC.4960.xml">
<!ENTITY RFC5061 SYSTEM "https://xml2rfc.ietf.org/public/rfc/bibxml/reference.RFC.5061.xml">
<!ENTITY RFC0768 SYSTEM "https://xml2rfc.ietf.org/public/rfc/bibxml/reference.RFC.0768.xml">
<!ENTITY RFC0793 SYSTEM "https://xml2rfc.ietf.org/public/rfc/bibxml/reference.RFC.0793.xml">
<!ENTITY RFC1644 SYSTEM "https://xml2rfc.ietf.org/public/rfc/bibxml/reference.RFC.1644.xml">
<!ENTITY RFC6083 SYSTEM "https://xml2rfc.ietf.org/public/rfc/bibxml/reference.RFC.6083.xml">
<!ENTITY RFC6247 SYSTEM "https://xml2rfc.ietf.org/public/rfc/bibxml/reference.RFC.6247.xml">
]>
<?xml-stylesheet type='text/xsl' href='rfc2629.xslt' ?>
<rfc submissionType="IETF" number="6458" category="info" consensus="yes" ipr="pre5378Trust200902">
<!-- Generated by id2xml 1.4.4 on 2018-11-20T06:57:15Z -->
<?rfc compact="yes"?>
<?rfc text-list-symbols="o*+-"?>
<?rfc subcompact="no"?>
<?rfc sortrefs="no"?>
<?rfc symrefs="yes"?>
<?rfc strict="yes"?>
<?rfc toc="yes"?>
<front>
<title abbrev="SCTP Sockets API">Sockets API Extensions for the Stream Control Transmission Protocol (SCTP)</title>
<author fullname="Randall R. Stewart" initials="R." surname="Stewart">
<organization>Adara Networks</organization>
<address><postal><street>Chapin, SC 29036</street>
<street>USA</street>
</postal>
<email>[email protected]</email>
</address>
</author>
<author fullname="Michael Tuexen" initials="M." surname="Tuexen">
<organization abbrev="Muenster Univ. of Appl. Sciences">Muenster University of Applied Sciences</organization>
<address><postal><street>Stegerwaldstr. 39</street>
<street>48565 Steinfurt</street>
<street>Germany</street>
</postal>
<email>[email protected]</email>
</address>
</author>
<author fullname="Kacheong Poon" initials="K." surname="Poon">
<organization>Oracle Corporation</organization>
<address><email>[email protected]</email>
</address>
</author>
<author fullname="Peter Lei" initials="P." surname="Lei">
<organization>Cisco Systems, Inc.</organization>
<address><postal><street>9501 Technology Blvd.</street>
<street>West Office Center</street>
<street>Rosemont, IL 60018</street>
<street>USA</street>
</postal>
<email>[email protected]</email>
</address>
</author>
<author fullname="Vladislav Yasevich" initials="V." surname="Yasevich">
<organization>HP</organization>
<address><postal><street>110 Spitrook Rd.</street>
<street>Nashua, NH 03062</street>
<street>USA</street>
</postal>
<email>[email protected]</email>
</address>
</author>
<date month="December" year="2011"/>
<abstract><t>
This document describes a mapping of the Stream Control Transmission
Protocol (SCTP) into a sockets API. The benefits of this mapping
include compatibility for TCP applications, access to new SCTP
features, and a consolidated error and event notification scheme.</t>
</abstract>
</front>
<middle>
<section title="Introduction" anchor="section-1"><t>
The sockets API has provided a standard mapping of the Internet
Protocol suite to many operating systems. Both TCP <xref target="RFC0793"/> and UDP
<xref target="RFC0768"/> have benefited from this standard representation and access
method across many diverse platforms. SCTP is a new protocol that
provides many of the characteristics of TCP but also incorporates
semantics more akin to UDP. This document defines a method to map
the existing sockets API for use with SCTP, providing both a base for
access to new features and compatibility so that most existing TCP
applications can be migrated to SCTP with few (if any) changes.</t>
<t>
There are three basic design objectives:</t>
<t><list style="numbers"><t>Maintain consistency with existing sockets APIs: We define a
sockets mapping for SCTP that is consistent with other sockets
API protocol mappings (for instance UDP, TCP, IPv4, and IPv6).</t>
<t>Support a one-to-many style interface: This set of semantics is
similar to that defined for connectionless protocols, such as
UDP. A one-to-many style SCTP socket should be able to control
multiple SCTP associations. This is similar to a UDP socket,
which can communicate with many peer endpoints. Each of these
associations is assigned an association identifier so that an<vspace blankLines="1"/>
application can use the ID to differentiate them. Note that SCTP
is connection-oriented in nature, and it does not support
broadcast or multicast communications, as UDP does.
</t>
<t>Support a one-to-one style interface: This interface supports a
similar semantics as sockets for connection-oriented protocols,
such as TCP. A one-to-one style SCTP socket should only control
one SCTP association. One purpose of defining this interface is
to allow existing applications built on other connection-oriented
protocols to be ported to use SCTP with very little effort.
Developers familiar with these semantics can easily adapt to
SCTP. Another purpose is to make sure that existing mechanisms
in most operating systems that support sockets, such as select(),
should continue to work with this style of socket. Extensions
are added to this mapping to provide mechanisms to exploit new
features of SCTP.</t>
</list>
</t>
<t>
Goals 2 and 3 are not compatible, so this document defines two modes
of mapping, namely the one-to-many style mapping and the one-to-one
style mapping. These two modes share some common data structures and
operations, but will require the use of two different application
programming styles. Note that all new SCTP features can be used with
both styles of socket. The decision on which one to use depends
mainly on the nature of the applications.</t>
<t>
A mechanism is defined to extract an SCTP association from a one-to-
many style socket into a one-to-one style socket.</t>
<t>
Some of the SCTP mechanisms cannot be adequately mapped to an
existing socket interface. In some cases, it is more desirable to
have a new interface instead of using existing socket calls.
Section 9 of this document describes these new interfaces.</t>
<t>
Please note that some elements of the SCTP sockets API are declared
as deprecated. During the evolution of this document, elements of
the API were introduced, implemented, and later on replaced by other
elements. These replaced elements are declared as deprecated, since
they are still available in some implementations and the replacement
functions are not. This applies especially to older versions of
operating systems supporting SCTP. New SCTP socket implementations
must implement at least the non-deprecated elements. Implementations
intending interoperability with older versions of the API should also
include the deprecated functions.</t>
</section>
<section title="Data Types" anchor="section-2"><t>
Whenever possible, Portable Operating System Interface (POSIX) data
types defined in <xref target="IEEE-1003.1-2008"/> are used: uintN_t means an
unsigned integer of exactly N bits (e.g., uint16_t). This document
also assumes the argument data types from POSIX when possible (e.g.,
the final argument to setsockopt() is a socklen_t value). Whenever
buffer sizes are specified, the POSIX size_t data type is used.</t>
</section>
<section title="One-to-Many Style Interface" anchor="section-3"><t>
In the one-to-many style interface, there is a one-to-many
relationship between sockets and associations.</t>
<section title="Basic Operation" anchor="section-3.1"><t>
A typical server in this style uses the following socket calls in
sequence to prepare an endpoint for servicing requests:</t>
<t><list style="symbols"><t>socket()</t>
<t>bind()</t>
<t>listen()</t>
<t>recvmsg()</t>
<t>sendmsg()</t>
<t>close()</t>
</list>
</t>
<t>
A typical client uses the following calls in sequence to set up an
association with a server to request services:</t>
<t><list style="symbols"><t>socket()</t>
<t>sendmsg()</t>
<t>recvmsg()</t>
<t>close()</t>
</list>
</t>
<t>
In this style, by default, all of the associations connected to the
endpoint are represented with a single socket. Each association is
assigned an association identifier (the type is sctp_assoc_t) so that
an application can use it to differentiate among them. In some
implementations, the peer endpoints' addresses can also be used for
this purpose. But this is not required for performance reasons. If
an implementation does not support using addresses to differentiate
between different associations, the sendto() call can only be used to
set up an association implicitly. It cannot be used to send data to
an established association, as the association identifier cannot be
specified.</t>
<t>
Once an association identifier is assigned to an SCTP association,
that identifier will not be reused until the application explicitly
terminates the use of the association. The resources belonging to
that association will not be freed until that happens. This is
similar to the close() operation on a normal socket. The only
exception is when the SCTP_AUTOCLOSE option (<xref target="section-8.1.8"/>) is set.
In this case, after the association is terminated gracefully and
automatically, the association identifier assigned to it can be
reused. All applications using this option should be aware of this
to avoid the possible problem of sending data to an incorrect peer
endpoint.</t>
<t>
If the server or client wishes to branch an existing association off
to a separate socket, it is required to call sctp_peeloff() and to
specify the association identifier. The sctp_peeloff() call will
return a new one-to-one style socket that can then be used with
recv() and send() functions for message passing. See <xref target="section-9.2"/> for
more on branched-off associations.</t>
<t>
Once an association is branched off to a separate socket, it becomes
completely separated from the original socket. All subsequent
control and data operations to that association must be done through
the new socket. For example, the close() operation on the original
socket will not terminate any associations that have been branched
off to a different socket.</t>
<t>
One-to-many style socket calls are discussed in more detail in the
following subsections.</t>
<section title="socket()" anchor="section-3.1.1"><t>
Applications use socket() to create a socket descriptor to represent
an SCTP endpoint.</t>
<t>
The function prototype is</t>
<t><list style="hanging" hangIndent="11"><t hangText="int socket(int domain,">
<vspace blankLines="0"/>
int type,
int protocol);
</t>
</list>
</t>
<t>
and one uses PF_INET or PF_INET6 as the domain, SOCK_SEQPACKET as the
type, and IPPROTO_SCTP as the protocol.</t>
<t>
Here, SOCK_SEQPACKET indicates the creation of a one-to-many style
socket.</t>
<t>
The function returns a socket descriptor, or -1 in case of an error.</t>
<t>
Using the PF_INET domain indicates the creation of an endpoint that
can use only IPv4 addresses, while PF_INET6 creates an endpoint that
can use both IPv6 and IPv4 addresses.</t>
</section>
<section title="bind()" anchor="section-3.1.2"><t>
Applications use bind() to specify with which local address and port
the SCTP endpoint should associate itself.</t>
<t>
An SCTP endpoint can be associated with multiple addresses. To do
this, sctp_bindx() is introduced in <xref target="section-9.1"/> to help applications
do the job of associating multiple addresses. But note that an
endpoint can only be associated with one local port.</t>
<t>
These addresses associated with a socket are the eligible transport
addresses for the endpoint to send and receive data. The endpoint
will also present these addresses to its peers during the association
initialization process; see <xref target="RFC4960"/>.</t>
<t>
After calling bind(), if the endpoint wishes to accept new
associations on the socket, it must call listen() (see
<xref target="section-3.1.3"/>).</t>
<t>
The function prototype of bind() is</t>
<t><list style="hanging" hangIndent="9"><t hangText="int bind(int sd,">
<vspace blankLines="0"/>
struct sockaddr *addr,
socklen_t addrlen);
</t>
</list>
</t>
<t>
and the arguments are</t>
<t><list style="hanging" hangIndent="-1"><t hangText="sd:">
The socket descriptor returned by socket().
<vspace blankLines="0"/>
</t>
<t hangText="addr:">
The address structure (struct sockaddr_in for an IPv4 address
<vspace blankLines="0"/>
or struct sockaddr_in6 for an IPv6 address; see <xref target="RFC3493"/>).
</t>
<t hangText="addrlen:">
The size of the address structure.
<vspace blankLines="0"/>
</t>
</list>
</t>
<t>
bind() returns 0 on success and -1 in case of an error.</t>
<t>
If sd is an IPv4 socket, the address passed must be an IPv4 address.
If the sd is an IPv6 socket, the address passed can either be an IPv4
or an IPv6 address.</t>
<t>
Applications cannot call bind() multiple times to associate multiple
addresses to an endpoint. After the first call to bind(), all
subsequent calls will return an error.</t>
<t>
If the IP address part of addr is specified as a wildcard (INADDR_ANY
for an IPv4 address, or as IN6ADDR_ANY_INIT or in6addr_any for an
IPv6 address), the operating system will associate the endpoint with
an optimal address set of the available interfaces. If the IPv4
sin_port or IPv6 sin6_port is set to 0, the operating system will
choose an ephemeral port for the endpoint.</t>
<t>
If bind() is not called prior to a sendmsg() call that initiates a
new association, the system picks an ephemeral port and will choose
an address set equivalent to binding with a wildcard address. One of
those addresses will be the primary address for the association.
This automatically enables the multi-homing capability of SCTP.</t>
<t>
The completion of this bind() process does not allow the SCTP
endpoint to accept inbound SCTP association requests. Until a
listen() system call, described below, is performed on the socket,
the SCTP endpoint will promptly reject an inbound SCTP INIT request
with an SCTP ABORT.</t>
</section>
<section title="listen()" anchor="section-3.1.3"><t>
By default, a one-to-many style socket does not accept new
association requests. An application uses listen() to mark a socket
as being able to accept new associations.</t>
<t>
The function prototype is</t>
<t><list style="hanging" hangIndent="11"><t hangText="int listen(int sd,">
<vspace blankLines="0"/>
int backlog);
</t>
</list>
</t>
<t>
and the arguments are</t>
<t><list style="hanging" hangIndent="-1"><t hangText="sd:">
The socket descriptor of the endpoint.
<vspace blankLines="0"/>
</t>
<t hangText="backlog:">
If backlog is non-zero, enable listening, else disable
<vspace blankLines="0"/>
listening.
</t>
</list>
</t>
<t>
listen() returns 0 on success and -1 in case of an error.</t>
<t>
Note that one-to-many style socket consumers do not need to call
accept() to retrieve new associations. Calling accept() on a one-to-
many style socket should return EOPNOTSUPP. Rather, new associations
are accepted automatically, and notifications of the new associations
are delivered via recvmsg() with the SCTP_ASSOC_CHANGE event (if
these notifications are enabled). Clients will typically not call
listen(), so that they can be assured that only actively initiated
associations are possible on the socket. Server or peer-to-peer
sockets, on the other hand, will always accept new associations, so a
well-written application using server one-to-many style sockets must
be prepared to handle new associations from unwanted peers.</t>
<t>
Also note that the SCTP_ASSOC_CHANGE event provides the association
identifier for a new association, so if applications wish to use the
association identifier as a parameter to other socket calls, they
should ensure that the SCTP_ASSOC_CHANGE event is enabled.</t>
</section>
<section title="sendmsg() and recvmsg()" anchor="section-3.1.4"><t>
An application uses the sendmsg() and recvmsg() calls to transmit
data to and receive data from its peer.</t>
<t>
The function prototypes are</t>
<t><list style="hanging" hangIndent="16"><t hangText="ssize_t sendmsg(int sd,">
<vspace blankLines="0"/>
const struct msghdr *message,
int flags);
</t>
</list>
</t>
<t>
and</t>
<t><list style="hanging" hangIndent="16"><t hangText="ssize_t recvmsg(int sd,">
<vspace blankLines="0"/>
struct msghdr *message,
int flags);
</t>
</list>
</t>
<t>
using the following arguments:</t>
<t><list style="hanging" hangIndent="-1"><t hangText="sd:">
The socket descriptor of the endpoint.
<vspace blankLines="0"/>
</t>
<t hangText="message:">
Pointer to the msghdr structure that contains a single user
<vspace blankLines="0"/>
message and possibly some ancillary data. See <xref target="section-5"/> for a
complete description of the data structures.
</t>
<t hangText="flags:">
No new flags are defined for SCTP at this level. See
<vspace blankLines="0"/>
<xref target="section-5"/> for SCTP-specific flags used in the msghdr structure.
</t>
</list>
</t>
<t>
sendmsg() returns the number of bytes accepted by the kernel or -1 in
case of an error. recvmsg() returns the number of bytes received or
-1 in case of an error.</t>
<t>
As described in <xref target="section-5"/>, different types of ancillary data can be
sent and received along with user data. When sending, the ancillary
data is used to specify the sent behavior, such as the SCTP stream
number to use. When receiving, the ancillary data is used to
describe the received data, such as the SCTP stream sequence number
of the message.</t>
<t>
When sending user data with sendmsg(), the msg_name field in the
msghdr structure will be filled with one of the transport addresses
of the intended receiver. If there is no existing association
between the sender and the intended receiver, the sender's SCTP stack
will set up a new association and then send the user data (see
<xref target="section-7.5"/> for more on implicit association setup). If sendmsg() is
called with no data and there is no existing association, a new one
will be established. The SCTP_INIT type ancillary data can be used
to change some of the parameters used to set up a new association.
If sendmsg() is called with NULL data, and there is no existing
association but the SCTP_ABORT or SCTP_EOF flags are set as described
in <xref target="section-5.3.4"/>, then -1 is returned and errno is set to EINVAL.
Sending a message using sendmsg() is atomic unless explicit end of
record (EOR) marking is enabled on the socket specified by sd (see
<xref target="section-8.1.26"/>).</t>
<t>
If a peer sends a SHUTDOWN, an SCTP_SHUTDOWN_EVENT notification will
be delivered if that notification has been enabled, and no more data
can be sent to that association. Any attempt to send more data will
cause sendmsg() to return with an ESHUTDOWN error. Note that the
socket is still open for reading at this point, so it is possible to
retrieve notifications.</t>
<t>
When receiving a user message with recvmsg(), the msg_name field in
the msghdr structure will be populated with the source transport
address of the user data. The caller of recvmsg() can use this
address information to determine to which association the received
user message belongs. Note that if SCTP_ASSOC_CHANGE events are
disabled, applications must use the peer transport address provided
in the msg_name field by recvmsg() to perform correlation to an
association, since they will not have the association identifier.</t>
<t>
If all data in a single message has been delivered, MSG_EOR will be
set in the msg_flags field of the msghdr structure (see <xref target="section-5.1"/>).</t>
<t>
If the application does not provide enough buffer space to completely
receive a data message, MSG_EOR will not be set in msg_flags.
Successive reads will consume more of the same message until the
entire message has been delivered, and MSG_EOR will be set.</t>
<t>
If the SCTP stack is running low on buffers, it may partially deliver
a message. In this case, MSG_EOR will not be set, and more calls to
recvmsg() will be necessary to completely consume the message. Only
one message at a time can be partially delivered in any stream. The
socket option SCTP_FRAGMENT_INTERLEAVE controls various aspects of
what interlacing of messages occurs for both the one-to-one and the
one-to-many style sockets. Please consult <xref target="section-8.1.20"/> for further
details on message delivery options.</t>
</section>
<section title="close()" anchor="section-3.1.5"><t>
Applications use close() to perform graceful shutdown (as described
in Section 10.1 of <xref target="RFC4960"/>) on all of the associations currently
represented by a one-to-many style socket.</t>
<t>
The function prototype is</t>
<t>
int close(int sd);</t>
<t>
and the argument is</t>
<t><list style="hanging" hangIndent="5"><t hangText="sd:">
The socket descriptor of the associations to be closed.
<vspace blankLines="0"/>
</t>
</list>
</t>
<t>
0 is returned on success and -1 in case of an error.</t>
<t>
To gracefully shut down a specific association represented by the
one-to-many style socket, an application should use the sendmsg()
call and include the SCTP_EOF flag. A user may optionally terminate
an association non-gracefully by using sendmsg() with the SCTP_ABORT
flag set and possibly passing a user-specified abort code in the data
field. Both flags SCTP_EOF and SCTP_ABORT are passed with ancillary
data (see <xref target="section-5.3.4"/>) in the sendmsg() call.</t>
<t>
If sd in the close() call is a branched-off socket representing only
one association, the shutdown is performed on that association only.</t>
</section>
<section title="connect()" anchor="section-3.1.6"><t>
An application may use the connect() call in the one-to-many style to
initiate an association without sending data.</t>
<t>
The function prototype is</t>
<t><list style="hanging" hangIndent="12"><t hangText="int connect(int sd,">
<vspace blankLines="0"/>
const struct sockaddr *nam,
socklen_t len);
</t>
</list>
</t>
<t>
and the arguments are</t>
<t><list style="hanging" hangIndent="-1"><t hangText="sd:">
The socket descriptor to which a new association is added.
<vspace blankLines="0"/>
</t>
<t hangText="nam:">
The address structure (struct sockaddr_in for an IPv4 address
<vspace blankLines="0"/>
or struct sockaddr_in6 for an IPv6 address; see <xref target="RFC3493"/>).
</t>
<t hangText="len:">
The size of the address.
<vspace blankLines="0"/>
</t>
</list>
</t>
<t>
0 is returned on success and -1 in case of an error.</t>
<t>
Multiple connect() calls can be made on the same socket to create
multiple associations. This is different from the semantics of
connect() on a UDP socket.</t>
<t>
Note that SCTP allows data exchange, similar to T/TCP <xref target="RFC1644"/> (made
Historic by <xref target="RFC6247"/>), during the association setup phase. If an
application wants to do this, it cannot use the connect() call.
Instead, it should use sendto() or sendmsg() to initiate an
association. If it uses sendto() and it wants to change the
initialization behavior, it needs to use the SCTP_INITMSG socket
option before calling sendto(). Or it can use sendmsg() with
SCTP_INIT type ancillary data to initiate an association without
calling setsockopt(). Note that the implicit setup is supported for
the one-to-many style sockets.</t>
<t>
SCTP does not support half close semantics. This means that unlike
T/TCP, MSG_EOF should not be set in the flags parameter when calling
sendto() or sendmsg() when the call is used to initiate a connection.
MSG_EOF is not an acceptable flag with an SCTP socket.</t>
</section>
</section>
<section title="Non-Blocking Mode" anchor="section-3.2"><t>
Some SCTP applications may wish to avoid being blocked when calling a
socket interface function.</t>
<t>
Once a bind() call and/or subsequent sctp_bindx() calls are complete
on a one-to-many style socket, an application may set the
non-blocking option via a fcntl() (such as O_NONBLOCK). After
setting the socket to non-blocking mode, the sendmsg() function
returns immediately. The success or failure of sending the data
message (with possible SCTP_INITMSG ancillary data) will be signaled
by the SCTP_ASSOC_CHANGE event with SCTP_COMM_UP or
SCTP_CANT_START_ASSOC. If user data could not be sent (due to an
SCTP_CANT_START_ASSOC), the sender will also receive an
SCTP_SEND_FAILED_EVENT event. Events can be received by the user
calling recvmsg(). A server (having called listen()) is also
notified of an association-up event via the reception of an
SCTP_ASSOC_CHANGE with SCTP_COMM_UP via the calling of recvmsg() and
possibly the reception of the first data message.</t>
<t>
To shut down the association gracefully, the user must call sendmsg()
with no data and with the SCTP_EOF flag set as described in
<xref target="section-5.3.4"/>. The function returns immediately, and completion of
the graceful shutdown is indicated by an SCTP_ASSOC_CHANGE
notification of type SCTP_SHUTDOWN_COMP (see <xref target="section-6.1.1"/>). Note
that this can also be done using the sctp_sendv() call described in
<xref target="section-9.12"/>.</t>
<t>
It is recommended that an application use caution when using select()
(or poll()) for writing on a one-to-many style socket, because the
interpretation of select() on write is implementation specific.
Generally, a positive return on a select() on write would only
indicate that one of the associations represented by the one-to-many
style socket is writable. An application that writes after the
select() returns may still block, since the association that was
writable is not the destination association of the write call.
Likewise, select() (or poll()) for reading from a one-to-many style
socket will only return an indication that one of the associations
represented by the socket has data to be read.</t>
<t>
An application that wishes to know that a particular association is
ready for reading or writing should either use the one-to-one style
or use the sctp_peeloff() function (see <xref target="section-9.2"/>) to separate the
association of interest from the one-to-many style socket.</t>
<t>
Note that some implementations may have an extended select call, such
as epoll or kqueue, that may escape this limitation and allow a
select on a specific association of a one-to-many style socket, but
this is an implementation-specific detail that a portable application
cannot depend on.</t>
</section>
<section title="Special Considerations" anchor="section-3.3"><t>
The fact that a one-to-many style socket can provide access to many
SCTP associations through a single socket descriptor has important
implications for both application programmers and system programmers
implementing this API. A key issue is how buffer space inside the
sockets layer is managed. Because this implementation detail
directly affects how application programmers must write their code to
ensure correct operation and portability, this section provides some
guidance to both implementers and application programmers.</t>
<t>
An important feature that SCTP shares with TCP is flow control.
Specifically, a sender may not send data faster than the receiver can
consume it.</t>
<t>
For TCP, flow control is typically provided for in the sockets API as
follows. If the reader stops reading, the sender queues messages in
the socket layer until the send socket buffer is completely filled.
This results in a "stalled connection". Further attempts to write to
the socket will block or return the error EAGAIN or EWOULDBLOCK for a
non-blocking socket. At some point, either the connection is closed,
or the receiver begins to read, again freeing space in the output
queue.</t>
<t>
For one-to-one style SCTP sockets (this includes sockets descriptors
that were separated from a one-to-many style socket with
sctp_peeloff()), the behavior is identical. For one-to-many style
SCTP sockets, there are multiple associations for a single socket,
which makes the situation more complicated. If the implementation
uses a single buffer space allocation shared by all associations, a
single stalled association can prevent the further sending of data on
all associations active on a particular one-to-many style socket.</t>
<t>
For a blocking socket, it should be clear that a single stalled
association can block the entire socket. For this reason,
application programmers may want to use non-blocking one-to-many
style sockets. The application should at least be able to send
messages to the non-stalled associations.</t>
<t>
But a non-blocking socket is not sufficient if the API implementer
has chosen a single shared buffer allocation for the socket. A
single stalled association would eventually cause the shared
allocation to fill, and it would become impossible to send even to
non-stalled associations.</t>
<t>
The API implementer can solve this problem by providing each
association with its own allocation of outbound buffer space. Each
association should conceptually have as much buffer space as it would
have if it had its own socket. As a bonus, this simplifies the
implementation of sctp_peeloff().</t>
<t>
To ensure that a given stalled association will not prevent other
non-stalled associations from being writable, application programmers
should either</t>
<t><list style="symbols"><t>demand that the underlying implementation dedicates independent
buffer space reservation to each association (as suggested
above), or</t>
<t>verify that their application-layer protocol does not permit large
amounts of unread data at the receiver (this is true of some
request-response protocols, for example), or</t>
<t>use one-to-one style sockets for association, which may
potentially stall (either from the beginning, or by using
sctp_peeloff() before sending large amounts of data that may cause
a stalled condition).</t>
</list>
</t>
</section>
</section>
<section title="One-to-One Style Interface" anchor="section-4"><t>
The goal of this style is to follow as closely as possible the
current practice of using the sockets interface for a connection-
oriented protocol such as TCP. This style enables existing
applications using connection-oriented protocols to be ported to SCTP
with very little effort.</t>
<t>
One-to-one style sockets can be connected (explicitly or implicitly)
at most once, similar to TCP sockets.</t>
<t>
Note that some new SCTP features and some new SCTP socket options can
only be utilized through the use of sendmsg() and recvmsg() calls;
see <xref target="section-4.1.8"/>.</t>
<section title="Basic Operation" anchor="section-4.1"><t>
A typical one-to-one style server uses the following system call
sequence to prepare an SCTP endpoint for servicing requests:</t>
<t><list style="symbols"><t>socket()</t>
<t>bind()</t>
<t>listen()</t>
<t>accept()</t>
</list>
</t>
<t>
The accept() call blocks until a new association is set up. It
returns with a new socket descriptor. The server then uses the new
socket descriptor to communicate with the client, using recv() and
send() calls to get requests and send back responses.</t>
<t>
Then it calls</t>
<t><list style="symbols"><t>close()</t>
</list>
</t>
<t>
to terminate the association.</t>
<t>
A typical client uses the following system call sequence to set up an
association with a server to request services:</t>
<t><list style="symbols"><t>socket()</t>
<t>connect()</t>
</list>
</t>
<t>
After returning from the connect() call, the client uses send()/
sendmsg() and recv()/recvmsg() calls to send out requests and receive
responses from the server.</t>
<t>
The client calls</t>
<t><list style="symbols"><t>close()</t>
</list>
</t>
<t>
to terminate this association when done.</t>
<section title="socket()" anchor="section-4.1.1"><t>
Applications call socket() to create a socket descriptor to represent
an SCTP endpoint.</t>
<t>
The function prototype is</t>
<t><list style="hanging" hangIndent="11"><t hangText="int socket(int domain,">
<vspace blankLines="0"/>
int type,
int protocol);
</t>
</list>
</t>
<t>
and one uses PF_INET or PF_INET6 as the domain, SOCK_STREAM as the
type, and IPPROTO_SCTP as the protocol.</t>
<t>
Here, SOCK_STREAM indicates the creation of a one-to-one style
socket.</t>
<t>
Using the PF_INET domain indicates the creation of an endpoint that
can use only IPv4 addresses, while PF_INET6 creates an endpoint that
can use both IPv6 and IPv4 addresses.</t>
</section>
<section title="bind()" anchor="section-4.1.2"><t>
Applications use bind() to specify with which local address and port
the SCTP endpoint should associate itself.</t>
<t>
An SCTP endpoint can be associated with multiple addresses. To do
this, sctp_bindx() is introduced in <xref target="section-9.1"/> to help applications
do the job of associating multiple addresses. But note that an
endpoint can only be associated with one local port.</t>
<t>
These addresses associated with a socket are the eligible transport
addresses for the endpoint to send and receive data. The endpoint
will also present these addresses to its peers during the association
initialization process; see <xref target="RFC4960"/>.</t>
<t>
The function prototype of bind() is</t>
<t><list style="hanging" hangIndent="9"><t hangText="int bind(int sd,">
<vspace blankLines="0"/>
struct sockaddr *addr,
socklen_t addrlen);
</t>
</list>
</t>
<t>
and the arguments are</t>
<t><list style="hanging" hangIndent="-1"><t hangText="sd:">
The socket descriptor returned by socket().
<vspace blankLines="0"/>
</t>
<t hangText="addr:">
The address structure (struct sockaddr_in for an IPv4 address
<vspace blankLines="0"/>
or struct sockaddr_in6 for an IPv6 address; see <xref target="RFC3493"/>).
</t>
<t hangText="addrlen:">
The size of the address structure.
<vspace blankLines="0"/>
</t>
</list>
</t>
<t>
If sd is an IPv4 socket, the address passed must be an IPv4 address.
If sd is an IPv6 socket, the address passed can either be an IPv4 or
an IPv6 address.</t>
<t>
Applications cannot call bind() multiple times to associate multiple
addresses to the endpoint. After the first call to bind(), all
subsequent calls will return an error.</t>
<t>
If the IP address part of addr is specified as a wildcard (INADDR_ANY
for an IPv4 address, or as IN6ADDR_ANY_INIT or in6addr_any for an
IPv6 address), the operating system will associate the endpoint with
an optimal address set of the available interfaces. If the IPv4
sin_port or IPv6 sin6_port is set to 0, the operating system will
choose an ephemeral port for the endpoint.</t>
<t>
If bind() is not called prior to the connect() call, the system picks
an ephemeral port and will choose an address set equivalent to
binding with a wildcard address. One of these addresses will be the
primary address for the association. This automatically enables the
multi-homing capability of SCTP.</t>
<t>
The completion of this bind() process does not allow the SCTP
endpoint to accept inbound SCTP association requests. Until a
listen() system call, described below, is performed on the socket,
the SCTP endpoint will promptly reject an inbound SCTP INIT request
with an SCTP ABORT.</t>
</section>
<section title="listen()" anchor="section-4.1.3"><t>
Applications use listen() to allow the SCTP endpoint to accept
inbound associations.</t>