|  | // Copyright 2018 The BoringSSL Authors | 
|  | // | 
|  | // Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | // you may not use this file except in compliance with the License. | 
|  | // You may obtain a copy of the License at | 
|  | // | 
|  | //     https://www.apache.org/licenses/LICENSE-2.0 | 
|  | // | 
|  | // Unless required by applicable law or agreed to in writing, software | 
|  | // distributed under the License is distributed on an "AS IS" BASIS, | 
|  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | // See the License for the specific language governing permissions and | 
|  | // limitations under the License. | 
|  |  | 
|  | #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; | 
|  | } |