aboutsummaryrefslogtreecommitdiffstats
path: root/net/tipc/socket.c
diff options
context:
space:
mode:
Diffstat (limited to 'net/tipc/socket.c')
-rw-r--r--net/tipc/socket.c143
1 files changed, 72 insertions, 71 deletions
diff --git a/net/tipc/socket.c b/net/tipc/socket.c
index 348e4ff881ab..df1960384364 100644
--- a/net/tipc/socket.c
+++ b/net/tipc/socket.c
@@ -1248,77 +1248,78 @@ static void wakeupdispatch(struct tipc_port *tport)
1248static int connect(struct socket *sock, struct sockaddr *dest, int destlen, 1248static int connect(struct socket *sock, struct sockaddr *dest, int destlen,
1249 int flags) 1249 int flags)
1250{ 1250{
1251 struct tipc_sock *tsock = tipc_sk(sock->sk); 1251 struct tipc_sock *tsock = tipc_sk(sock->sk);
1252 struct sockaddr_tipc *dst = (struct sockaddr_tipc *)dest; 1252 struct sockaddr_tipc *dst = (struct sockaddr_tipc *)dest;
1253 struct msghdr m = {NULL,}; 1253 struct msghdr m = {NULL,};
1254 struct sk_buff *buf; 1254 struct sk_buff *buf;
1255 struct tipc_msg *msg; 1255 struct tipc_msg *msg;
1256 int res; 1256 int res;
1257 1257
1258 /* For now, TIPC does not allow use of connect() with DGRAM or RDM types */ 1258 /* For now, TIPC does not allow use of connect() with DGRAM/RDM types */
1259 1259
1260 if (sock->state == SS_READY) 1260 if (sock->state == SS_READY)
1261 return -EOPNOTSUPP; 1261 return -EOPNOTSUPP;
1262 1262
1263 /* For now, TIPC does not support the non-blocking form of connect() */ 1263 /* For now, TIPC does not support the non-blocking form of connect() */
1264 1264
1265 if (flags & O_NONBLOCK) 1265 if (flags & O_NONBLOCK)
1266 return -EWOULDBLOCK; 1266 return -EWOULDBLOCK;
1267 1267
1268 /* Issue Posix-compliant error code if socket is in the wrong state */ 1268 /* Issue Posix-compliant error code if socket is in the wrong state */
1269 1269
1270 if (sock->state == SS_LISTENING) 1270 if (sock->state == SS_LISTENING)
1271 return -EOPNOTSUPP; 1271 return -EOPNOTSUPP;
1272 if (sock->state == SS_CONNECTING) 1272 if (sock->state == SS_CONNECTING)
1273 return -EALREADY; 1273 return -EALREADY;
1274 if (sock->state != SS_UNCONNECTED) 1274 if (sock->state != SS_UNCONNECTED)
1275 return -EISCONN; 1275 return -EISCONN;
1276 1276
1277 /* 1277 /*
1278 * Reject connection attempt using multicast address 1278 * Reject connection attempt using multicast address
1279 * 1279 *
1280 * Note: send_msg() validates the rest of the address fields, 1280 * Note: send_msg() validates the rest of the address fields,
1281 * so there's no need to do it here 1281 * so there's no need to do it here
1282 */ 1282 */
1283 1283
1284 if (dst->addrtype == TIPC_ADDR_MCAST) 1284 if (dst->addrtype == TIPC_ADDR_MCAST)
1285 return -EINVAL; 1285 return -EINVAL;
1286 1286
1287 /* Send a 'SYN-' to destination */ 1287 /* Send a 'SYN-' to destination */
1288 1288
1289 m.msg_name = dest; 1289 m.msg_name = dest;
1290 m.msg_namelen = destlen; 1290 m.msg_namelen = destlen;
1291 if ((res = send_msg(NULL, sock, &m, 0)) < 0) { 1291 res = send_msg(NULL, sock, &m, 0);
1292 sock->state = SS_DISCONNECTING; 1292 if (res < 0) {
1293 return res; 1293 sock->state = SS_DISCONNECTING;
1294 } 1294 return res;
1295 1295 }
1296 if (mutex_lock_interruptible(&tsock->lock)) 1296
1297 return -ERESTARTSYS; 1297 if (mutex_lock_interruptible(&tsock->lock))
1298 1298 return -ERESTARTSYS;
1299 /* Wait for destination's 'ACK' response */ 1299
1300 1300 /* Wait for destination's 'ACK' response */
1301 res = wait_event_interruptible_timeout(*sock->sk->sk_sleep, 1301
1302 skb_queue_len(&sock->sk->sk_receive_queue), 1302 res = wait_event_interruptible_timeout(*sock->sk->sk_sleep,
1303 sock->sk->sk_rcvtimeo); 1303 skb_queue_len(&sock->sk->sk_receive_queue),
1304 buf = skb_peek(&sock->sk->sk_receive_queue); 1304 sock->sk->sk_rcvtimeo);
1305 if (res > 0) { 1305 buf = skb_peek(&sock->sk->sk_receive_queue);
1306 msg = buf_msg(buf); 1306 if (res > 0) {
1307 res = auto_connect(sock, tsock, msg); 1307 msg = buf_msg(buf);
1308 if (!res) { 1308 res = auto_connect(sock, tsock, msg);
1309 if (!msg_data_sz(msg)) 1309 if (!res) {
1310 advance_queue(tsock); 1310 if (!msg_data_sz(msg))
1311 } 1311 advance_queue(tsock);
1312 } else { 1312 }
1313 if (res == 0) { 1313 } else {
1314 res = -ETIMEDOUT; 1314 if (res == 0)
1315 } else 1315 res = -ETIMEDOUT;
1316 { /* leave "res" unchanged */ } 1316 else
1317 sock->state = SS_DISCONNECTING; 1317 ; /* leave "res" unchanged */
1318 } 1318 sock->state = SS_DISCONNECTING;
1319 1319 }
1320 mutex_unlock(&tsock->lock); 1320
1321 return res; 1321 mutex_unlock(&tsock->lock);
1322 return res;
1322} 1323}
1323 1324
1324/** 1325/**