1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
|
// Copyright 2008, Google Inc.
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
// * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above
// copyright notice, this list of conditions and the following disclaimer
// in the documentation and/or other materials provided with the
// distribution.
// * Neither the name of Google Inc. nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
// A DnsMaster object is instantiated once in the browser
// process, and delivers DNS prefetch assignments (hostnames)
// to any of several DnsSlave objects.
// Most hostname lists are sent out by renderer processes, and
// involve lists of hostnames that *might* be used in the near
// future by the browsing user. The goal of this class is to
// cause the underlying DNS structure to lookup a hostname before
// it is really needed, and hence reduce latency in the standard
// lookup paths. Since some DNS lookups may take a LONG time, we
// use several DnsSlave threads to concurrently perform the
// lookups.
#ifndef CHROME_BROWSER_NET_DNS_MASTER_H__
#define CHROME_BROWSER_NET_DNS_MASTER_H__
#include <map>
#include <queue>
#include <string>
#include "base/condition_variable.h"
#include "base/scoped_ptr.h"
#include "chrome/browser/net/dns_host_info.h"
#include "chrome/common/net/dns.h"
#include "googleurl/src/url_canon.h"
namespace chrome_browser_net {
class DnsSlave;
typedef chrome_common_net::NameList NameList;
typedef std::map<std::string, DnsHostInfo> Results;
class DnsMaster {
public:
explicit DnsMaster(TimeDelta shutdown_wait_time);
~DnsMaster() {
if (!shutdown_)
ShutdownSlaves(); // Ensure we did our cleanup.
}
// ShutdownSlaves() gets all spawned threads to terminate, closes
// their handles, and deletes their DnsSlave instances.
// Return value of true means all operations succeeded.
// Return value of false means that the threads wouldn't terminate,
// and that resources may leak. If this returns false, it is best
// to NOT delete this DnsMaster, as slave threads may still call into
// this object.
bool ShutdownSlaves();
// In some circumstances, for privacy reasons, all results should be
// discarded. This method gracefully handles that activity.
// Destroy all our internal state, which shows what names we've looked up, and
// how long each has taken, etc. etc. We also destroy records of suggesses
// (cache hits etc.).
void DiscardAllResults();
// Add hostname(s) to the queue for processing by slaves
void ResolveList(const NameList& hostnames);
void Resolve(const std::string& hostname);
// Get latency benefit of the prefetch that we are navigating to.
bool AcruePrefetchBenefits(DnsHostInfo* host_info);
void GetHtmlInfo(std::string* output);
// For testing only...
// Currently testing only provides a crude measure of success.
bool WasFound(const std::string& hostname) {
AutoLock auto_lock(lock_);
return (results_.find(hostname) != results_.end()) &&
results_[hostname].was_found();
}
// Accessor methods, used mostly for testing.
// Both functions return DnsHostInfo::kNullDuration if name was not yet
// processed enough.
TimeDelta GetResolutionDuration(const std::string hostname) {
AutoLock auto_lock(lock_);
if (results_.find(hostname) == results_.end())
return DnsHostInfo::kNullDuration;
return results_[hostname].resolve_duration();
}
TimeDelta GetQueueDuration(const std::string hostname) {
AutoLock auto_lock(lock_);
if (results_.find(hostname) == results_.end())
return DnsHostInfo::kNullDuration;
return results_[hostname].queue_duration();
}
int running_slave_count() {
AutoLock auto_lock(lock_);
return running_slave_count_;
}
//----------------------------------------------------------------------------
// Methods below this line should only be called by slave processes.
// Thread names can only be set after the thread has been running a bit.
void SetSlaveName(int slave_index);
// GetNextAssignment() gets the next hostname from queue for processing
// It is not meant to be public, and should only be used by the slave.
// GetNextAssignment() waits on a condition variable if there are no more
// names in queue.
// Return false if slave thread should terminate.
// Return true if slave thread should process the value.
bool GetNextAssignment(std::string* hostname);
// Access methods for use by slave threads to callback with state updates.
void SetFoundState(const std::string hostname);
void SetNoSuchNameState(const std::string hostname);
// Notification during ShutdownSlaves.
void SetSlaveHasTerminated(int slave_index);
private:
//----------------------------------------------------------------------------
// Internal helper functions
// "PreLocked" means that the caller has already Acquired lock_ in the
// following method names.
void PreLockedResolve(const std::string& hostname);
bool PreLockedCreateNewSlaveIfNeeded(); // Lazy slave processes creation.
// The number of slave processes that will do DNS prefetching
static const int kSlaveCountMax = 8;
// Number of slave processes started early (to help with startup prefetch).
static const int kSlaveCountMin = 4;
Lock lock_;
// name_buffer_ holds a list of names we need to look up.
std::queue<std::string> name_buffer_;
// results_ contains information progress for existing/prior prefetches.
Results results_;
// Signaling slaves to process elements in the queue, or to terminate,
// is done using ConditionVariables.
ConditionVariable slaves_have_work_;
int slave_count_; // Count of slave processes started.
int running_slave_count_; // Count of slaves process still running.
// The following arrays are only initialized as
// slave_count_ grows (up to the indicated max).
DWORD thread_ids_[kSlaveCountMax];
HANDLE thread_handles_[kSlaveCountMax];
DnsSlave* slaves_[kSlaveCountMax];
// shutdown_ is set to tell the slaves to terminate.
bool shutdown_;
// The following is the maximum time the ShutdownSlaves method
// will wait for all the slave processes to terminate.
const TimeDelta kShutdownWaitTime_;
// A list of successful events resulting from pre-fetching.
DnsHostInfo::DnsInfoTable cache_hits_;
// A map of hosts that were evicted from our cache (after we prefetched them)
// and before the HTTP stack tried to look them up.
Results cache_eviction_map_;
DISALLOW_EVIL_CONSTRUCTORS(DnsMaster);
};
} // namespace chrome_browser_net
#endif // CHROME_BROWSER_NET_DNS_MASTER_H__
|