summaryrefslogtreecommitdiffstats
path: root/net/base
diff options
context:
space:
mode:
authorericroman@google.com <ericroman@google.com@0039d316-1c4b-4281-b951-d872f2087c98>2009-06-27 01:50:14 +0000
committerericroman@google.com <ericroman@google.com@0039d316-1c4b-4281-b951-d872f2087c98>2009-06-27 01:50:14 +0000
commit94a0d3d91b7200e66a4d23bb49b38cae7e74f7db (patch)
tree7f6b01af9e6e2a741dc4938d8e524be9f277aa3a /net/base
parentafef5e483bc3782ecaf2572493442e0a2bd22bac (diff)
downloadchromium_src-94a0d3d91b7200e66a4d23bb49b38cae7e74f7db.zip
chromium_src-94a0d3d91b7200e66a4d23bb49b38cae7e74f7db.tar.gz
chromium_src-94a0d3d91b7200e66a4d23bb49b38cae7e74f7db.tar.bz2
Make net::HostResolver refcounted.
This way it can be properly shared between the url request contexts, and the dns prefetcher, and dns observer. BUG=http://crbug.com/14664 TEST=existing unit tests. Review URL: http://codereview.chromium.org/149053 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@19451 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'net/base')
-rw-r--r--net/base/host_resolver.h4
-rw-r--r--net/base/host_resolver_unittest.cc121
2 files changed, 66 insertions, 59 deletions
diff --git a/net/base/host_resolver.h b/net/base/host_resolver.h
index 0b5c3fc..a912661 100644
--- a/net/base/host_resolver.h
+++ b/net/base/host_resolver.h
@@ -57,7 +57,7 @@ class HostMapper;
// Thread safety: This class is not threadsafe, and must only be called
// from one thread!
//
-class HostResolver {
+class HostResolver : public base::RefCounted<HostResolver> {
public:
// The parameters for doing a Resolve(). |hostname| and |port| are required,
// the rest are optional (and have reasonable defaults).
@@ -236,7 +236,7 @@ class SingleRequestHostResolver {
void OnResolveCompletion(int result);
// The actual host resolver that will handle the request.
- HostResolver* resolver_;
+ scoped_refptr<HostResolver> resolver_;
// The current request (if any).
HostResolver::Request* cur_request_;
diff --git a/net/base/host_resolver_unittest.cc b/net/base/host_resolver_unittest.cc
index 8fb0d16..32b6af6 100644
--- a/net/base/host_resolver_unittest.cc
+++ b/net/base/host_resolver_unittest.cc
@@ -140,6 +140,8 @@ class ResolveRequest {
int result_;
net::AddressList addrlist_;
+ // We don't use a scoped_refptr, to simplify deleting shared resolver in
+ // DeleteWithinCallback test.
net::HostResolver* resolver_;
Delegate* delegate_;
@@ -170,7 +172,7 @@ class HostResolverTest : public testing::Test {
};
TEST_F(HostResolverTest, SynchronousLookup) {
- net::HostResolver host_resolver;
+ scoped_refptr<net::HostResolver> host_resolver(new net::HostResolver);
net::AddressList adrlist;
const int kPortnum = 80;
@@ -179,7 +181,7 @@ TEST_F(HostResolverTest, SynchronousLookup) {
ScopedHostMapper scoped_mapper(mapper.get());
net::HostResolver::RequestInfo info("just.testing", kPortnum);
- int err = host_resolver.Resolve(info, &adrlist, NULL, NULL);
+ int err = host_resolver->Resolve(info, &adrlist, NULL, NULL);
EXPECT_EQ(net::OK, err);
const struct addrinfo* ainfo = adrlist.head();
@@ -193,7 +195,7 @@ TEST_F(HostResolverTest, SynchronousLookup) {
}
TEST_F(HostResolverTest, AsynchronousLookup) {
- net::HostResolver host_resolver;
+ scoped_refptr<net::HostResolver> host_resolver(new net::HostResolver);
net::AddressList adrlist;
const int kPortnum = 80;
@@ -202,7 +204,7 @@ TEST_F(HostResolverTest, AsynchronousLookup) {
ScopedHostMapper scoped_mapper(mapper.get());
net::HostResolver::RequestInfo info("just.testing", kPortnum);
- int err = host_resolver.Resolve(info, &adrlist, &callback_, NULL);
+ int err = host_resolver->Resolve(info, &adrlist, &callback_, NULL);
EXPECT_EQ(net::ERR_IO_PENDING, err);
MessageLoop::current()->Run();
@@ -225,12 +227,12 @@ TEST_F(HostResolverTest, CanceledAsynchronousLookup) {
ScopedHostMapper scoped_mapper(mapper.get());
{
- net::HostResolver host_resolver;
+ scoped_refptr<net::HostResolver> host_resolver(new net::HostResolver);
net::AddressList adrlist;
const int kPortnum = 80;
net::HostResolver::RequestInfo info("just.testing", kPortnum);
- int err = host_resolver.Resolve(info, &adrlist, &callback_, NULL);
+ int err = host_resolver->Resolve(info, &adrlist, &callback_, NULL);
EXPECT_EQ(net::ERR_IO_PENDING, err);
// Make sure we will exit the queue even when callback is not called.
@@ -252,11 +254,11 @@ TEST_F(HostResolverTest, NumericIPv4Address) {
mapper->AllowDirectLookup("*");
ScopedHostMapper scoped_mapper(mapper.get());
- net::HostResolver host_resolver;
+ scoped_refptr<net::HostResolver> host_resolver(new net::HostResolver);
net::AddressList adrlist;
const int kPortnum = 5555;
net::HostResolver::RequestInfo info("127.1.2.3", kPortnum);
- int err = host_resolver.Resolve(info, &adrlist, NULL, NULL);
+ int err = host_resolver->Resolve(info, &adrlist, NULL, NULL);
EXPECT_EQ(net::OK, err);
const struct addrinfo* ainfo = adrlist.head();
@@ -276,14 +278,14 @@ TEST_F(HostResolverTest, NumericIPv6Address) {
// Resolve a plain IPv6 address. Don't worry about [brackets], because
// the caller should have removed them.
- net::HostResolver host_resolver;
+ scoped_refptr<net::HostResolver> host_resolver(new net::HostResolver);
net::AddressList adrlist;
const int kPortnum = 5555;
net::HostResolver::RequestInfo info("2001:db8::1", kPortnum);
- int err = host_resolver.Resolve(info, &adrlist, NULL, NULL);
+ int err = host_resolver->Resolve(info, &adrlist, NULL, NULL);
// On computers without IPv6 support, getaddrinfo cannot convert IPv6
// address literals to addresses (getaddrinfo returns EAI_NONAME). So this
- // test has to allow host_resolver.Resolve to fail.
+ // test has to allow host_resolver->Resolve to fail.
if (err == net::ERR_NAME_NOT_RESOLVED)
return;
EXPECT_EQ(net::OK, err);
@@ -310,11 +312,11 @@ TEST_F(HostResolverTest, EmptyHost) {
mapper->AllowDirectLookup("*");
ScopedHostMapper scoped_mapper(mapper.get());
- net::HostResolver host_resolver;
+ scoped_refptr<net::HostResolver> host_resolver(new net::HostResolver);
net::AddressList adrlist;
const int kPortnum = 5555;
net::HostResolver::RequestInfo info("", kPortnum);
- int err = host_resolver.Resolve(info, &adrlist, NULL, NULL);
+ int err = host_resolver->Resolve(info, &adrlist, NULL, NULL);
EXPECT_EQ(net::ERR_NAME_NOT_RESOLVED, err);
}
@@ -370,7 +372,7 @@ TEST_F(HostResolverTest, DeDupeRequests) {
scoped_refptr<CapturingHostMapper> mapper = new CapturingHostMapper();
ScopedHostMapper scoped_mapper(mapper.get());
- net::HostResolver host_resolver;
+ scoped_refptr<net::HostResolver> host_resolver(new net::HostResolver);
// The class will receive callbacks for when each resolve completes. It
// checks that the right things happened.
@@ -379,11 +381,11 @@ TEST_F(HostResolverTest, DeDupeRequests) {
// Start 5 requests, duplicating hosts "a" and "b". Since the mapper is
// blocked, these should all pile up until we signal it.
- ResolveRequest req1(&host_resolver, "a", 80, &verifier);
- ResolveRequest req2(&host_resolver, "b", 80, &verifier);
- ResolveRequest req3(&host_resolver, "b", 81, &verifier);
- ResolveRequest req4(&host_resolver, "a", 82, &verifier);
- ResolveRequest req5(&host_resolver, "b", 83, &verifier);
+ ResolveRequest req1(host_resolver, "a", 80, &verifier);
+ ResolveRequest req2(host_resolver, "b", 80, &verifier);
+ ResolveRequest req3(host_resolver, "b", 81, &verifier);
+ ResolveRequest req4(host_resolver, "a", 82, &verifier);
+ ResolveRequest req5(host_resolver, "b", 83, &verifier);
// Ready, Set, GO!!!
mapper->Signal();
@@ -419,7 +421,7 @@ TEST_F(HostResolverTest, CancelMultipleRequests) {
scoped_refptr<CapturingHostMapper> mapper = new CapturingHostMapper();
ScopedHostMapper scoped_mapper(mapper.get());
- net::HostResolver host_resolver;
+ scoped_refptr<net::HostResolver> host_resolver(new net::HostResolver);
// The class will receive callbacks for when each resolve completes. It
// checks that the right things happened.
@@ -428,11 +430,11 @@ TEST_F(HostResolverTest, CancelMultipleRequests) {
// Start 5 requests, duplicating hosts "a" and "b". Since the mapper is
// blocked, these should all pile up until we signal it.
- ResolveRequest req1(&host_resolver, "a", 80, &verifier);
- ResolveRequest req2(&host_resolver, "b", 80, &verifier);
- ResolveRequest req3(&host_resolver, "b", 81, &verifier);
- ResolveRequest req4(&host_resolver, "a", 82, &verifier);
- ResolveRequest req5(&host_resolver, "b", 83, &verifier);
+ ResolveRequest req1(host_resolver, "a", 80, &verifier);
+ ResolveRequest req2(host_resolver, "b", 80, &verifier);
+ ResolveRequest req3(host_resolver, "b", 81, &verifier);
+ ResolveRequest req4(host_resolver, "a", 82, &verifier);
+ ResolveRequest req5(host_resolver, "b", 83, &verifier);
// Cancel everything except request 4.
req1.Cancel();
@@ -504,7 +506,7 @@ TEST_F(HostResolverTest, CancelWithinCallback) {
scoped_refptr<CapturingHostMapper> mapper = new CapturingHostMapper();
ScopedHostMapper scoped_mapper(mapper.get());
- net::HostResolver host_resolver;
+ scoped_refptr<net::HostResolver> host_resolver(new net::HostResolver);
// The class will receive callbacks for when each resolve completes. It
// checks that the right things happened.
@@ -513,10 +515,10 @@ TEST_F(HostResolverTest, CancelWithinCallback) {
// Start 4 requests, duplicating hosts "a". Since the mapper is
// blocked, these should all pile up until we signal it.
- ResolveRequest req1(&host_resolver, "a", 80, &verifier);
- ResolveRequest req2(&host_resolver, "a", 81, &verifier);
- ResolveRequest req3(&host_resolver, "a", 82, &verifier);
- ResolveRequest req4(&host_resolver, "a", 83, &verifier);
+ ResolveRequest req1(host_resolver, "a", 80, &verifier);
+ ResolveRequest req2(host_resolver, "a", 81, &verifier);
+ ResolveRequest req3(host_resolver, "a", 82, &verifier);
+ ResolveRequest req4(host_resolver, "a", 83, &verifier);
// Once "a:80" completes, it will cancel "a:81" and "a:82".
verifier.SetRequestsToCancel(&req2, &req3);
@@ -531,12 +533,18 @@ TEST_F(HostResolverTest, CancelWithinCallback) {
// Helper class used by HostResolverTest.DeleteWithinCallback.
class DeleteWithinCallbackVerifier : public ResolveRequest::Delegate {
public:
- DeleteWithinCallbackVerifier() {}
+ // |host_resolver| is the resolver that the the resolve requests were started
+ // with.
+ DeleteWithinCallbackVerifier(net::HostResolver* host_resolver)
+ : host_resolver_(host_resolver) {}
virtual void OnCompleted(ResolveRequest* resolve) {
EXPECT_EQ("a", resolve->hostname());
EXPECT_EQ(80, resolve->port());
- delete resolve->resolver();
+
+ // Release the last reference to the host resolver that started the
+ // requests.
+ host_resolver_ = NULL;
// Quit after returning from OnCompleted (to give it a chance at
// incorrectly running the cancelled tasks).
@@ -544,6 +552,7 @@ class DeleteWithinCallbackVerifier : public ResolveRequest::Delegate {
}
private:
+ scoped_refptr<net::HostResolver> host_resolver_;
DISALLOW_COPY_AND_ASSIGN(DeleteWithinCallbackVerifier);
};
@@ -553,13 +562,11 @@ TEST_F(HostResolverTest, DeleteWithinCallback) {
scoped_refptr<CapturingHostMapper> mapper = new CapturingHostMapper();
ScopedHostMapper scoped_mapper(mapper.get());
- // This should be deleted by DeleteWithinCallbackVerifier -- if it leaks
- // then the test has failed.
- net::HostResolver* host_resolver = new net::HostResolver;
-
// The class will receive callbacks for when each resolve completes. It
- // checks that the right things happened.
- DeleteWithinCallbackVerifier verifier;
+ // checks that the right things happened. Note that the verifier holds the
+ // only reference to |host_resolver|, so it can delete it within callback.
+ net::HostResolver* host_resolver = new net::HostResolver;
+ DeleteWithinCallbackVerifier verifier(host_resolver);
// Start 4 requests, duplicating hosts "a". Since the mapper is
// blocked, these should all pile up until we signal it.
@@ -609,7 +616,7 @@ TEST_F(HostResolverTest, StartWithinCallback) {
ScopedHostMapper scoped_mapper(mapper.get());
// Turn off caching for this host resolver.
- net::HostResolver host_resolver(0, 0);
+ scoped_refptr<net::HostResolver> host_resolver(new net::HostResolver(0, 0));
// The class will receive callbacks for when each resolve completes. It
// checks that the right things happened.
@@ -618,10 +625,10 @@ TEST_F(HostResolverTest, StartWithinCallback) {
// Start 4 requests, duplicating hosts "a". Since the mapper is
// blocked, these should all pile up until we signal it.
- ResolveRequest req1(&host_resolver, "a", 80, &verifier);
- ResolveRequest req2(&host_resolver, "a", 81, &verifier);
- ResolveRequest req3(&host_resolver, "a", 82, &verifier);
- ResolveRequest req4(&host_resolver, "a", 83, &verifier);
+ ResolveRequest req1(host_resolver, "a", 80, &verifier);
+ ResolveRequest req2(host_resolver, "a", 81, &verifier);
+ ResolveRequest req3(host_resolver, "a", 82, &verifier);
+ ResolveRequest req4(host_resolver, "a", 83, &verifier);
// Ready, Set, GO!!!
mapper->Signal();
@@ -673,14 +680,14 @@ class BypassCacheVerifier : public ResolveRequest::Delegate {
};
TEST_F(HostResolverTest, BypassCache) {
- net::HostResolver host_resolver;
+ scoped_refptr<net::HostResolver> host_resolver(new net::HostResolver);
// The class will receive callbacks for when each resolve completes. It
// checks that the right things happened.
BypassCacheVerifier verifier;
// Start a request.
- ResolveRequest req1(&host_resolver, "a", 80, &verifier);
+ ResolveRequest req1(host_resolver, "a", 80, &verifier);
// |verifier| will send quit message once all the requests have finished.
MessageLoop::current()->Run();
@@ -756,17 +763,17 @@ class CapturingObserver : public net::HostResolver::Observer {
// Does not test the cancellation notification since all resolves are
// synchronous.
TEST_F(HostResolverTest, Observers) {
- net::HostResolver host_resolver;
+ scoped_refptr<net::HostResolver> host_resolver(new net::HostResolver);
CapturingObserver observer;
- host_resolver.AddObserver(&observer);
+ host_resolver->AddObserver(&observer);
net::AddressList addrlist;
// Resolve "host1".
net::HostResolver::RequestInfo info1("host1", 70);
- int rv = host_resolver.Resolve(info1, &addrlist, NULL, NULL);
+ int rv = host_resolver->Resolve(info1, &addrlist, NULL, NULL);
EXPECT_EQ(net::OK, rv);
EXPECT_EQ(1U, observer.start_log.size());
@@ -780,7 +787,7 @@ TEST_F(HostResolverTest, Observers) {
// Resolve "host1" again -- this time it will be served from cache, but it
// should still notify of completion.
TestCompletionCallback callback;
- rv = host_resolver.Resolve(info1, &addrlist, &callback, NULL);
+ rv = host_resolver->Resolve(info1, &addrlist, &callback, NULL);
ASSERT_EQ(net::OK, rv); // Should complete synchronously.
EXPECT_EQ(2U, observer.start_log.size());
@@ -794,7 +801,7 @@ TEST_F(HostResolverTest, Observers) {
// Resolve "host2", setting referrer to "http://foobar.com"
net::HostResolver::RequestInfo info2("host2", 70);
info2.set_referrer(GURL("http://foobar.com"));
- rv = host_resolver.Resolve(info2, &addrlist, NULL, NULL);
+ rv = host_resolver->Resolve(info2, &addrlist, NULL, NULL);
EXPECT_EQ(net::OK, rv);
EXPECT_EQ(3U, observer.start_log.size());
@@ -806,11 +813,11 @@ TEST_F(HostResolverTest, Observers) {
CapturingObserver::FinishEntry(2, true, info2));
// Unregister the observer.
- host_resolver.RemoveObserver(&observer);
+ host_resolver->RemoveObserver(&observer);
// Resolve "host3"
net::HostResolver::RequestInfo info3("host3", 70);
- host_resolver.Resolve(info3, &addrlist, NULL, NULL);
+ host_resolver->Resolve(info3, &addrlist, NULL, NULL);
// No effect this time, since observer was removed.
EXPECT_EQ(3U, observer.start_log.size());
@@ -826,8 +833,8 @@ TEST_F(HostResolverTest, CancellationObserver) {
CapturingObserver observer;
{
// Create a host resolver and attach an observer.
- net::HostResolver host_resolver;
- host_resolver.AddObserver(&observer);
+ scoped_refptr<net::HostResolver> host_resolver(new net::HostResolver);
+ host_resolver->AddObserver(&observer);
TestCompletionCallback callback;
@@ -839,7 +846,7 @@ TEST_F(HostResolverTest, CancellationObserver) {
net::HostResolver::RequestInfo info1("host1", 70);
net::HostResolver::Request* req = NULL;
net::AddressList addrlist;
- int rv = host_resolver.Resolve(info1, &addrlist, &callback, &req);
+ int rv = host_resolver->Resolve(info1, &addrlist, &callback, &req);
EXPECT_EQ(net::ERR_IO_PENDING, rv);
EXPECT_TRUE(NULL != req);
@@ -851,7 +858,7 @@ TEST_F(HostResolverTest, CancellationObserver) {
CapturingObserver::StartOrCancelEntry(0, info1));
// Cancel the request (host mapper is blocked so it cant be finished yet).
- host_resolver.CancelRequest(req);
+ host_resolver->CancelRequest(req);
EXPECT_EQ(1U, observer.start_log.size());
EXPECT_EQ(0U, observer.finish_log.size());
@@ -862,7 +869,7 @@ TEST_F(HostResolverTest, CancellationObserver) {
// Start an async request for (host2:60)
net::HostResolver::RequestInfo info2("host2", 60);
- rv = host_resolver.Resolve(info2, &addrlist, &callback, NULL);
+ rv = host_resolver->Resolve(info2, &addrlist, &callback, NULL);
EXPECT_EQ(net::ERR_IO_PENDING, rv);
EXPECT_TRUE(NULL != req);