blob: c45788bb66dafa51b1d3d2647b907513d9bac742 (
plain)
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
|
// Copyright (c) 2009 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "net/flip/flip_session_pool.h"
#include "base/logging.h"
#include "net/flip/flip_session.h"
namespace net {
// The maximum number of sessions to open to a single domain.
static const size_t kMaxSessionsPerDomain = 1;
FlipSessionPool::FlipSessionPool() {}
FlipSessionPool::~FlipSessionPool() {
CloseAllSessions();
}
scoped_refptr<FlipSession> FlipSessionPool::Get(
const HostResolver::RequestInfo& info, HttpNetworkSession* session) {
const std::string& domain = info.hostname();
scoped_refptr<FlipSession> flip_session;
FlipSessionList* list = GetSessionList(domain);
if (list) {
if (list->size() >= kMaxSessionsPerDomain) {
flip_session = list->front();
list->pop_front();
}
} else {
list = AddSessionList(domain);
}
DCHECK(list);
if (!flip_session)
flip_session = new FlipSession(domain, session);
DCHECK(flip_session);
list->push_back(flip_session);
DCHECK(list->size() <= kMaxSessionsPerDomain);
return flip_session;
}
scoped_refptr<FlipSession> FlipSessionPool::GetFlipSessionFromSocket(
const HostResolver::RequestInfo& info,
HttpNetworkSession* session,
ClientSocketHandle* connection) {
const std::string& domain = info.hostname();
FlipSessionList* list = GetSessionList(domain);
if (!list)
list = AddSessionList(domain);
DCHECK(list->empty());
scoped_refptr<FlipSession> flip_session(new FlipSession(domain, session));
flip_session->InitializeWithSocket(connection);
list->push_back(flip_session);
return flip_session;
}
bool FlipSessionPool::HasSession(const HostResolver::RequestInfo& info) const {
const std::string& domain = info.hostname();
if (GetSessionList(domain))
return true;
return false;
}
void FlipSessionPool::Remove(const scoped_refptr<FlipSession>& session) {
std::string domain = session->domain();
FlipSessionList* list = GetSessionList(domain);
CHECK(list);
list->remove(session);
if (list->empty())
RemoveSessionList(domain);
}
FlipSessionPool::FlipSessionList*
FlipSessionPool::AddSessionList(const std::string& domain) {
DCHECK(sessions_.find(domain) == sessions_.end());
return sessions_[domain] = new FlipSessionList();
}
FlipSessionPool::FlipSessionList*
FlipSessionPool::GetSessionList(const std::string& domain) {
FlipSessionsMap::iterator it = sessions_.find(domain);
if (it == sessions_.end())
return NULL;
return it->second;
}
const FlipSessionPool::FlipSessionList*
FlipSessionPool::GetSessionList(const std::string& domain) const {
FlipSessionsMap::const_iterator it = sessions_.find(domain);
if (it == sessions_.end())
return NULL;
return it->second;
}
void FlipSessionPool::RemoveSessionList(const std::string& domain) {
FlipSessionList* list = GetSessionList(domain);
if (list) {
delete list;
sessions_.erase(domain);
} else {
DCHECK(false) << "removing orphaned session list";
}
}
void FlipSessionPool::CloseAllSessions() {
while (sessions_.size()) {
FlipSessionList* list = sessions_.begin()->second;
DCHECK(list);
sessions_.erase(sessions_.begin()->first);
while (list->size()) {
scoped_refptr<FlipSession> session = list->front();
list->pop_front();
session->CloseAllStreams(net::ERR_ABORTED);
}
delete list;
}
}
} // namespace net
|