Chromium Code Reviews| 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/base/host_resolver_impl.h" | 5 #include "net/base/host_resolver_impl.h" |
| 6 | 6 |
| 7 #if defined(OS_WIN) | 7 #if defined(OS_WIN) |
| 8 #include <Winsock2.h> | 8 #include <Winsock2.h> |
| 9 #elif defined(OS_POSIX) | 9 #elif defined(OS_POSIX) |
| 10 #include <netdb.h> | 10 #include <netdb.h> |
| (...skipping 21 matching lines...) Expand all Loading... | |
| 32 #include "base/values.h" | 32 #include "base/values.h" |
| 33 #include "net/base/address_family.h" | 33 #include "net/base/address_family.h" |
| 34 #include "net/base/address_list.h" | 34 #include "net/base/address_list.h" |
| 35 #include "net/base/address_list_net_log_param.h" | 35 #include "net/base/address_list_net_log_param.h" |
| 36 #include "net/base/dns_reloader.h" | 36 #include "net/base/dns_reloader.h" |
| 37 #include "net/base/host_port_pair.h" | 37 #include "net/base/host_port_pair.h" |
| 38 #include "net/base/host_resolver_proc.h" | 38 #include "net/base/host_resolver_proc.h" |
| 39 #include "net/base/net_errors.h" | 39 #include "net/base/net_errors.h" |
| 40 #include "net/base/net_log.h" | 40 #include "net/base/net_log.h" |
| 41 #include "net/base/net_util.h" | 41 #include "net/base/net_util.h" |
| 42 #include "net/dns/address_sorter.h" | |
| 42 #include "net/dns/dns_client.h" | 43 #include "net/dns/dns_client.h" |
| 43 #include "net/dns/dns_config_service.h" | 44 #include "net/dns/dns_config_service.h" |
| 44 #include "net/dns/dns_protocol.h" | 45 #include "net/dns/dns_protocol.h" |
| 45 #include "net/dns/dns_response.h" | 46 #include "net/dns/dns_response.h" |
| 46 #include "net/dns/dns_transaction.h" | 47 #include "net/dns/dns_transaction.h" |
| 47 | 48 |
| 48 #if defined(OS_WIN) | 49 #if defined(OS_WIN) |
| 49 #include "net/base/winsock_init.h" | 50 #include "net/base/winsock_init.h" |
| 50 #endif | 51 #endif |
| 51 | 52 |
| (...skipping 994 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1046 //----------------------------------------------------------------------------- | 1047 //----------------------------------------------------------------------------- |
| 1047 | 1048 |
| 1048 // Resolves the hostname using DnsTransaction. | 1049 // Resolves the hostname using DnsTransaction. |
| 1049 // TODO(szym): This could be moved to separate source file as well. | 1050 // TODO(szym): This could be moved to separate source file as well. |
| 1050 class HostResolverImpl::DnsTask { | 1051 class HostResolverImpl::DnsTask { |
| 1051 public: | 1052 public: |
| 1052 typedef base::Callback<void(int net_error, | 1053 typedef base::Callback<void(int net_error, |
| 1053 const AddressList& addr_list, | 1054 const AddressList& addr_list, |
| 1054 base::TimeDelta ttl)> Callback; | 1055 base::TimeDelta ttl)> Callback; |
| 1055 | 1056 |
| 1056 DnsTask(DnsTransactionFactory* factory, | 1057 DnsTask(DnsClient* client, |
| 1057 const Key& key, | 1058 const Key& key, |
| 1058 const Callback& callback, | 1059 const Callback& callback, |
| 1059 const BoundNetLog& job_net_log) | 1060 const BoundNetLog& job_net_log) |
| 1060 : callback_(callback), net_log_(job_net_log) { | 1061 : client_(client), |
| 1061 DCHECK(factory); | 1062 family_(key.address_family), |
| 1063 callback_(callback), | |
| 1064 net_log_(job_net_log) { | |
| 1065 DCHECK(client); | |
| 1062 DCHECK(!callback.is_null()); | 1066 DCHECK(!callback.is_null()); |
| 1063 | 1067 |
| 1064 // For now we treat ADDRESS_FAMILY_UNSPEC as if it was IPV4. | 1068 uint16 qtype = (family_ == ADDRESS_FAMILY_IPV4) ? |
| 1065 uint16 qtype = (key.address_family == ADDRESS_FAMILY_IPV6) | 1069 dns_protocol::kTypeA : |
| 1066 ? dns_protocol::kTypeAAAA | 1070 dns_protocol::kTypeAAAA; // If unspecified, do IPv6 first. |
| 1067 : dns_protocol::kTypeA; | 1071 transaction_ = client_->GetTransactionFactory()->CreateTransaction( |
| 1068 // TODO(szym): Implement "happy eyeballs". | |
| 1069 transaction_ = factory->CreateTransaction( | |
| 1070 key.hostname, | 1072 key.hostname, |
| 1071 qtype, | 1073 qtype, |
| 1072 base::Bind(&DnsTask::OnTransactionComplete, base::Unretained(this), | 1074 base::Bind(&DnsTask::OnTransactionComplete, base::Unretained(this), |
| 1073 base::TimeTicks::Now()), | 1075 true /* first_query */, |
| 1076 base::TimeTicks::Now()), | |
| 1074 net_log_); | 1077 net_log_); |
| 1075 DCHECK(transaction_.get()); | |
| 1076 } | 1078 } |
| 1077 | 1079 |
| 1078 int Start() { | 1080 int Start() { |
| 1079 net_log_.BeginEvent(NetLog::TYPE_HOST_RESOLVER_IMPL_DNS_TASK, NULL); | 1081 net_log_.BeginEvent(NetLog::TYPE_HOST_RESOLVER_IMPL_DNS_TASK, NULL); |
| 1080 return transaction_->Start(); | 1082 return transaction_->Start(); |
| 1081 } | 1083 } |
| 1082 | 1084 |
| 1083 void OnTransactionComplete(const base::TimeTicks& start_time, | 1085 private: |
| 1086 void OnTransactionComplete(bool first_query, | |
| 1087 const base::TimeTicks& start_time, | |
| 1084 DnsTransaction* transaction, | 1088 DnsTransaction* transaction, |
| 1085 int net_error, | 1089 int net_error, |
| 1086 const DnsResponse* response) { | 1090 const DnsResponse* response) { |
| 1087 DCHECK(transaction); | 1091 DCHECK(transaction); |
| 1088 // Run |callback_| last since the owning Job will then delete this DnsTask. | 1092 // Run |callback_| last since the owning Job will then delete this DnsTask. |
| 1089 DnsResponse::Result result = DnsResponse::DNS_SUCCESS; | 1093 if (net_error != OK) { |
| 1090 if (net_error == OK) { | 1094 DNS_HISTOGRAM("AsyncDNS.TransactionFailure", |
| 1091 CHECK(response); | |
| 1092 DNS_HISTOGRAM("AsyncDNS.TransactionSuccess", | |
| 1093 base::TimeTicks::Now() - start_time); | 1095 base::TimeTicks::Now() - start_time); |
| 1094 AddressList addr_list; | 1096 OnFailure(net_error, DnsResponse::DNS_SUCCESS); |
|
mmenke
2012/06/07 18:32:34
These DNS_SUCCESSes everywhere caused me a fair bi
| |
| 1095 base::TimeDelta ttl; | 1097 return; |
| 1096 result = response->ParseToAddressList(&addr_list, &ttl); | 1098 } |
| 1097 UMA_HISTOGRAM_ENUMERATION("AsyncDNS.ParseToAddressList", | 1099 |
| 1098 result, | 1100 CHECK(response); |
| 1099 DnsResponse::DNS_PARSE_RESULT_MAX); | 1101 DNS_HISTOGRAM("AsyncDNS.TransactionSuccess", |
| 1100 if (result == DnsResponse::DNS_SUCCESS) { | 1102 base::TimeTicks::Now() - start_time); |
| 1101 net_log_.EndEvent(NetLog::TYPE_HOST_RESOLVER_IMPL_DNS_TASK, | 1103 AddressList addr_list; |
| 1102 new AddressListNetLogParam(addr_list)); | 1104 base::TimeDelta ttl; |
| 1103 callback_.Run(net_error, addr_list, ttl); | 1105 DnsResponse::Result result = response->ParseToAddressList(&addr_list, &ttl); |
| 1106 UMA_HISTOGRAM_ENUMERATION("AsyncDNS.ParseToAddressList", | |
| 1107 result, | |
| 1108 DnsResponse::DNS_PARSE_RESULT_MAX); | |
| 1109 if (result != DnsResponse::DNS_SUCCESS) { | |
| 1110 // Fail even if the other query succeeds. | |
| 1111 OnFailure(ERR_DNS_MALFORMED_RESPONSE, result); | |
| 1112 return; | |
| 1113 } | |
| 1114 | |
| 1115 bool needs_sort = false; | |
| 1116 if (first_query) { | |
| 1117 if (family_ == ADDRESS_FAMILY_UNSPECIFIED) { | |
|
mmenke
2012/06/07 18:32:34
nit: You might want to swap these two if's, so th
| |
| 1118 first_addr_list_ = addr_list; | |
| 1119 first_ttl_ = ttl; | |
| 1120 // Use fully-qualified domain name to avoid search. | |
| 1121 transaction_ = client_->GetTransactionFactory()->CreateTransaction( | |
| 1122 response->GetDottedName() + ".", | |
| 1123 dns_protocol::kTypeA, | |
| 1124 base::Bind(&DnsTask::OnTransactionComplete, base::Unretained(this), | |
| 1125 false /* first_query */, | |
| 1126 base::TimeTicks::Now()), | |
| 1127 net_log_); | |
| 1128 net_error = transaction_->Start(); | |
| 1129 if (net_error != ERR_IO_PENDING) | |
| 1130 OnFailure(net_error, DnsResponse::DNS_SUCCESS); | |
| 1131 return; | |
| 1132 } else if (family_ == ADDRESS_FAMILY_IPV6) { | |
| 1133 needs_sort = true; | |
| 1134 } | |
| 1135 } else { | |
| 1136 if (!first_addr_list_.empty()) { | |
| 1137 ttl = std::min(ttl, first_ttl_); | |
| 1138 addr_list.insert(addr_list.begin(), first_addr_list_.begin(), | |
| 1139 first_addr_list_.end()); | |
| 1140 needs_sort = true; | |
| 1141 } | |
| 1142 } | |
| 1143 | |
| 1144 if (addr_list.empty()) { | |
| 1145 // TODO(szym): Don't fallback to ProcTask in this case. | |
| 1146 OnFailure(ERR_NAME_NOT_RESOLVED, DnsResponse::DNS_NO_ADDRESSES); | |
| 1147 return; | |
| 1148 } | |
| 1149 | |
| 1150 if (needs_sort) { | |
| 1151 base::TimeTicks start_time = base::TimeTicks::Now(); | |
| 1152 if (!client_->GetAddressSorter()->Sort(&addr_list)) { | |
| 1153 DNS_HISTOGRAM("AsyncDNS.SortFailure", | |
| 1154 base::TimeTicks::Now() - start_time); | |
| 1155 OnFailure(ERR_DNS_SORT_ERROR, DnsResponse::DNS_SUCCESS); | |
| 1104 return; | 1156 return; |
| 1105 } | 1157 } |
| 1106 net_error = ERR_DNS_MALFORMED_RESPONSE; | 1158 DNS_HISTOGRAM("AsyncDNS.SortSuccess", |
| 1107 } else { | |
| 1108 DNS_HISTOGRAM("AsyncDNS.TransactionFailure", | |
| 1109 base::TimeTicks::Now() - start_time); | 1159 base::TimeTicks::Now() - start_time); |
| 1110 } | 1160 } |
| 1161 | |
| 1162 // AddressSorter prunes unusable destinations. | |
| 1163 if (addr_list.empty()) { | |
| 1164 OnFailure(ERR_NAME_NOT_RESOLVED, DnsResponse::DNS_SUCCESS); | |
| 1165 return; | |
| 1166 } | |
| 1167 | |
| 1168 OnSuccess(addr_list, ttl); | |
| 1169 } | |
| 1170 | |
| 1171 void OnFailure(int net_error, DnsResponse::Result result) { | |
| 1172 DCHECK_NE(OK, net_error); | |
| 1111 net_log_.EndEvent(NetLog::TYPE_HOST_RESOLVER_IMPL_DNS_TASK, | 1173 net_log_.EndEvent(NetLog::TYPE_HOST_RESOLVER_IMPL_DNS_TASK, |
| 1112 new DnsTaskFailedParams(net_error, result)); | 1174 new DnsTaskFailedParams(net_error, result)); |
| 1113 callback_.Run(net_error, AddressList(), base::TimeDelta()); | 1175 callback_.Run(net_error, AddressList(), base::TimeDelta()); |
| 1114 } | 1176 } |
| 1115 | 1177 |
| 1116 private: | 1178 void OnSuccess(const AddressList& addr_list, base::TimeDelta ttl) { |
| 1179 net_log_.EndEvent(NetLog::TYPE_HOST_RESOLVER_IMPL_DNS_TASK, | |
| 1180 new AddressListNetLogParam(addr_list)); | |
| 1181 callback_.Run(OK, addr_list, ttl); | |
| 1182 } | |
| 1183 | |
| 1184 DnsClient* client_; | |
| 1185 AddressFamily family_; | |
| 1117 // The listener to the results of this DnsTask. | 1186 // The listener to the results of this DnsTask. |
| 1118 Callback callback_; | 1187 Callback callback_; |
| 1119 | |
| 1120 const BoundNetLog net_log_; | 1188 const BoundNetLog net_log_; |
| 1121 | 1189 |
| 1122 scoped_ptr<DnsTransaction> transaction_; | 1190 scoped_ptr<DnsTransaction> transaction_; |
| 1191 | |
| 1192 // TODO(szym): wrap this up as HostResolver::Result, add canonical name. | |
| 1193 // Results from the first transaction. Used only if |family_| is unspecified. | |
| 1194 AddressList first_addr_list_; | |
| 1195 base::TimeDelta first_ttl_; | |
| 1123 }; | 1196 }; |
| 1124 | 1197 |
| 1125 //----------------------------------------------------------------------------- | 1198 //----------------------------------------------------------------------------- |
| 1126 | 1199 |
| 1127 // Aggregates all Requests for the same Key. Dispatched via PriorityDispatch. | 1200 // Aggregates all Requests for the same Key. Dispatched via PriorityDispatch. |
| 1128 class HostResolverImpl::Job : public PrioritizedDispatcher::Job { | 1201 class HostResolverImpl::Job : public PrioritizedDispatcher::Job { |
| 1129 public: | 1202 public: |
| 1130 // Creates new job for |key| where |request_net_log| is bound to the | 1203 // Creates new job for |key| where |request_net_log| is bound to the |
| 1131 // request that spawned it. | 1204 // request that spawned it. |
| 1132 Job(HostResolverImpl* resolver, | 1205 Job(HostResolverImpl* resolver, |
| (...skipping 73 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1206 proc_task_->set_had_non_speculative_request(); | 1279 proc_task_->set_had_non_speculative_request(); |
| 1207 } | 1280 } |
| 1208 | 1281 |
| 1209 requests_.push_back(req.release()); | 1282 requests_.push_back(req.release()); |
| 1210 | 1283 |
| 1211 if (is_queued()) | 1284 if (is_queued()) |
| 1212 handle_ = resolver_->dispatcher_.ChangePriority(handle_, priority()); | 1285 handle_ = resolver_->dispatcher_.ChangePriority(handle_, priority()); |
| 1213 } | 1286 } |
| 1214 | 1287 |
| 1215 // Marks |req| as cancelled. If it was the last active Request, also finishes | 1288 // Marks |req| as cancelled. If it was the last active Request, also finishes |
| 1216 // this Job marking it either as aborted or cancelled, and deletes it. | 1289 // this Job marking it as cancelled, and deletes it. |
| 1217 void CancelRequest(Request* req) { | 1290 void CancelRequest(Request* req) { |
| 1218 DCHECK_EQ(key_.hostname, req->info().hostname()); | 1291 DCHECK_EQ(key_.hostname, req->info().hostname()); |
| 1219 DCHECK(!req->was_canceled()); | 1292 DCHECK(!req->was_canceled()); |
| 1220 | 1293 |
| 1221 // Don't remove it from |requests_| just mark it canceled. | 1294 // Don't remove it from |requests_| just mark it canceled. |
| 1222 req->MarkAsCanceled(); | 1295 req->MarkAsCanceled(); |
| 1223 LogCancelRequest(req->source_net_log(), req->request_net_log(), | 1296 LogCancelRequest(req->source_net_log(), req->request_net_log(), |
| 1224 req->info()); | 1297 req->info()); |
| 1225 | 1298 |
| 1226 priority_tracker_.Remove(req->info().priority()); | 1299 priority_tracker_.Remove(req->info().priority()); |
| (...skipping 115 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1342 kNegativeCacheEntryTTLSeconds); | 1415 kNegativeCacheEntryTTLSeconds); |
| 1343 if (net_error == OK) | 1416 if (net_error == OK) |
| 1344 ttl = base::TimeDelta::FromSeconds(kCacheEntryTTLSeconds); | 1417 ttl = base::TimeDelta::FromSeconds(kCacheEntryTTLSeconds); |
| 1345 | 1418 |
| 1346 CompleteRequests(net_error, addr_list, ttl); | 1419 CompleteRequests(net_error, addr_list, ttl); |
| 1347 } | 1420 } |
| 1348 | 1421 |
| 1349 void StartDnsTask() { | 1422 void StartDnsTask() { |
| 1350 DCHECK(resolver_->HaveDnsConfig()); | 1423 DCHECK(resolver_->HaveDnsConfig()); |
| 1351 dns_task_.reset(new DnsTask( | 1424 dns_task_.reset(new DnsTask( |
| 1352 resolver_->dns_client_->GetTransactionFactory(), | 1425 resolver_->dns_client_.get(), |
| 1353 key_, | 1426 key_, |
| 1354 base::Bind(&Job::OnDnsTaskComplete, base::Unretained(this)), | 1427 base::Bind(&Job::OnDnsTaskComplete, base::Unretained(this)), |
| 1355 net_log_)); | 1428 net_log_)); |
| 1356 | 1429 |
| 1357 int rv = dns_task_->Start(); | 1430 int rv = dns_task_->Start(); |
| 1358 if (rv != ERR_IO_PENDING) { | 1431 if (rv != ERR_IO_PENDING) { |
| 1359 DCHECK_NE(OK, rv); | 1432 DCHECK_NE(OK, rv); |
| 1360 dns_task_.reset(); | 1433 dns_task_.reset(); |
| 1361 StartProcTask(); | 1434 StartProcTask(); |
| 1362 } | 1435 } |
| (...skipping 11 matching lines...) Expand all Loading... | |
| 1374 // TODO(szym): Run ServeFromHosts now if nsswitch.conf says so. | 1447 // TODO(szym): Run ServeFromHosts now if nsswitch.conf says so. |
| 1375 // http://crbug.com/117655 | 1448 // http://crbug.com/117655 |
| 1376 | 1449 |
| 1377 // TODO(szym): Some net errors indicate lack of connectivity. Starting | 1450 // TODO(szym): Some net errors indicate lack of connectivity. Starting |
| 1378 // ProcTask in that case is a waste of time. | 1451 // ProcTask in that case is a waste of time. |
| 1379 StartProcTask(); | 1452 StartProcTask(); |
| 1380 return; | 1453 return; |
| 1381 } | 1454 } |
| 1382 | 1455 |
| 1383 UmaAsyncDnsResolveStatus(RESOLVE_STATUS_DNS_SUCCESS); | 1456 UmaAsyncDnsResolveStatus(RESOLVE_STATUS_DNS_SUCCESS); |
| 1457 | |
| 1384 CompleteRequests(net_error, addr_list, ttl); | 1458 CompleteRequests(net_error, addr_list, ttl); |
| 1385 } | 1459 } |
| 1386 | 1460 |
| 1387 // Performs Job's last rites. Completes all Requests. Deletes this. | 1461 // Performs Job's last rites. Completes all Requests. Deletes this. |
| 1388 void CompleteRequests(int net_error, | 1462 void CompleteRequests(int net_error, |
| 1389 const AddressList& addr_list, | 1463 const AddressList& addr_list, |
| 1390 base::TimeDelta ttl) { | 1464 base::TimeDelta ttl) { |
| 1391 CHECK(resolver_); | 1465 CHECK(resolver_); |
| 1392 | 1466 |
| 1393 // This job must be removed from resolver's |jobs_| now to make room for a | 1467 // This job must be removed from resolver's |jobs_| now to make room for a |
| (...skipping 586 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1980 if (self) | 2054 if (self) |
| 1981 TryServingAllJobsFromHosts(); | 2055 TryServingAllJobsFromHosts(); |
| 1982 } | 2056 } |
| 1983 } | 2057 } |
| 1984 | 2058 |
| 1985 bool HostResolverImpl::HaveDnsConfig() const { | 2059 bool HostResolverImpl::HaveDnsConfig() const { |
| 1986 return (dns_client_.get() != NULL) && (dns_client_->GetConfig() != NULL); | 2060 return (dns_client_.get() != NULL) && (dns_client_->GetConfig() != NULL); |
| 1987 } | 2061 } |
| 1988 | 2062 |
| 1989 } // namespace net | 2063 } // namespace net |
| OLD | NEW |