mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-10-31 22:25:30 +00:00
198 lines
5.2 KiB
C++
198 lines
5.2 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set sw=2 ts=8 et tw=80 : */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
// HttpLog.h should generally be included first
|
|
#include "HttpLog.h"
|
|
|
|
/*
|
|
Currently supported are HTTP-draft-[see nshttp.h]/2.0 spdy/3.1
|
|
*/
|
|
|
|
#include "nsHttp.h"
|
|
#include "nsHttpHandler.h"
|
|
|
|
#include "ASpdySession.h"
|
|
#include "PSpdyPush.h"
|
|
#include "SpdyPush31.h"
|
|
#include "Http2Push.h"
|
|
#include "SpdySession31.h"
|
|
#include "Http2Session.h"
|
|
|
|
#include "mozilla/Telemetry.h"
|
|
|
|
namespace mozilla {
|
|
namespace net {
|
|
|
|
ASpdySession::ASpdySession()
|
|
{
|
|
}
|
|
|
|
ASpdySession::~ASpdySession()
|
|
{
|
|
}
|
|
|
|
ASpdySession *
|
|
ASpdySession::NewSpdySession(uint32_t version,
|
|
nsISocketTransport *aTransport)
|
|
{
|
|
// This is a necko only interface, so we can enforce version
|
|
// requests as a precondition
|
|
MOZ_ASSERT(version == SPDY_VERSION_31 ||
|
|
version == HTTP_VERSION_2 ||
|
|
version == HTTP_VERSION_2_DRAFT_LATEST ||
|
|
version == HTTP_VERSION_2_DRAFT_15,
|
|
"Unsupported spdy version");
|
|
|
|
// Don't do a runtime check of IsSpdyV?Enabled() here because pref value
|
|
// may have changed since starting negotiation. The selected protocol comes
|
|
// from a list provided in the SERVER HELLO filtered by our acceptable
|
|
// versions, so there is no risk of the server ignoring our prefs.
|
|
|
|
Telemetry::Accumulate(Telemetry::SPDY_VERSION2, version);
|
|
|
|
if (version == SPDY_VERSION_31) {
|
|
return new SpdySession31(aTransport);
|
|
} else if (version == HTTP_VERSION_2_DRAFT_LATEST || version == HTTP_VERSION_2 ||
|
|
version == HTTP_VERSION_2_DRAFT_15) {
|
|
return new Http2Session(aTransport, version);
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
static bool SpdySessionTrue(nsISupports *securityInfo)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
SpdyInformation::SpdyInformation()
|
|
{
|
|
// highest index of enabled protocols is the
|
|
// most preferred for ALPN negotiaton
|
|
Version[0] = SPDY_VERSION_31;
|
|
VersionString[0] = NS_LITERAL_CSTRING("spdy/3.1");
|
|
ALPNCallbacks[0] = SpdySessionTrue;
|
|
|
|
Version[1] = HTTP_VERSION_2;
|
|
VersionString[1] = NS_LITERAL_CSTRING("h2");
|
|
ALPNCallbacks[1] = Http2Session::ALPNCallback;
|
|
|
|
Version[2] = HTTP_VERSION_2_DRAFT_15; // 14 and 15 are aliased
|
|
VersionString[2] = NS_LITERAL_CSTRING("h2-14");
|
|
ALPNCallbacks[2] = Http2Session::ALPNCallback;
|
|
|
|
Version[3] = HTTP_VERSION_2_DRAFT_15; // 14 and 15 are aliased
|
|
VersionString[3] = NS_LITERAL_CSTRING("h2-15");
|
|
ALPNCallbacks[3] = Http2Session::ALPNCallback;
|
|
|
|
Version[4] = HTTP_VERSION_2_DRAFT_LATEST;
|
|
VersionString[4] = NS_LITERAL_CSTRING(HTTP2_DRAFT_LATEST_TOKEN);
|
|
ALPNCallbacks[4] = Http2Session::ALPNCallback;
|
|
}
|
|
|
|
bool
|
|
SpdyInformation::ProtocolEnabled(uint32_t index) const
|
|
{
|
|
MOZ_ASSERT(index < kCount, "index out of range");
|
|
|
|
switch (index) {
|
|
case 0:
|
|
return gHttpHandler->IsSpdyV31Enabled();
|
|
case 1:
|
|
return gHttpHandler->IsHttp2Enabled();
|
|
case 2:
|
|
case 3:
|
|
case 4:
|
|
return gHttpHandler->IsHttp2DraftEnabled();
|
|
}
|
|
return false;
|
|
}
|
|
|
|
nsresult
|
|
SpdyInformation::GetNPNIndex(const nsACString &npnString,
|
|
uint32_t *result) const
|
|
{
|
|
if (npnString.IsEmpty())
|
|
return NS_ERROR_FAILURE;
|
|
|
|
for (uint32_t index = 0; index < kCount; ++index) {
|
|
if (npnString.Equals(VersionString[index])) {
|
|
*result = index;
|
|
return NS_OK;
|
|
}
|
|
}
|
|
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
//////////////////////////////////////////
|
|
// SpdyPushCache
|
|
//////////////////////////////////////////
|
|
|
|
SpdyPushCache::SpdyPushCache()
|
|
{
|
|
}
|
|
|
|
SpdyPushCache::~SpdyPushCache()
|
|
{
|
|
mHashSpdy31.Clear();
|
|
mHashHttp2.Clear();
|
|
}
|
|
|
|
bool
|
|
SpdyPushCache::RegisterPushedStreamSpdy31(nsCString key,
|
|
SpdyPushedStream31 *stream)
|
|
{
|
|
LOG3(("SpdyPushCache::RegisterPushedStreamSpdy31 %s 0x%X\n",
|
|
key.get(), stream->StreamID()));
|
|
if(mHashSpdy31.Get(key)) {
|
|
LOG3(("SpdyPushCache::RegisterPushedStreamSpdy31 %s 0x%X duplicate key\n",
|
|
key.get(), stream->StreamID()));
|
|
return false;
|
|
}
|
|
mHashSpdy31.Put(key, stream);
|
|
return true;
|
|
}
|
|
|
|
SpdyPushedStream31 *
|
|
SpdyPushCache::RemovePushedStreamSpdy31(nsCString key)
|
|
{
|
|
SpdyPushedStream31 *rv = mHashSpdy31.Get(key);
|
|
LOG3(("SpdyPushCache::RemovePushedStream %s 0x%X\n",
|
|
key.get(), rv ? rv->StreamID() : 0));
|
|
if (rv)
|
|
mHashSpdy31.Remove(key);
|
|
return rv;
|
|
}
|
|
|
|
bool
|
|
SpdyPushCache::RegisterPushedStreamHttp2(nsCString key,
|
|
Http2PushedStream *stream)
|
|
{
|
|
LOG3(("SpdyPushCache::RegisterPushedStreamHttp2 %s 0x%X\n",
|
|
key.get(), stream->StreamID()));
|
|
if(mHashHttp2.Get(key)) {
|
|
LOG3(("SpdyPushCache::RegisterPushedStreamHttp2 %s 0x%X duplicate key\n",
|
|
key.get(), stream->StreamID()));
|
|
return false;
|
|
}
|
|
mHashHttp2.Put(key, stream);
|
|
return true;
|
|
}
|
|
|
|
Http2PushedStream *
|
|
SpdyPushCache::RemovePushedStreamHttp2(nsCString key)
|
|
{
|
|
Http2PushedStream *rv = mHashHttp2.Get(key);
|
|
LOG3(("SpdyPushCache::RemovePushedStreamHttp2 %s 0x%X\n",
|
|
key.get(), rv ? rv->StreamID() : 0));
|
|
if (rv)
|
|
mHashHttp2.Remove(key);
|
|
return rv;
|
|
}
|
|
} // namespace mozilla::net
|
|
} // namespace mozilla
|
|
|