blob: 4c90b0757f4bb1a337de44cfa974a356fef031e6 [file] [log] [blame]
Austin Schuh8d0a2852019-12-28 22:54:28 -08001/* SCTP kernel Implementation: User API extensions.
2 *
3 * sendmsg.c
4 *
5 * Distributed under the terms of the LGPL v2.1 as described in
6 * http://www.gnu.org/copyleft/lesser.txt
7 *
8 * This file is part of the user library that offers support for the
9 * SCTP kernel Implementation. The main purpose of this
10 * code is to provide the SCTP Socket API mappings for user
11 * application to interface with the SCTP in kernel.
12 *
13 * This implementation is based on the Socket API Extensions for SCTP
14 * defined in <draft-ietf-tsvwg-sctpsocket-10.txt>
15 *
16 * Copyright (c) 2003 Intel Corp.
17 *
18 * Written or modified by:
19 * Ardelle Fan <ardelle.fan@intel.com>
20 */
21
22#include <errno.h>
23#include <string.h>
24#include <stdlib.h>
25#include <netinet/in.h>
26#include <sys/socket.h> /* struct sockaddr_storage, setsockopt() */
27#include <netinet/sctp.h>
28
29/* This library function assists the user with the advanced features
30 * of SCTP. This is a new SCTP API described in the section 8.7 of the
31 * Sockets API Extensions for SCTP. This is implemented using the
32 * sendmsg() interface.
33 */
34int
35sctp_sendmsg(int s, const void *msg, size_t len, struct sockaddr *to,
36 socklen_t tolen, uint32_t ppid, uint32_t flags,
37 uint16_t stream_no, uint32_t timetolive, uint32_t context)
38{
39 struct msghdr outmsg;
40 struct iovec iov;
41 char outcmsg[CMSG_SPACE(sizeof(struct sctp_sndrcvinfo))];
42 struct cmsghdr *cmsg;
43 struct sctp_sndrcvinfo *sinfo;
44
45 outmsg.msg_name = to;
46 outmsg.msg_namelen = tolen;
47 outmsg.msg_iov = &iov;
48 iov.iov_base = (void *)msg;
49 iov.iov_len = len;
50 outmsg.msg_iovlen = 1;
51
52 outmsg.msg_control = outcmsg;
53 outmsg.msg_controllen = sizeof(outcmsg);
54 outmsg.msg_flags = 0;
55
56 cmsg = CMSG_FIRSTHDR(&outmsg);
57 cmsg->cmsg_level = IPPROTO_SCTP;
58 cmsg->cmsg_type = SCTP_SNDRCV;
59 cmsg->cmsg_len = CMSG_LEN(sizeof(struct sctp_sndrcvinfo));
60
61 outmsg.msg_controllen = cmsg->cmsg_len;
62 sinfo = (struct sctp_sndrcvinfo *)CMSG_DATA(cmsg);
63 memset(sinfo, 0, sizeof(struct sctp_sndrcvinfo));
64 sinfo->sinfo_ppid = ppid;
65 sinfo->sinfo_flags = flags;
66 sinfo->sinfo_stream = stream_no;
67 sinfo->sinfo_timetolive = timetolive;
68 sinfo->sinfo_context = context;
69
70 return sendmsg(s, &outmsg, 0);
71}
72
73/* This library function assist the user with sending a message without
74 * dealing directly with the CMSG header.
75 */
76int
77sctp_send(int s, const void *msg, size_t len,
78 const struct sctp_sndrcvinfo *sinfo, int flags)
79{
80 struct msghdr outmsg;
81 struct iovec iov;
82 char outcmsg[CMSG_SPACE(sizeof(struct sctp_sndrcvinfo))];
83
84 outmsg.msg_name = NULL;
85 outmsg.msg_namelen = 0;
86 outmsg.msg_iov = &iov;
87 iov.iov_base = (void *)msg;
88 iov.iov_len = len;
89 outmsg.msg_iovlen = 1;
90 outmsg.msg_control = NULL;
91 outmsg.msg_controllen = 0;
92
93 if (sinfo) {
94 struct cmsghdr *cmsg;
95
96 outmsg.msg_control = outcmsg;
97 outmsg.msg_controllen = sizeof(outcmsg);
98 outmsg.msg_flags = 0;
99
100 cmsg = CMSG_FIRSTHDR(&outmsg);
101 cmsg->cmsg_level = IPPROTO_SCTP;
102 cmsg->cmsg_type = SCTP_SNDRCV;
103 cmsg->cmsg_len = CMSG_LEN(sizeof(struct sctp_sndrcvinfo));
104
105 outmsg.msg_controllen = cmsg->cmsg_len;
106 memcpy(CMSG_DATA(cmsg), sinfo, sizeof(struct sctp_sndrcvinfo));
107 }
108
109 return sendmsg(s, &outmsg, flags);
110}
111
112#ifdef HAVE_SCTP_SENDV
113static struct cmsghdr *sctp_sendv_store_cmsg(struct cmsghdr *cmsg, int *cmsglen,
114 int type, int len, void *data)
115{
116 cmsg->cmsg_level = IPPROTO_SCTP;
117 cmsg->cmsg_type = type;
118 cmsg->cmsg_len = CMSG_LEN(len);
119 memcpy(CMSG_DATA(cmsg), data, len);
120
121 *cmsglen += CMSG_SPACE(len);
122
123 return (struct cmsghdr *)((char *)cmsg + CMSG_SPACE(len));
124}
125
126int sctp_sendv(int s, const struct iovec *iov, int iovcnt,
127 struct sockaddr *addrs, int addrcnt, void *info,
128 socklen_t infolen, unsigned int infotype, int flags)
129{
130 char _cmsg[CMSG_SPACE(sizeof(struct sctp_sendv_spa))];
131 struct cmsghdr *cmsg = (struct cmsghdr *)_cmsg;
132 struct sockaddr *addr;
133 struct msghdr outmsg;
134 int len, cmsglen = 0;
135 int err, type, i;
136 char *addrbuf;
137
138 /* set msg_iov, msg_iovlen, msg_flags */
139 memset(&outmsg, 0x00, sizeof(outmsg));
140 outmsg.msg_iov = (struct iovec *)iov;
141 outmsg.msg_iovlen = iovcnt;
142 outmsg.msg_flags = flags;
143
144 /* set msg_name and msg_namelen */
145 if (addrs && addrcnt) {
146 outmsg.msg_name = addrs;
147 if (addrs->sa_family == AF_INET)
148 outmsg.msg_namelen = sizeof(struct sockaddr_in);
149 else if (addrs->sa_family == AF_INET6)
150 outmsg.msg_namelen = sizeof(struct sockaddr_in6);
151 else
152 return -EINVAL;
153 addrcnt -= 1;
154 addrbuf = (char *)addrs;
155 addrs = (struct sockaddr *)(addrbuf + outmsg.msg_namelen);
156 }
157
158 /* alloc memory only when it's multi-address */
159 if (addrcnt) {
160 len = CMSG_SPACE(sizeof(struct sockaddr_in6)) * addrcnt;
161 cmsg = malloc(sizeof(_cmsg) + len);
162 if (!cmsg)
163 return -ENOMEM;
164 }
165
166 outmsg.msg_control = cmsg;
167
168 /* add cmsg info for addr info */
169 for (i = 0, addrbuf = (char *)addrs; i < addrcnt; i++) {
170 void *ainfo;
171
172 addr = (struct sockaddr *)addrbuf;
173 if (addr->sa_family == AF_INET) {
174 struct sockaddr_in *a = (struct sockaddr_in *)addrbuf;
175
176 len = sizeof(struct in_addr);
177 type = SCTP_DSTADDRV4;
178 ainfo = &a->sin_addr;
179 addrbuf += sizeof(*a);
180 } else if (addr->sa_family == AF_INET6) {
181 struct sockaddr_in6 *a = (struct sockaddr_in6 *)addrbuf;
182
183 len = sizeof(struct in6_addr);
184 type = SCTP_DSTADDRV6;
185 ainfo = &a->sin6_addr;
186 addrbuf += sizeof(*a);
187 } else {
188 free(outmsg.msg_control);
189 return -EINVAL;
190 }
191
192 cmsg = sctp_sendv_store_cmsg(cmsg, &cmsglen, type, len, ainfo);
193 }
194 /* add cmsg info for addr info for snd/pr/auth info */
195 if (infotype == SCTP_SENDV_SPA) {
196 struct sctp_sendv_spa *spa = info;
197
198 if (spa->sendv_flags & SCTP_SEND_SNDINFO_VALID) {
199 type = SCTP_SNDINFO;
200 len = sizeof(struct sctp_sndinfo);
201 cmsg = sctp_sendv_store_cmsg(cmsg, &cmsglen, type, len,
202 &spa->sendv_sndinfo);
203 }
204 if (spa->sendv_flags & SCTP_SEND_PRINFO_VALID) {
205 type = SCTP_PRINFO;
206 len = sizeof(struct sctp_prinfo);
207 cmsg = sctp_sendv_store_cmsg(cmsg, &cmsglen, type, len,
208 &spa->sendv_prinfo);
209 }
210 if (spa->sendv_flags & SCTP_SEND_AUTHINFO_VALID) {
211 type = SCTP_AUTHINFO;
212 len = sizeof(struct sctp_authinfo);
213 sctp_sendv_store_cmsg(cmsg, &cmsglen, type, len,
214 &spa->sendv_authinfo);
215 }
216 } else if (infotype == SCTP_SENDV_SNDINFO) {
217 type = SCTP_SNDINFO;
218 len = sizeof(struct sctp_sndinfo);
219 sctp_sendv_store_cmsg(cmsg, &cmsglen, type, len, info);
220 } else if (infotype == SCTP_SENDV_PRINFO) {
221 type = SCTP_PRINFO;
222 len = sizeof(struct sctp_prinfo);
223 sctp_sendv_store_cmsg(cmsg, &cmsglen, type, len, info);
224 } else if (infotype == SCTP_SENDV_AUTHINFO) {
225 type = SCTP_AUTHINFO;
226 len = sizeof(struct sctp_authinfo);
227 sctp_sendv_store_cmsg(cmsg, &cmsglen, type, len, info);
228 }
229
230 outmsg.msg_controllen = cmsglen;
231
232 err = sendmsg(s, &outmsg, 0);
233
234 if (outmsg.msg_control != _cmsg)
235 free(outmsg.msg_control);
236
237 return err;
238}
239#endif