blob: 4c188149689091cf298f0e91a13562163970c2b8 [file] [log] [blame] [edit]
/******************************************************************************
*
* Copyright (C) 2020 The Android Open Source Project
*
* 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.
*
*****************************************************************************
* Originally developed and contributed by Ittiam Systems Pvt. Ltd, Bangalore
*/
#include <fuzzer/FuzzedDataProvider.h>
#include <stdlib.h>
#include <utils/String8.h>
#include "FLAC/stream_decoder.h"
constexpr FLAC__MetadataType kMetadataTypes[8] = {
FLAC__METADATA_TYPE_PICTURE,
FLAC__METADATA_TYPE_STREAMINFO,
FLAC__METADATA_TYPE_PADDING,
FLAC__METADATA_TYPE_APPLICATION,
FLAC__METADATA_TYPE_SEEKTABLE,
FLAC__METADATA_TYPE_VORBIS_COMMENT,
FLAC__METADATA_TYPE_CUESHEET,
FLAC__METADATA_TYPE_UNDEFINED,
};
const char *kMetadataIDs[3] = {
"aiff",
"riff",
"w64",
};
// First four bytes are always "fLaC" in ASCII format.
#define FIRST_ENCODED_BYTE_OFFSET 4
class Codec {
public:
Codec(const uint8_t* data, size_t size) : mFdp(data, size) {
mBuffer = data;
mBufferLen = size;
};
~Codec() { deInitDecoder(); }
bool initDecoder();
void decodeFrames();
void deInitDecoder();
private:
FLAC__StreamDecoder *mDecoder = nullptr;
const uint8_t *mBuffer = nullptr;
size_t mBufferLen = 0;
size_t mBufferPos = 0;
FLAC__StreamDecoderReadStatus readCallback(FLAC__byte buffer[], size_t *bytes);
FLAC__StreamDecoderWriteStatus writeCallback(const FLAC__Frame *frame,
const FLAC__int32 *const buffer[]);
void errorCallback(FLAC__StreamDecoderErrorStatus status);
void metadataCallback(const FLAC__StreamMetadata *metadata);
FuzzedDataProvider mFdp;
};
bool Codec::initDecoder() {
mDecoder = FLAC__stream_decoder_new();
if (!mDecoder) {
return false;
}
FLAC__stream_decoder_set_metadata_ignore_all(mDecoder);
FLAC__stream_decoder_set_md5_checking(mDecoder, true);
// read_callback, write_callback, error_callback and metadata_callback cannot be nullptrs
static auto read_callback = [](const FLAC__StreamDecoder *, FLAC__byte buffer[], size_t *bytes,
void *client_data) -> FLAC__StreamDecoderReadStatus {
Codec *client = reinterpret_cast<Codec *>(client_data);
return client->readCallback(buffer, bytes);
};
static auto write_callback = [](const FLAC__StreamDecoder *, const FLAC__Frame *frame,
const FLAC__int32 *const buffer[],
void *client_data) -> FLAC__StreamDecoderWriteStatus {
Codec *client = reinterpret_cast<Codec *>(client_data);
return client->writeCallback(frame, buffer);
};
static auto error_callback = [](const FLAC__StreamDecoder *,
FLAC__StreamDecoderErrorStatus status, void *client_data) {
Codec *client = reinterpret_cast<Codec *>(client_data);
client->errorCallback(status);
};
if (mFdp.ConsumeBool()) {
FLAC__stream_decoder_set_metadata_ignore(mDecoder, mFdp.PickValueInArray(kMetadataTypes));
}
if (mFdp.ConsumeBool()) {
FLAC__stream_decoder_skip_single_frame(mDecoder);
}
if (mFdp.ConsumeBool()) {
FLAC__MetadataType metadataType = mFdp.PickValueInArray(kMetadataTypes);
FLAC__stream_decoder_set_metadata_respond(mDecoder, metadataType);
}
else {
FLAC__byte* metadataIgnoreID = (FLAC__byte*)mFdp.PickValueInArray(kMetadataIDs);
FLAC__byte* metadataRespondID = (FLAC__byte*)mFdp.PickValueInArray(kMetadataIDs);
FLAC__stream_decoder_set_metadata_ignore_application(mDecoder, metadataIgnoreID);
FLAC__stream_decoder_set_metadata_respond_application(mDecoder, metadataRespondID);
}
static auto metadata_callback = [](const FLAC__StreamDecoder *,
const FLAC__StreamMetadata *metadata, void *client_data) {
Codec *client = reinterpret_cast<Codec *>(client_data);
client->metadataCallback(metadata);
};
void *client_data = reinterpret_cast<void *>(this);
FLAC__StreamDecoderInitStatus initStatus = FLAC__stream_decoder_init_stream(
mDecoder, read_callback, nullptr, nullptr, nullptr, nullptr, write_callback,
metadata_callback, error_callback, client_data);
if (initStatus != FLAC__STREAM_DECODER_INIT_STATUS_OK) {
return false;
}
if (mFdp.ConsumeBool()) {
FLAC__stream_decoder_set_metadata_respond_all(mDecoder);
}
if (mFdp.ConsumeBool()) {
FLAC__stream_decoder_seek_absolute(mDecoder, (FLAC__uint64)mFdp.ConsumeIntegral<uint64_t>());
}
return true;
}
FLAC__StreamDecoderReadStatus Codec::readCallback(FLAC__byte buffer[], size_t *bytes) {
if (!mBuffer || mBufferLen == 0) {
*bytes = 0;
return FLAC__STREAM_DECODER_READ_STATUS_ABORT;
}
size_t bytesRequested = *bytes;
if (bytesRequested > mBufferLen - mBufferPos) {
bytesRequested = mBufferLen - mBufferPos;
}
memcpy(buffer, mBuffer + mBufferPos, bytesRequested);
mBufferPos += bytesRequested;
*bytes = bytesRequested;
return (bytesRequested == 0 ? FLAC__STREAM_DECODER_READ_STATUS_END_OF_STREAM
: FLAC__STREAM_DECODER_READ_STATUS_CONTINUE);
}
FLAC__StreamDecoderWriteStatus Codec::writeCallback(const FLAC__Frame *frame,
const FLAC__int32 *const buffer[]) {
(void)frame;
(void)buffer;
return FLAC__STREAM_DECODER_WRITE_STATUS_CONTINUE;
}
void Codec::errorCallback(FLAC__StreamDecoderErrorStatus status) {
(void)status;
return;
}
void Codec::metadataCallback(const FLAC__StreamMetadata *metadata) {
(void)metadata;
return;
}
void Codec::decodeFrames() {
if (mFdp.ConsumeBool()) {
if (!FLAC__stream_decoder_process_until_end_of_metadata(mDecoder)) {
return;
}
while(mBufferPos <= mBufferLen) {
FLAC__stream_decoder_process_single(mDecoder);
if (FLAC__STREAM_DECODER_END_OF_STREAM == FLAC__stream_decoder_get_state(mDecoder)) {
return;
}
}
} else {
FLAC__stream_decoder_process_until_end_of_stream(mDecoder);
}
FLAC__stream_decoder_finish(mDecoder);
}
void Codec::deInitDecoder() {
if (mDecoder != nullptr) {
FLAC__stream_decoder_delete(mDecoder);
}
}
extern "C" int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
if (size == 0) {
return 0;
}
auto codec = std::make_unique<Codec>(data, size);
if (codec->initDecoder()) {
codec->decodeFrames();
}
return 0;
}