Coverage Report

Created: 2024-04-26 11:10

/src/decoder_fuzzer.cpp
Line
Count
Source (jump to first uncovered line)
1
// TODO: This should be moved to the openh264 repo.
2
3
#include <stddef.h>
4
#include <stdint.h>
5
#include <stdio.h>
6
#include <stdlib.h>
7
#include <string.h>
8
9
#include <memory>
10
11
#include "codec_def.h"
12
#include "codec_app_def.h"
13
#include "codec_api.h"
14
#include "read_config.h"
15
#include "typedefs.h"
16
#include "measure_time.h"
17
18
/*
19
 * To build locally:
20
 * CC=clang CXX=clang++ CFLAGS="-fsanitize=address,fuzzer-no-link -g" CXXFLAGS="-fsanitize=address,fuzzer-no-link -g" LDFLAGS="-fsanitize=address,fuzzer-no-link" make -j$(nproc) USE_ASM=No BUILDTYPE=Debug libraries
21
 * clang++ -o decoder_fuzzer -fsanitize=address -g -O1 -I./codec/api/svc -I./codec/console/common/inc -I./codec/common/inc -L. -lFuzzer -lstdc++ decoder_fuzzer.cpp libopenh264.a
22
 */
23
24
extern "C" int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size)
25
1.75M
{
26
1.75M
  int32_t i;
27
1.75M
  int32_t iBufPos = 0;
28
1.75M
  int32_t iEndOfStreamFlag;
29
1.75M
  int iLevelSetting = (int) WELS_LOG_QUIET; // disable logging while fuzzing
30
1.75M
  int32_t iSliceSize;
31
1.75M
  ISVCDecoder *pDecoder;
32
1.75M
  SDecodingParam sDecParam = {0};
33
1.75M
  SBufferInfo sDstBufInfo;
34
1.75M
  std::unique_ptr<uint8_t[]> pBuf(new uint8_t[size + 4]);
35
1.75M
  uint8_t* pData[3] = {NULL};
36
1.75M
  uint8_t uiStartCode[4] = {0, 0, 0, 1};
37
38
1.75M
  memcpy(pBuf.get(), data, size);
39
1.75M
  memcpy(pBuf.get() + size, &uiStartCode[0], 4);
40
1.75M
  memset(&sDstBufInfo, 0, sizeof(SBufferInfo));
41
42
  // TODO: is this the best/fastest ERROR_CON to use?
43
1.75M
  sDecParam.eEcActiveIdc = ERROR_CON_SLICE_COPY;
44
  // TODO: should we also fuzz VIDEO_BITSTREAM_SVC?
45
1.75M
  sDecParam.sVideoProperty.eVideoBsType = VIDEO_BITSTREAM_AVC;
46
  
47
1.75M
  WelsCreateDecoder (&pDecoder);
48
1.75M
  pDecoder->Initialize (&sDecParam);
49
1.75M
  pDecoder->SetOption (DECODER_OPTION_TRACE_LEVEL, &iLevelSetting);
50
51
30.2M
  while (1) {
52
30.2M
    if (iBufPos >= size) {
53
1.75M
      iEndOfStreamFlag = 1;
54
1.75M
      if (iEndOfStreamFlag)
55
1.75M
        pDecoder->SetOption (DECODER_OPTION_END_OF_STREAM, (void*)&iEndOfStreamFlag);
56
1.75M
      break;
57
1.75M
    }
58
59
670M
    for (i = 0; i < size; i++) {
60
670M
      if ((pBuf[iBufPos + i] == 0 && pBuf[iBufPos + i + 1] == 0 && pBuf[iBufPos + i + 2] == 0 && pBuf[iBufPos + i + 3] == 1
61
670M
          && i > 0) || (pBuf[iBufPos + i] == 0 && pBuf[iBufPos + i + 1] == 0 && pBuf[iBufPos + i + 2] == 1 && i > 0)) {
62
28.4M
        break;
63
28.4M
      }
64
670M
    }
65
28.5M
    iSliceSize = i;
66
28.5M
    if (iSliceSize < 4) {
67
7.80M
      if (iSliceSize == 0) {
68
        // I don't think this should happen but let's just avoid the hang
69
0
        goto label_cleanup;
70
0
      }
71
7.80M
      iBufPos += iSliceSize;
72
7.80M
      continue;
73
7.80M
    }
74
75
20.7M
    pDecoder->DecodeFrameNoDelay (pBuf.get() + iBufPos, iSliceSize, pData, &sDstBufInfo);
76
20.7M
    iBufPos += iSliceSize;
77
20.7M
  }
78
79
1.75M
label_cleanup:
80
1.75M
  pDecoder->Uninitialize ();
81
1.75M
  WelsDestroyDecoder (pDecoder);
82
83
1.75M
  return 0;
84
1.75M
}