blob: 9c9bbf69d595c93316f16eae0a1238a287ed3ba7 [file] [log] [blame]
Henrik Lundinfe32a762015-12-08 11:27:27 +01001/*
2 * Copyright (c) 2015 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "test/fuzzers/audio_decoder_fuzzer.h"
Henrik Lundinfe32a762015-12-08 11:27:27 +010012
henrik.lundinfd2be272016-02-09 00:00:23 -080013#include <limits>
14
Danil Chapovalov431abd92018-06-18 12:54:17 +020015#include "absl/types/optional.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020016#include "api/audio_codecs/audio_decoder.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020017#include "modules/rtp_rtcp/source/byte_io.h"
18#include "rtc_base/checks.h"
Henrik Lundinfe32a762015-12-08 11:27:27 +010019
20namespace webrtc {
21namespace {
henrik.lundinfd2be272016-02-09 00:00:23 -080022template <typename T, unsigned int B = sizeof(T)>
23bool ParseInt(const uint8_t** data, size_t* remaining_size, T* value) {
24 static_assert(std::numeric_limits<T>::is_integer, "Type must be an integer.");
25 static_assert(sizeof(T) <= sizeof(uint64_t),
26 "Cannot read wider than uint64_t.");
27 static_assert(B <= sizeof(T), "T must be at least B bytes wide.");
28 if (B > *remaining_size)
29 return false;
30 uint64_t val = ByteReader<uint64_t, B>::ReadBigEndian(*data);
31 *data += B;
32 *remaining_size -= B;
33 *value = static_cast<T>(val);
34 return true;
Henrik Lundinfe32a762015-12-08 11:27:27 +010035}
36} // namespace
37
38// This function reads two bytes from the beginning of |data|, interprets them
39// as the first packet length, and reads this many bytes if available. The
40// payload is inserted into the decoder, and the process continues until no more
henrik.lundinfd2be272016-02-09 00:00:23 -080041// data is available. Either AudioDecoder::Decode or
42// AudioDecoder::DecodeRedundant is used, depending on the value of
43// |decode_type|.
44void FuzzAudioDecoder(DecoderFunctionType decode_type,
45 const uint8_t* data,
Henrik Lundinfe32a762015-12-08 11:27:27 +010046 size_t size,
47 AudioDecoder* decoder,
48 int sample_rate_hz,
49 size_t max_decoded_bytes,
50 int16_t* decoded) {
51 const uint8_t* data_ptr = data;
52 size_t remaining_size = size;
henrik.lundinfd2be272016-02-09 00:00:23 -080053 size_t packet_len;
Sam Zackrisson35c773d2018-07-13 16:00:31 +020054 constexpr size_t kMaxNumFuzzedPackets = 200;
55 for (size_t num_packets = 0; num_packets < kMaxNumFuzzedPackets;
56 ++num_packets) {
57 if (!(ParseInt<size_t, 2>(&data_ptr, &remaining_size, &packet_len) &&
58 packet_len <= remaining_size)) {
59 break;
60 }
Henrik Lundinfe32a762015-12-08 11:27:27 +010061 AudioDecoder::SpeechType speech_type;
henrik.lundinfd2be272016-02-09 00:00:23 -080062 switch (decode_type) {
63 case DecoderFunctionType::kNormalDecode:
64 decoder->Decode(data_ptr, packet_len, sample_rate_hz, max_decoded_bytes,
65 decoded, &speech_type);
66 break;
67 case DecoderFunctionType::kRedundantDecode:
68 decoder->DecodeRedundant(data_ptr, packet_len, sample_rate_hz,
69 max_decoded_bytes, decoded, &speech_type);
70 break;
71 }
Henrik Lundinfe32a762015-12-08 11:27:27 +010072 data_ptr += packet_len;
73 remaining_size -= packet_len;
henrik.lundinfd2be272016-02-09 00:00:23 -080074 }
75}
76
Henrik Lundinfe32a762015-12-08 11:27:27 +010077} // namespace webrtc