| Index: net/http/http_stream_factory_impl_request.cc | 
| diff --git a/net/http/http_stream_factory_impl_request.cc b/net/http/http_stream_factory_impl_request.cc | 
| index b6e6b9f6544eed9093ec4bd83559d22ae7c552ec..b9e0cee74664f4652863ccadf3ab358aec6f0252 100644 | 
| --- a/net/http/http_stream_factory_impl_request.cc | 
| +++ b/net/http/http_stream_factory_impl_request.cc | 
| @@ -7,6 +7,8 @@ | 
| #include "base/logging.h" | 
| #include "base/stl_util-inl.h" | 
| #include "net/http/http_stream_factory_impl_job.h" | 
| +#include "net/spdy/spdy_http_stream.h" | 
| +#include "net/spdy/spdy_session.h" | 
|  | 
| namespace net { | 
|  | 
| @@ -18,7 +20,6 @@ HttpStreamFactoryImpl::Request::Request(const GURL& url, | 
| factory_(factory), | 
| delegate_(delegate), | 
| net_log_(net_log), | 
| -      job_(NULL), | 
| completed_(false), | 
| was_alternate_protocol_available_(false), | 
| was_npn_negotiated_(false), | 
| @@ -30,12 +31,17 @@ HttpStreamFactoryImpl::Request::Request(const GURL& url, | 
| } | 
|  | 
| HttpStreamFactoryImpl::Request::~Request() { | 
| +  if (bound_job_.get()) | 
| +    DCHECK(jobs_.empty()); | 
| +  else | 
| +    DCHECK(!jobs_.empty()); | 
| + | 
| net_log_.EndEvent(NetLog::TYPE_HTTP_STREAM_REQUEST, NULL); | 
|  | 
| -  factory_->request_map_.erase(job_); | 
| +  for (std::set<Job*>::iterator it = jobs_.begin(); it != jobs_.end(); ++it) | 
| +    factory_->request_map_.erase(*it); | 
|  | 
| -  // TODO(willchan): Remove this when we decouple requests and jobs. | 
| -  delete job_; | 
| +  STLDeleteElements(&jobs_); | 
|  | 
| RemoveRequestFromSpdySessionRequestMap(); | 
| } | 
| @@ -50,10 +56,10 @@ void HttpStreamFactoryImpl::Request::SetSpdySessionKey( | 
| request_set.insert(this); | 
| } | 
|  | 
| -void HttpStreamFactoryImpl::Request::BindJob(HttpStreamFactoryImpl::Job* job) { | 
| +void HttpStreamFactoryImpl::Request::AttachJob(Job* job) { | 
| DCHECK(job); | 
| -  DCHECK(!job_); | 
| -  job_ = job; | 
| +  jobs_.insert(job); | 
| +  factory_->request_map_[job] = this; | 
| } | 
|  | 
| void HttpStreamFactoryImpl::Request::Complete( | 
| @@ -73,49 +79,96 @@ void HttpStreamFactoryImpl::Request::Complete( | 
| } | 
|  | 
| void HttpStreamFactoryImpl::Request::OnStreamReady( | 
| +    Job* job, | 
| const SSLConfig& used_ssl_config, | 
| const ProxyInfo& used_proxy_info, | 
| HttpStream* stream) { | 
| DCHECK(stream); | 
| DCHECK(completed_); | 
| + | 
| +  // |job| should only be NULL if we're being serviced by a late bound | 
| +  // SpdySession (one that was not created by a job in our |jobs_| set). | 
| +  if (!job) { | 
| +    DCHECK(!bound_job_.get()); | 
| +    DCHECK(!jobs_.empty()); | 
| +    // NOTE(willchan): We do *NOT* call OrphanJobs() here. The reason is because | 
| +    // we *WANT* to cancel the unnecessary Jobs from other requests if another | 
| +    // Job completes first. | 
| +    // TODO(mbelshe): Revisit this when we implement ip connection pooling of | 
| +    // SpdySessions. Do we want to orphan the jobs for a different hostname so | 
| +    // they complete? Or do we want to prevent connecting a new SpdySession if | 
| +    // we've already got one available for a different hostname where the ip | 
| +    // address matches up? | 
| +  } else if (!bound_job_.get()) { | 
| +    // We may have other jobs in |jobs_|. For example, if we start multiple jobs | 
| +    // for Alternate-Protocol. | 
| +    OrphanJobsExcept(job); | 
| +  } else { | 
| +    DCHECK(jobs_.empty()); | 
| +  } | 
| delegate_->OnStreamReady(used_ssl_config, used_proxy_info, stream); | 
| } | 
|  | 
| void HttpStreamFactoryImpl::Request::OnStreamFailed( | 
| +    Job* job, | 
| int status, | 
| const SSLConfig& used_ssl_config) { | 
| DCHECK_NE(OK, status); | 
| +  if (!bound_job_.get()) | 
| +    OrphanJobsExcept(job); | 
| +  else | 
| +    DCHECK(jobs_.empty()); | 
| delegate_->OnStreamFailed(status, used_ssl_config); | 
| } | 
|  | 
| void HttpStreamFactoryImpl::Request::OnCertificateError( | 
| +    Job* job, | 
| int status, | 
| const SSLConfig& used_ssl_config, | 
| const SSLInfo& ssl_info) { | 
| DCHECK_NE(OK, status); | 
| +  if (!bound_job_.get()) | 
| +    OrphanJobsExcept(job); | 
| +  else | 
| +    DCHECK(jobs_.empty()); | 
| delegate_->OnCertificateError(status, used_ssl_config, ssl_info); | 
| } | 
|  | 
| void HttpStreamFactoryImpl::Request::OnNeedsProxyAuth( | 
| +    Job* job, | 
| const HttpResponseInfo& proxy_response, | 
| const SSLConfig& used_ssl_config, | 
| const ProxyInfo& used_proxy_info, | 
| HttpAuthController* auth_controller) { | 
| +  if (!bound_job_.get()) | 
| +    OrphanJobsExcept(job); | 
| +  else | 
| +    DCHECK(jobs_.empty()); | 
| delegate_->OnNeedsProxyAuth( | 
| proxy_response, used_ssl_config, used_proxy_info, auth_controller); | 
| } | 
|  | 
| void HttpStreamFactoryImpl::Request::OnNeedsClientAuth( | 
| +    Job* job, | 
| const SSLConfig& used_ssl_config, | 
| SSLCertRequestInfo* cert_info) { | 
| +  if (!bound_job_.get()) | 
| +    OrphanJobsExcept(job); | 
| +  else | 
| +    DCHECK(jobs_.empty()); | 
| delegate_->OnNeedsClientAuth(used_ssl_config, cert_info); | 
| } | 
|  | 
| void HttpStreamFactoryImpl::Request::OnHttpsProxyTunnelResponse( | 
| +    Job *job, | 
| const HttpResponseInfo& response_info, | 
| const SSLConfig& used_ssl_config, | 
| const ProxyInfo& used_proxy_info, | 
| HttpStream* stream) { | 
| +  if (!bound_job_.get()) | 
| +    OrphanJobsExcept(job); | 
| +  else | 
| +    DCHECK(jobs_.empty()); | 
| delegate_->OnHttpsProxyTunnelResponse( | 
| response_info, used_ssl_config, used_proxy_info, stream); | 
| } | 
| @@ -123,15 +176,17 @@ void HttpStreamFactoryImpl::Request::OnHttpsProxyTunnelResponse( | 
| int HttpStreamFactoryImpl::Request::RestartTunnelWithProxyAuth( | 
| const string16& username, | 
| const string16& password) { | 
| -  // We're restarting the job, so ditch the old key. Note that we could actually | 
| -  // keep it around and eliminate the DCHECK in set_spdy_session_key() that | 
| -  // |spdy_session_key_| is NULL, but I prefer to keep the assertion. | 
| -  RemoveRequestFromSpdySessionRequestMap(); | 
| -  return job_->RestartTunnelWithProxyAuth(username, password); | 
| +  DCHECK(bound_job_.get()); | 
| +  return bound_job_->RestartTunnelWithProxyAuth(username, password); | 
| } | 
|  | 
| LoadState HttpStreamFactoryImpl::Request::GetLoadState() const { | 
| -  return factory_->GetLoadState(*this); | 
| +  if (bound_job_.get()) | 
| +    return bound_job_->GetLoadState(); | 
| +  DCHECK(!jobs_.empty()); | 
| + | 
| +  // Just pick the first one. | 
| +  return (*jobs_.begin())->GetLoadState(); | 
| } | 
|  | 
| bool HttpStreamFactoryImpl::Request::was_alternate_protocol_available() const { | 
| @@ -165,4 +220,68 @@ HttpStreamFactoryImpl::Request::RemoveRequestFromSpdySessionRequestMap() { | 
| } | 
| } | 
|  | 
| +void HttpStreamFactoryImpl::Request::OnSpdySessionReady( | 
| +    Job* job, | 
| +    scoped_refptr<SpdySession> spdy_session, | 
| +    bool direct) { | 
| +  DCHECK(job); | 
| +  DCHECK(job->using_spdy()); | 
| + | 
| +  // The first case is the usual case. | 
| +  if (!bound_job_.get()) { | 
| +    OrphanJobsExcept(job); | 
| +  } else { // This is the case for HTTPS proxy tunneling. | 
| +    DCHECK_EQ(bound_job_.get(), job); | 
| +    DCHECK(jobs_.empty()); | 
| +  } | 
| + | 
| +  // Cache these values in case the job gets deleted. | 
| +  const SSLConfig used_ssl_config = job->ssl_config(); | 
| +  const ProxyInfo used_proxy_info = job->proxy_info(); | 
| +  const bool was_alternate_protocol_available = | 
| +      job->was_alternate_protocol_available(); | 
| +  const bool was_npn_negotiated = job->was_npn_negotiated(); | 
| +  const bool using_spdy = job->using_spdy(); | 
| +  const NetLog::Source source = job->net_log().source(); | 
| + | 
| +  Complete(was_alternate_protocol_available, | 
| +           was_npn_negotiated, | 
| +           using_spdy, | 
| +           source); | 
| + | 
| +  // Cache this so we can still use it if the request is deleted. | 
| +  HttpStreamFactoryImpl* factory = factory_; | 
| + | 
| +  bool use_relative_url = direct || url().SchemeIs("https"); | 
| +  delegate_->OnStreamReady( | 
| +      job->ssl_config(), | 
| +      job->proxy_info(), | 
| +      new SpdyHttpStream(spdy_session, use_relative_url)); | 
| +  // |this| may be deleted after this point. | 
| +  factory->OnSpdySessionReady( | 
| +      spdy_session, direct, used_ssl_config, used_proxy_info, | 
| +      was_alternate_protocol_available, was_npn_negotiated, using_spdy, source); | 
| +} | 
| + | 
| +void HttpStreamFactoryImpl::Request::OrphanJobsExcept(Job* job) { | 
| +  DCHECK(job); | 
| +  DCHECK(!bound_job_.get()); | 
| +  DCHECK(ContainsKey(jobs_, job)); | 
| +  bound_job_.reset(job); | 
| +  jobs_.erase(job); | 
| +  factory_->request_map_.erase(job); | 
| + | 
| +  OrphanJobs(); | 
| +} | 
| + | 
| +void HttpStreamFactoryImpl::Request::OrphanJobs() { | 
| +  RemoveRequestFromSpdySessionRequestMap(); | 
| + | 
| +  std::set<Job*> tmp; | 
| +  tmp.swap(jobs_); | 
| + | 
| +  for (std::set<Job*>::iterator it = tmp.begin(); it != tmp.end(); ++it) | 
| +    factory_->OrphanJob(*it, this); | 
| +} | 
| + | 
| }  // namespace net | 
|  |