使用C语言使用libFLAC解码数据

时间:2016-07-06 17:12:32

标签: c testing decoder flac fuzzing

我的目的是解码flac数据以进行测试。我将创建一个小存根,它具有一个将数据和大小作为输入参数的函数。没有必要制作任何输出文件,因为我只是要进行解码测试。我已经从libflac页面(https://xiph.org/flac/api/)阅读了一些示例和api文档。

现在这给了我:错误:初始化解码器:( null)因为FLAC__stream_decoder_init_stream被评论。评论的原因是我不知道如何正确使用它并进行解码工作。有什么建议和意见可以帮助我解码工作吗?

#include <stdio.h>
#include <stdlib.h>
#include "share/compat.h"
#include "FLAC/stream_decoder.h"

static void error_callback(const FLAC__StreamDecoder *decoder, FLAC__StreamDecoderErrorStatus status, void *data);
static FLAC__StreamDecoderWriteStatus write_callback(const FLAC__StreamDecoder *decoder, const FLAC__Frame *frame, const FLAC__int32 * const buffer[], void *data);
static FLAC__StreamDecoderReadStatus read_callback(const FLAC__StreamDecoder *decoder, FLAC__byte buffer[], size_t *bytes, void *data);

extern "C" int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
  FLAC__bool ok = true;
  FLAC__StreamDecoder *decoder = 0;
  FLAC__StreamDecoderInitStatus init_status;

  // init decoder
  if((decoder = FLAC__stream_decoder_new()) == NULL) {
        fprintf(stderr, "ERROR: allocating decoder\n");
        return 1;
    }

  (void)FLAC__stream_decoder_set_md5_checking(decoder, true);

  init_status = FLAC__stream_decoder_init_stream (  decoder, read_callback, /*seek_callback*/ NULL, /*tell_callback*/ NULL, /*length_callback*/ NULL, /*eof_callback*/ NULL, write_callback, /*metadata_callback*/ NULL, error_callback, data);

  if(init_status != FLAC__STREAM_DECODER_INIT_STATUS_OK) {
    fprintf(stderr, "ERROR: initializing decoder: %s\n", FLAC__StreamDecoderInitStatusString[init_status]);
    ok = false;
  }

  if(ok) {
    ok = FLAC__stream_decoder_process_until_end_of_stream(decoder);
    fprintf(stderr, "decoding: %s\n", ok? "succeeded" : "FAILED");
    fprintf(stderr, "   state: %s\n", FLAC__StreamDecoderStateString[FLAC__stream_decoder_get_state(decoder)]);
  }

  FLAC__stream_decoder_delete(decoder);

  return 0;
}

void error_callback(const FLAC__StreamDecoder *decoder, FLAC__StreamDecoderErrorStatus status, void *data)
{
  (void)decoder, (void)data;
  fprintf(stderr, "Got error callback: %s\n", FLAC__StreamDecoderErrorStatusString[status]);
}

更新: 添加回调函数后,我收到此错误:

flac_fuzzer.c:23:16: error: no matching function for call to 'FLAC__stream_decoder_init_stream'
init_status = FLAC__stream_decoder_init_stream (        decoder,
              ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
include/FLAC/stream_decoder.h:1073:40: note: candidate function not viable: no known conversion from 'const uint8_t *'
(aka 'const unsigned char *') to 'void *' for 10th argument; take the address of the argument with &
FLAC_API FLAC__StreamDecoderInitStatus FLAC__stream_decoder_init_stream(
                                       ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

2 个答案:

答案 0 :(得分:0)

您通常需要实现一个事件处理框架,类似于ffmpeg所做的事情,其中​​flac库使用回调挂钩到事件处理中。到目前为止,您实现的内容看起来像流程图的(最小)初始化部分,但是您缺少所有处理元素。

只是看着别人做了什么herehere,你需要使用像

这样的签名来实现回调函数

写回调示例:

 static FLAC__StreamDecoderWriteStatus flac_write_music_cb(
                                   const FLAC__StreamDecoder *decoder,
                                   const FLAC__Frame *frame,
                                   const FLAC__int32 *const buffer[],
                                   void *client_data)

错误回调示例:

 static void flac_error_music_cb(
                const FLAC__StreamDecoder *decoder,
                FLAC__StreamDecoderErrorStatus status,
                void *client_data)

您还应该看看如何读取flac数据,并实现支持功能。

答案 1 :(得分:0)

由于您使用了extern "C",因此您必须将其实际编译为C ++代码。现在,在C ++中,您需要使用const_cast<>来传递const uint8_t*参数,其中预期非常数(即void*)。