diff options
author | Jon Maloy <jon.maloy@ericsson.com> | 2017-10-13 05:04:24 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2017-10-13 11:46:00 -0400 |
commit | 31c82a2d9d51fccbb85cbd2be983eb115225301c (patch) | |
tree | a6681b56dfea5bdae4c6d1459abf6c04131aebec /net/tipc/socket.c | |
parent | 75da2163dbb6af9f2dce1d80056d11d290dd19a5 (diff) |
tipc: add second source address to recvmsg()/recvfrom()
With group communication, it becomes important for a message receiver to
identify not only from which socket (identfied by a node:port tuple) the
message was sent, but also the logical identity (type:instance) of the
sending member.
We fix this by adding a second instance of struct sockaddr_tipc to the
source address area when a message is read. The extra address struct
is filled in with data found in the received message header (type,) and
in the local member representation struct (instance.)
Signed-off-by: Jon Maloy <jon.maloy@ericsson.com>
Acked-by: Ying Xue <ying.xue@windriver.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/tipc/socket.c')
-rw-r--r-- | net/tipc/socket.c | 53 |
1 files changed, 35 insertions, 18 deletions
diff --git a/net/tipc/socket.c b/net/tipc/socket.c index 64bbf9d03629..25ecf1201527 100644 --- a/net/tipc/socket.c +++ b/net/tipc/socket.c | |||
@@ -62,6 +62,11 @@ enum { | |||
62 | TIPC_CONNECTING = TCP_SYN_SENT, | 62 | TIPC_CONNECTING = TCP_SYN_SENT, |
63 | }; | 63 | }; |
64 | 64 | ||
65 | struct sockaddr_pair { | ||
66 | struct sockaddr_tipc sock; | ||
67 | struct sockaddr_tipc member; | ||
68 | }; | ||
69 | |||
65 | /** | 70 | /** |
66 | * struct tipc_sock - TIPC socket structure | 71 | * struct tipc_sock - TIPC socket structure |
67 | * @sk: socket - interacts with 'port' and with user via the socket API | 72 | * @sk: socket - interacts with 'port' and with user via the socket API |
@@ -1222,26 +1227,38 @@ static void tipc_sk_finish_conn(struct tipc_sock *tsk, u32 peer_port, | |||
1222 | } | 1227 | } |
1223 | 1228 | ||
1224 | /** | 1229 | /** |
1225 | * set_orig_addr - capture sender's address for received message | 1230 | * tipc_sk_set_orig_addr - capture sender's address for received message |
1226 | * @m: descriptor for message info | 1231 | * @m: descriptor for message info |
1227 | * @msg: received message header | 1232 | * @hdr: received message header |
1228 | * | 1233 | * |
1229 | * Note: Address is not captured if not requested by receiver. | 1234 | * Note: Address is not captured if not requested by receiver. |
1230 | */ | 1235 | */ |
1231 | static void set_orig_addr(struct msghdr *m, struct tipc_msg *msg) | 1236 | static void tipc_sk_set_orig_addr(struct msghdr *m, struct sk_buff *skb) |
1232 | { | 1237 | { |
1233 | DECLARE_SOCKADDR(struct sockaddr_tipc *, addr, m->msg_name); | 1238 | DECLARE_SOCKADDR(struct sockaddr_pair *, srcaddr, m->msg_name); |
1234 | 1239 | struct tipc_msg *hdr = buf_msg(skb); | |
1235 | if (addr) { | 1240 | |
1236 | addr->family = AF_TIPC; | 1241 | if (!srcaddr) |
1237 | addr->addrtype = TIPC_ADDR_ID; | 1242 | return; |
1238 | memset(&addr->addr, 0, sizeof(addr->addr)); | 1243 | |
1239 | addr->addr.id.ref = msg_origport(msg); | 1244 | srcaddr->sock.family = AF_TIPC; |
1240 | addr->addr.id.node = msg_orignode(msg); | 1245 | srcaddr->sock.addrtype = TIPC_ADDR_ID; |
1241 | addr->addr.name.domain = 0; /* could leave uninitialized */ | 1246 | srcaddr->sock.addr.id.ref = msg_origport(hdr); |
1242 | addr->scope = 0; /* could leave uninitialized */ | 1247 | srcaddr->sock.addr.id.node = msg_orignode(hdr); |
1243 | m->msg_namelen = sizeof(struct sockaddr_tipc); | 1248 | srcaddr->sock.addr.name.domain = 0; |
1244 | } | 1249 | srcaddr->sock.scope = 0; |
1250 | m->msg_namelen = sizeof(struct sockaddr_tipc); | ||
1251 | |||
1252 | if (!msg_in_group(hdr)) | ||
1253 | return; | ||
1254 | |||
1255 | /* Group message users may also want to know sending member's id */ | ||
1256 | srcaddr->member.family = AF_TIPC; | ||
1257 | srcaddr->member.addrtype = TIPC_ADDR_NAME; | ||
1258 | srcaddr->member.addr.name.name.type = msg_nametype(hdr); | ||
1259 | srcaddr->member.addr.name.name.instance = TIPC_SKB_CB(skb)->orig_member; | ||
1260 | srcaddr->member.addr.name.domain = 0; | ||
1261 | m->msg_namelen = sizeof(*srcaddr); | ||
1245 | } | 1262 | } |
1246 | 1263 | ||
1247 | /** | 1264 | /** |
@@ -1432,7 +1449,7 @@ static int tipc_recvmsg(struct socket *sock, struct msghdr *m, | |||
1432 | } while (1); | 1449 | } while (1); |
1433 | 1450 | ||
1434 | /* Collect msg meta data, including error code and rejected data */ | 1451 | /* Collect msg meta data, including error code and rejected data */ |
1435 | set_orig_addr(m, hdr); | 1452 | tipc_sk_set_orig_addr(m, skb); |
1436 | rc = tipc_sk_anc_data_recv(m, hdr, tsk); | 1453 | rc = tipc_sk_anc_data_recv(m, hdr, tsk); |
1437 | if (unlikely(rc)) | 1454 | if (unlikely(rc)) |
1438 | goto exit; | 1455 | goto exit; |
@@ -1526,7 +1543,7 @@ static int tipc_recvstream(struct socket *sock, struct msghdr *m, | |||
1526 | 1543 | ||
1527 | /* Collect msg meta data, incl. error code and rejected data */ | 1544 | /* Collect msg meta data, incl. error code and rejected data */ |
1528 | if (!copied) { | 1545 | if (!copied) { |
1529 | set_orig_addr(m, hdr); | 1546 | tipc_sk_set_orig_addr(m, skb); |
1530 | rc = tipc_sk_anc_data_recv(m, hdr, tsk); | 1547 | rc = tipc_sk_anc_data_recv(m, hdr, tsk); |
1531 | if (rc) | 1548 | if (rc) |
1532 | break; | 1549 | break; |