blob: 45c103be49118ba2cfe8cbb7e7730ddf193de243 [file] [log] [blame]
//
//
// Copyright 2015 gRPC 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
//
// http://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 <algorithm>
#include <memory>
#include <string>
#include <utility>
#include <grpc/grpc.h>
#include <grpc/slice.h>
#include <grpc/support/log.h>
#include "src/core/ext/transport/chttp2/transport/hpack_parser.h"
#include "src/core/lib/gprpp/ref_counted_ptr.h"
#include "src/core/lib/iomgr/exec_ctx.h"
#include "src/core/lib/resource_quota/arena.h"
#include "src/core/lib/resource_quota/memory_quota.h"
#include "src/core/lib/resource_quota/resource_quota.h"
#include "src/core/lib/transport/metadata_batch.h"
#include "src/libfuzzer/libfuzzer_macro.h"
#include "test/core/transport/chttp2/hpack_parser_fuzzer.pb.h"
// IWYU pragma: no_include <google/protobuf/repeated_ptr_field.h>
bool squelch = true;
bool leak_check = true;
static void dont_log(gpr_log_func_args* /*args*/) {}
DEFINE_PROTO_FUZZER(const hpack_parser_fuzzer::Msg& msg) {
if (squelch) gpr_set_log_function(dont_log);
grpc_init();
auto memory_allocator = grpc_core::ResourceQuota::Default()
->memory_quota()
->CreateMemoryAllocator("test-allocator");
{
std::unique_ptr<grpc_core::HPackParser> parser(new grpc_core::HPackParser);
for (int i = 0; i < msg.frames_size(); i++) {
auto arena = grpc_core::MakeScopedArena(1024, &memory_allocator);
grpc_core::ExecCtx exec_ctx;
grpc_metadata_batch b(arena.get());
const auto& frame = msg.frames(i);
grpc_core::HPackParser::Boundary boundary =
grpc_core::HPackParser::Boundary::None;
if (frame.end_of_headers()) {
boundary = grpc_core::HPackParser::Boundary::EndOfHeaders;
}
if (frame.end_of_stream()) {
boundary = grpc_core::HPackParser::Boundary::EndOfStream;
}
grpc_core::HPackParser::Priority priority =
grpc_core::HPackParser::Priority::None;
if (frame.priority()) {
priority = grpc_core::HPackParser::Priority::Included;
}
int max_length = 1024;
int absolute_max_length = 1024;
if (absolute_max_length < max_length) {
std::swap(absolute_max_length, max_length);
}
if (frame.max_metadata_length() != 0) {
max_length = frame.max_metadata_length();
}
if (frame.absolute_max_metadata_length() != 0) {
absolute_max_length = frame.absolute_max_metadata_length();
}
parser->BeginFrame(
&b, max_length, absolute_max_length, boundary, priority,
grpc_core::HPackParser::LogInfo{
1, grpc_core::HPackParser::LogInfo::kHeaders, false});
int stop_buffering_ctr =
std::max(-1, frame.stop_buffering_after_segments());
for (const auto& parse : frame.parse()) {
grpc_slice buffer =
grpc_slice_from_copied_buffer(parse.data(), parse.size());
(void)parser->Parse(buffer, i == msg.frames_size() - 1);
grpc_slice_unref(buffer);
stop_buffering_ctr--;
if (0 == stop_buffering_ctr) parser->StopBufferingFrame();
}
parser->FinishFrame();
}
}
grpc_shutdown();
}