xref: /btstack/3rd-party/lc3-google/fuzz/efuzz.cc (revision 6897da5c53aac5b1f90f41b5b15d0bd43d61dfff)
1*6897da5cSDirk Helbig /******************************************************************************
2*6897da5cSDirk Helbig  *
3*6897da5cSDirk Helbig  *  Copyright 2022 Google LLC
4*6897da5cSDirk Helbig  *
5*6897da5cSDirk Helbig  *  Licensed under the Apache License, Version 2.0 (the "License");
6*6897da5cSDirk Helbig  *  you may not use this file except in compliance with the License.
7*6897da5cSDirk Helbig  *  You may obtain a copy of the License at:
8*6897da5cSDirk Helbig  *
9*6897da5cSDirk Helbig  *  http://www.apache.org/licenses/LICENSE-2.0
10*6897da5cSDirk Helbig  *
11*6897da5cSDirk Helbig  *  Unless required by applicable law or agreed to in writing, software
12*6897da5cSDirk Helbig  *  distributed under the License is distributed on an "AS IS" BASIS,
13*6897da5cSDirk Helbig  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14*6897da5cSDirk Helbig  *  See the License for the specific language governing permissions and
15*6897da5cSDirk Helbig  *  limitations under the License.
16*6897da5cSDirk Helbig  *
17*6897da5cSDirk Helbig  ******************************************************************************/
18*6897da5cSDirk Helbig 
19*6897da5cSDirk Helbig #include <lc3_cpp.h>
20*6897da5cSDirk Helbig #include <fuzzer/FuzzedDataProvider.h>
21*6897da5cSDirk Helbig 
22*6897da5cSDirk Helbig using namespace lc3;
23*6897da5cSDirk Helbig 
24*6897da5cSDirk Helbig template <typename T>
ConsumeInRange(FuzzedDataProvider & fdp,T min,T max)25*6897da5cSDirk Helbig T ConsumeInRange(FuzzedDataProvider &fdp, T min, T max) {
26*6897da5cSDirk Helbig   return fdp.ConsumeIntegralInRange<T>(min, max);
27*6897da5cSDirk Helbig }
28*6897da5cSDirk Helbig 
29*6897da5cSDirk Helbig template <>
ConsumeInRange(FuzzedDataProvider & fdp,float min,float max)30*6897da5cSDirk Helbig float ConsumeInRange(FuzzedDataProvider &fdp, float min, float max) {
31*6897da5cSDirk Helbig   return fdp.ConsumeFloatingPointInRange<float>(min, max);
32*6897da5cSDirk Helbig }
33*6897da5cSDirk Helbig 
34*6897da5cSDirk Helbig template <typename T>
encode(Encoder & e,int nchannels,int frame_size,FuzzedDataProvider & fdp,T min=std::numeric_limits<T>::min (),T max=std::numeric_limits<T>::max ())35*6897da5cSDirk Helbig int encode(Encoder &e, int nchannels, int frame_size, FuzzedDataProvider &fdp,
36*6897da5cSDirk Helbig   T min = std::numeric_limits<T>::min(), T max = std::numeric_limits<T>::max())
37*6897da5cSDirk Helbig {
38*6897da5cSDirk Helbig   int pcm_samples = nchannels * e.GetFrameSamples();
39*6897da5cSDirk Helbig   if (fdp.remaining_bytes() < pcm_samples * sizeof(T))
40*6897da5cSDirk Helbig     return -1;
41*6897da5cSDirk Helbig 
42*6897da5cSDirk Helbig   std::vector<T> pcm(pcm_samples);
43*6897da5cSDirk Helbig   for (auto &s: pcm)
44*6897da5cSDirk Helbig     s = ConsumeInRange<T>(fdp, min, max);
45*6897da5cSDirk Helbig 
46*6897da5cSDirk Helbig   e.Encode(pcm.data(),
47*6897da5cSDirk Helbig     frame_size, std::vector<uint8_t>(nchannels * frame_size).data());
48*6897da5cSDirk Helbig 
49*6897da5cSDirk Helbig   return 0;
50*6897da5cSDirk Helbig }
51*6897da5cSDirk Helbig 
encode(Encoder & e,int frame_size,int nchannels,PcmFormat fmt,FuzzedDataProvider & fdp)52*6897da5cSDirk Helbig int encode(Encoder &e, int frame_size, int nchannels,
53*6897da5cSDirk Helbig   PcmFormat fmt, FuzzedDataProvider &fdp)
54*6897da5cSDirk Helbig {
55*6897da5cSDirk Helbig   int sample_bytes =
56*6897da5cSDirk Helbig     fmt == PcmFormat::kS16 ? sizeof(int16_t) :
57*6897da5cSDirk Helbig     fmt == PcmFormat::kS24 ? sizeof(int32_t) :
58*6897da5cSDirk Helbig     fmt == PcmFormat::kS24In3Le ? sizeof(uint8_t) * 3 :
59*6897da5cSDirk Helbig     fmt == PcmFormat::kF32 ? sizeof(float) : 0;
60*6897da5cSDirk Helbig 
61*6897da5cSDirk Helbig   int pcm_bytes = nchannels * e.GetFrameSamples() * sample_bytes;
62*6897da5cSDirk Helbig   if (fdp.remaining_bytes() < pcm_bytes)
63*6897da5cSDirk Helbig     return -1;
64*6897da5cSDirk Helbig 
65*6897da5cSDirk Helbig   e.Encode(fmt, fdp.ConsumeBytes<uint8_t>(pcm_bytes).data(),
66*6897da5cSDirk Helbig     frame_size, std::vector<uint8_t>(nchannels * frame_size).data());
67*6897da5cSDirk Helbig 
68*6897da5cSDirk Helbig   return 0;
69*6897da5cSDirk Helbig }
70*6897da5cSDirk Helbig 
LLVMFuzzerTestOneInput(const uint8_t * data,size_t size)71*6897da5cSDirk Helbig extern "C" int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size)
72*6897da5cSDirk Helbig {
73*6897da5cSDirk Helbig   const int dt_list[] = { 2500, 5000, 7500, 10000 };
74*6897da5cSDirk Helbig   const int sr_list[] = { 8000, 16000, 24000, 32000, 48000 };
75*6897da5cSDirk Helbig 
76*6897da5cSDirk Helbig   FuzzedDataProvider fdp(data, size);
77*6897da5cSDirk Helbig 
78*6897da5cSDirk Helbig   int dt_us = fdp.PickValueInArray(dt_list);
79*6897da5cSDirk Helbig   int sr_hz = fdp.PickValueInArray(sr_list);
80*6897da5cSDirk Helbig   int nchannels = fdp.PickValueInArray({1, 2});
81*6897da5cSDirk Helbig   bool hrmode = fdp.ConsumeBool();
82*6897da5cSDirk Helbig 
83*6897da5cSDirk Helbig   int sr_pcm_hz = fdp.PickValueInArray(sr_list);
84*6897da5cSDirk Helbig   if (sr_pcm_hz < sr_hz)
85*6897da5cSDirk Helbig     sr_pcm_hz = 0;
86*6897da5cSDirk Helbig 
87*6897da5cSDirk Helbig   Encoder enc(dt_us, sr_hz, sr_pcm_hz, nchannels, hrmode);
88*6897da5cSDirk Helbig 
89*6897da5cSDirk Helbig   PcmFormat fmt = fdp.PickValueInArray(
90*6897da5cSDirk Helbig     { PcmFormat::kS16, PcmFormat::kS24,
91*6897da5cSDirk Helbig       PcmFormat::kS24In3Le, PcmFormat::kF32 });
92*6897da5cSDirk Helbig 
93*6897da5cSDirk Helbig   int frame_size = fdp.ConsumeIntegralInRange(
94*6897da5cSDirk Helbig     LC3_MIN_FRAME_BYTES, LC3_MAX_FRAME_BYTES);
95*6897da5cSDirk Helbig 
96*6897da5cSDirk Helbig   switch (fmt) {
97*6897da5cSDirk Helbig 
98*6897da5cSDirk Helbig   case PcmFormat::kS16:
99*6897da5cSDirk Helbig     return encode<int16_t>(enc, nchannels, frame_size, fdp);
100*6897da5cSDirk Helbig 
101*6897da5cSDirk Helbig   case PcmFormat::kS24: {
102*6897da5cSDirk Helbig     const int32_t s24_min = -(1 << 23);
103*6897da5cSDirk Helbig     const int32_t s24_max =  (1 << 23) - 1;
104*6897da5cSDirk Helbig     return encode<int32_t>(enc, nchannels, frame_size, fdp, s24_min, s24_max);
105*6897da5cSDirk Helbig   }
106*6897da5cSDirk Helbig 
107*6897da5cSDirk Helbig   case PcmFormat::kF32: {
108*6897da5cSDirk Helbig     const float f32_min = -1.0;
109*6897da5cSDirk Helbig     const float f32_max =  1.0;
110*6897da5cSDirk Helbig     return encode<float>(enc, nchannels, frame_size, fdp, f32_min, f32_max);
111*6897da5cSDirk Helbig   }
112*6897da5cSDirk Helbig 
113*6897da5cSDirk Helbig   case PcmFormat::kS24In3Le:
114*6897da5cSDirk Helbig     return encode(enc, nchannels, frame_size, fmt, fdp);
115*6897da5cSDirk Helbig   }
116*6897da5cSDirk Helbig 
117*6897da5cSDirk Helbig   return 0;
118*6897da5cSDirk Helbig }
119