| OLD | NEW |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "net/tools/quic/quic_socket_utils.h" | 5 #include "net/tools/quic/quic_socket_utils.h" |
| 6 | 6 |
| 7 #include <errno.h> | 7 #include <errno.h> |
| 8 #include <netinet/in.h> | 8 #include <netinet/in.h> |
| 9 #include <string.h> | 9 #include <string.h> |
| 10 #include <sys/socket.h> | 10 #include <sys/socket.h> |
| (...skipping 136 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 147 sizeof(struct sockaddr_in))); | 147 sizeof(struct sockaddr_in))); |
| 148 } else if (raw_address.ss_family == AF_INET6) { | 148 } else if (raw_address.ss_family == AF_INET6) { |
| 149 CHECK(peer_address->FromSockAddr( | 149 CHECK(peer_address->FromSockAddr( |
| 150 reinterpret_cast<const sockaddr*>(&raw_address), | 150 reinterpret_cast<const sockaddr*>(&raw_address), |
| 151 sizeof(struct sockaddr_in6))); | 151 sizeof(struct sockaddr_in6))); |
| 152 } | 152 } |
| 153 | 153 |
| 154 return bytes_read; | 154 return bytes_read; |
| 155 } | 155 } |
| 156 | 156 |
| 157 void QuicSocketUtils::SetIpInfoInCmsg(const IPAddressNumber& self_address, |
| 158 cmsghdr* cmsg) { |
| 159 if (GetAddressFamily(self_address) == ADDRESS_FAMILY_IPV4) { |
| 160 cmsg->cmsg_len = CMSG_LEN(sizeof(in_pktinfo)); |
| 161 cmsg->cmsg_level = IPPROTO_IP; |
| 162 cmsg->cmsg_type = IP_PKTINFO; |
| 163 in_pktinfo* pktinfo = reinterpret_cast<in_pktinfo*>(CMSG_DATA(cmsg)); |
| 164 memset(pktinfo, 0, sizeof(in_pktinfo)); |
| 165 pktinfo->ipi_ifindex = 0; |
| 166 memcpy(&pktinfo->ipi_spec_dst, &self_address[0], self_address.size()); |
| 167 } else { |
| 168 cmsg->cmsg_len = CMSG_LEN(sizeof(in6_pktinfo)); |
| 169 cmsg->cmsg_level = IPPROTO_IPV6; |
| 170 cmsg->cmsg_type = IPV6_PKTINFO; |
| 171 in6_pktinfo* pktinfo = reinterpret_cast<in6_pktinfo*>(CMSG_DATA(cmsg)); |
| 172 memset(pktinfo, 0, sizeof(in6_pktinfo)); |
| 173 memcpy(&pktinfo->ipi6_addr, &self_address[0], self_address.size()); |
| 174 } |
| 175 } |
| 176 |
| 157 // static | 177 // static |
| 158 WriteResult QuicSocketUtils::WritePacket(int fd, | 178 WriteResult QuicSocketUtils::WritePacket(int fd, |
| 159 const char* buffer, | 179 const char* buffer, |
| 160 size_t buf_len, | 180 size_t buf_len, |
| 161 const IPAddressNumber& self_address, | 181 const IPAddressNumber& self_address, |
| 162 const IPEndPoint& peer_address) { | 182 const IPEndPoint& peer_address) { |
| 163 sockaddr_storage raw_address; | 183 sockaddr_storage raw_address; |
| 164 socklen_t address_len = sizeof(raw_address); | 184 socklen_t address_len = sizeof(raw_address); |
| 165 CHECK(peer_address.ToSockAddr( | 185 CHECK(peer_address.ToSockAddr( |
| 166 reinterpret_cast<struct sockaddr*>(&raw_address), | 186 reinterpret_cast<struct sockaddr*>(&raw_address), |
| 167 &address_len)); | 187 &address_len)); |
| 168 iovec iov = {const_cast<char*>(buffer), buf_len}; | 188 iovec iov = {const_cast<char*>(buffer), buf_len}; |
| 169 | 189 |
| 170 msghdr hdr; | 190 msghdr hdr; |
| 171 hdr.msg_name = &raw_address; | 191 hdr.msg_name = &raw_address; |
| 172 hdr.msg_namelen = address_len; | 192 hdr.msg_namelen = address_len; |
| 173 hdr.msg_iov = &iov; | 193 hdr.msg_iov = &iov; |
| 174 hdr.msg_iovlen = 1; | 194 hdr.msg_iovlen = 1; |
| 175 hdr.msg_flags = 0; | 195 hdr.msg_flags = 0; |
| 176 | 196 |
| 177 const int kSpaceForIpv4 = CMSG_SPACE(sizeof(in_pktinfo)); | 197 const int kSpaceForIpv4 = CMSG_SPACE(sizeof(in_pktinfo)); |
| 178 const int kSpaceForIpv6 = CMSG_SPACE(sizeof(in6_pktinfo)); | 198 const int kSpaceForIpv6 = CMSG_SPACE(sizeof(in6_pktinfo)); |
| 179 // kSpaceForIp should be big enough to hold both IPv4 and IPv6 packet info. | 199 // kSpaceForIp should be big enough to hold both IPv4 and IPv6 packet info. |
| 180 const int kSpaceForIp = | 200 const int kSpaceForIp = |
| 181 (kSpaceForIpv4 < kSpaceForIpv6) ? kSpaceForIpv6 : kSpaceForIpv4; | 201 (kSpaceForIpv4 < kSpaceForIpv6) ? kSpaceForIpv6 : kSpaceForIpv4; |
| 182 char cbuf[kSpaceForIp]; | 202 char cbuf[kSpaceForIp]; |
| 183 if (self_address.empty()) { | 203 if (self_address.empty()) { |
| 184 hdr.msg_control = 0; | 204 hdr.msg_control = 0; |
| 185 hdr.msg_controllen = 0; | 205 hdr.msg_controllen = 0; |
| 186 } else if (GetAddressFamily(self_address) == ADDRESS_FAMILY_IPV4) { | |
| 187 hdr.msg_control = cbuf; | |
| 188 hdr.msg_controllen = kSpaceForIp; | |
| 189 cmsghdr* cmsg = CMSG_FIRSTHDR(&hdr); | |
| 190 | |
| 191 cmsg->cmsg_len = CMSG_LEN(sizeof(in_pktinfo)); | |
| 192 cmsg->cmsg_level = IPPROTO_IP; | |
| 193 cmsg->cmsg_type = IP_PKTINFO; | |
| 194 in_pktinfo* pktinfo = reinterpret_cast<in_pktinfo*>(CMSG_DATA(cmsg)); | |
| 195 memset(pktinfo, 0, sizeof(in_pktinfo)); | |
| 196 pktinfo->ipi_ifindex = 0; | |
| 197 memcpy(&pktinfo->ipi_spec_dst, &self_address[0], self_address.size()); | |
| 198 hdr.msg_controllen = cmsg->cmsg_len; | |
| 199 } else { | 206 } else { |
| 200 hdr.msg_control = cbuf; | 207 hdr.msg_control = cbuf; |
| 201 hdr.msg_controllen = kSpaceForIp; | 208 hdr.msg_controllen = kSpaceForIp; |
| 202 cmsghdr* cmsg = CMSG_FIRSTHDR(&hdr); | 209 cmsghdr *cmsg = CMSG_FIRSTHDR(&hdr); |
| 203 | 210 SetIpInfoInCmsg(self_address, cmsg); |
| 204 cmsg->cmsg_len = CMSG_LEN(sizeof(in6_pktinfo)); | |
| 205 cmsg->cmsg_level = IPPROTO_IPV6; | |
| 206 cmsg->cmsg_type = IPV6_PKTINFO; | |
| 207 in6_pktinfo* pktinfo = reinterpret_cast<in6_pktinfo*>(CMSG_DATA(cmsg)); | |
| 208 memset(pktinfo, 0, sizeof(in6_pktinfo)); | |
| 209 memcpy(&pktinfo->ipi6_addr, &self_address[0], self_address.size()); | |
| 210 hdr.msg_controllen = cmsg->cmsg_len; | 211 hdr.msg_controllen = cmsg->cmsg_len; |
| 211 } | 212 } |
| 212 | 213 |
| 213 int rc = sendmsg(fd, &hdr, 0); | 214 int rc = sendmsg(fd, &hdr, 0); |
| 214 if (rc >= 0) { | 215 if (rc >= 0) { |
| 215 return WriteResult(WRITE_STATUS_OK, rc); | 216 return WriteResult(WRITE_STATUS_OK, rc); |
| 216 } | 217 } |
| 217 return WriteResult((errno == EAGAIN || errno == EWOULDBLOCK) ? | 218 return WriteResult((errno == EAGAIN || errno == EWOULDBLOCK) ? |
| 218 WRITE_STATUS_BLOCKED : WRITE_STATUS_ERROR, errno); | 219 WRITE_STATUS_BLOCKED : WRITE_STATUS_ERROR, errno); |
| 219 } | 220 } |
| 220 | 221 |
| 221 } // namespace tools | 222 } // namespace tools |
| 222 } // namespace net | 223 } // namespace net |
| OLD | NEW |