Mirror of BoringSSL (grpc依赖)
https://boringssl.googlesource.com/boringssl
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
115 lines
3.2 KiB
115 lines
3.2 KiB
/* Copyright (c) 2018, Google Inc. |
|
* |
|
* Permission to use, copy, modify, and/or distribute this software for any |
|
* purpose with or without fee is hereby granted, provided that the above |
|
* copyright notice and this permission notice appear in all copies. |
|
* |
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES |
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF |
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY |
|
* SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES |
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION |
|
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN |
|
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ |
|
|
|
#include "settings_writer.h" |
|
|
|
#include <stdio.h> |
|
|
|
#include <openssl/ssl.h> |
|
|
|
#include "fuzzer_tags.h" |
|
#include "test_config.h" |
|
|
|
|
|
SettingsWriter::SettingsWriter() {} |
|
|
|
bool SettingsWriter::Init(int i, const TestConfig *config, |
|
SSL_SESSION *session) { |
|
if (config->write_settings.empty()) { |
|
return true; |
|
} |
|
// Treat write_settings as a path prefix for each connection in the run. |
|
char buf[DECIMAL_SIZE(int)]; |
|
snprintf(buf, sizeof(buf), "%d", i); |
|
path_ = config->write_settings + buf; |
|
|
|
if (!CBB_init(cbb_.get(), 64)) { |
|
return false; |
|
} |
|
|
|
if (session != nullptr) { |
|
uint8_t *data; |
|
size_t len; |
|
if (!SSL_SESSION_to_bytes(session, &data, &len)) { |
|
return false; |
|
} |
|
bssl::UniquePtr<uint8_t> free_data(data); |
|
CBB child; |
|
if (!CBB_add_u16(cbb_.get(), kSessionTag) || |
|
!CBB_add_u24_length_prefixed(cbb_.get(), &child) || |
|
!CBB_add_bytes(&child, data, len) || !CBB_flush(cbb_.get())) { |
|
return false; |
|
} |
|
} |
|
|
|
if (config->is_server && |
|
(config->require_any_client_certificate || config->verify_peer) && |
|
!CBB_add_u16(cbb_.get(), kRequestClientCert)) { |
|
return false; |
|
} |
|
|
|
return true; |
|
} |
|
|
|
bool SettingsWriter::Commit() { |
|
if (path_.empty()) { |
|
return true; |
|
} |
|
|
|
uint8_t *settings; |
|
size_t settings_len; |
|
if (!CBB_add_u16(cbb_.get(), kDataTag) || |
|
!CBB_finish(cbb_.get(), &settings, &settings_len)) { |
|
return false; |
|
} |
|
bssl::UniquePtr<uint8_t> free_settings(settings); |
|
|
|
struct FileCloser { |
|
void operator()(FILE *f) const { fclose(f); } |
|
}; |
|
using ScopedFILE = std::unique_ptr<FILE, FileCloser>; |
|
ScopedFILE file(fopen(path_.c_str(), "w")); |
|
if (!file) { |
|
return false; |
|
} |
|
|
|
return fwrite(settings, settings_len, 1, file.get()) == 1; |
|
} |
|
|
|
bool SettingsWriter::WriteHandoff(bssl::Span<const uint8_t> handoff) { |
|
return WriteData(kHandoffTag, handoff); |
|
} |
|
|
|
bool SettingsWriter::WriteHandback(bssl::Span<const uint8_t> handback) { |
|
return WriteData(kHandbackTag, handback); |
|
} |
|
|
|
bool SettingsWriter::WriteHints(bssl::Span<const uint8_t> hints) { |
|
return WriteData(kHintsTag, hints); |
|
} |
|
|
|
bool SettingsWriter::WriteData(uint16_t tag, bssl::Span<const uint8_t> data) { |
|
if (path_.empty()) { |
|
return true; |
|
} |
|
|
|
CBB child; |
|
if (!CBB_add_u16(cbb_.get(), tag) || |
|
!CBB_add_u24_length_prefixed(cbb_.get(), &child) || |
|
!CBB_add_bytes(&child, data.data(), data.size()) || |
|
!CBB_flush(cbb_.get())) { |
|
return false; |
|
} |
|
return true; |
|
}
|
|
|