blob: cc29545fa1c93ac92c2c9b63660fcaa7a753ec51 (
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
|
// Copyright (c) 2010 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/socket/ssl_host_info.h"
#include "net/socket/ssl_client_socket.h"
#ifdef ANDROID
// the android platform build system use a fixed include path relative to the
// top directory (root of the source tree).
#include "external/chromium/net/socket/ssl_host_info.pb.h"
#else
#include "net/socket/ssl_host_info.pb.h"
#endif
namespace net {
SSLHostInfo::SSLHostInfo() {
state_.npn_valid = false;
}
SSLHostInfo::~SSLHostInfo() {}
// This array and the next two functions serve to map between the internal NPN
// status enum (which might change across versions) and the protocol buffer
// based enum (which will not).
static const struct {
SSLClientSocket::NextProtoStatus npn_status;
SSLHostInfoProto::NextProtoStatus proto_status;
} kNPNStatusMapping[] = {
{ SSLClientSocket::kNextProtoUnsupported, SSLHostInfoProto::UNSUPPORTED },
{ SSLClientSocket::kNextProtoNegotiated, SSLHostInfoProto::NEGOTIATED },
{ SSLClientSocket::kNextProtoNoOverlap, SSLHostInfoProto::NO_OVERLAP },
};
static SSLClientSocket::NextProtoStatus NPNStatusFromProtoStatus(
SSLHostInfoProto::NextProtoStatus proto_status) {
for (size_t i = 0; i < ARRAYSIZE_UNSAFE(kNPNStatusMapping) - 1; i++) {
if (kNPNStatusMapping[i].proto_status == proto_status)
return kNPNStatusMapping[i].npn_status;
}
return kNPNStatusMapping[ARRAYSIZE_UNSAFE(kNPNStatusMapping) - 1].npn_status;
}
static SSLHostInfoProto::NextProtoStatus ProtoStatusFromNPNStatus(
SSLClientSocket::NextProtoStatus npn_status) {
for (size_t i = 0; i < ARRAYSIZE_UNSAFE(kNPNStatusMapping) - 1; i++) {
if (kNPNStatusMapping[i].npn_status == npn_status)
return kNPNStatusMapping[i].proto_status;
}
return kNPNStatusMapping[ARRAYSIZE_UNSAFE(kNPNStatusMapping)-1].proto_status;
}
const SSLHostInfo::State& SSLHostInfo::state() const {
return state_;
}
SSLHostInfo::State* SSLHostInfo::mutable_state() {
return &state_;
}
bool SSLHostInfo::Parse(const std::string& data) {
SSLHostInfoProto proto;
State* state = mutable_state();
state->certs.clear();
state->server_hello.clear();
state->npn_valid = false;
if (!proto.ParseFromString(data))
return false;
for (int i = 0; i < proto.certificate_der_size(); i++)
state->certs.push_back(proto.certificate_der(i));
if (proto.has_server_hello())
state->server_hello = proto.server_hello();
if (proto.has_npn_status() && proto.has_npn_protocol()) {
state->npn_valid = true;
state->npn_status = NPNStatusFromProtoStatus(proto.npn_status());
state->npn_protocol = proto.npn_protocol();
}
return true;
}
std::string SSLHostInfo::Serialize() const {
SSLHostInfoProto proto;
for (std::vector<std::string>::const_iterator
i = state_.certs.begin(); i != state_.certs.end(); i++) {
proto.add_certificate_der(*i);
}
if (!state_.server_hello.empty())
proto.set_server_hello(state_.server_hello);
if (state_.npn_valid) {
proto.set_npn_status(ProtoStatusFromNPNStatus(state_.npn_status));
proto.set_npn_protocol(state_.npn_protocol);
}
return proto.SerializeAsString();
}
SSLHostInfoFactory::~SSLHostInfoFactory() {}
} // namespace net
|