Path: blob/master/thirdparty/libwebp/src/mux/muxread.c
9912 views
// Copyright 2011 Google Inc. All Rights Reserved.1//2// Use of this source code is governed by a BSD-style license3// that can be found in the COPYING file in the root of the source4// tree. An additional intellectual property rights grant can be found5// in the file PATENTS. All contributing project authors may6// be found in the AUTHORS file in the root of the source tree.7// -----------------------------------------------------------------------------8//9// Read APIs for mux.10//11// Authors: Urvang ([email protected])12// Vikas ([email protected])1314#include <assert.h>15#include "src/mux/muxi.h"16#include "src/utils/utils.h"1718//------------------------------------------------------------------------------19// Helper method(s).2021// Handy MACRO.22#define SWITCH_ID_LIST(INDEX, LIST) \23do { \24if (idx == (INDEX)) { \25const WebPChunk* const chunk = ChunkSearchList((LIST), nth, \26kChunks[(INDEX)].tag); \27if (chunk) { \28*data = chunk->data_; \29return WEBP_MUX_OK; \30} else { \31return WEBP_MUX_NOT_FOUND; \32} \33} \34} while (0)3536static WebPMuxError MuxGet(const WebPMux* const mux, CHUNK_INDEX idx,37uint32_t nth, WebPData* const data) {38assert(mux != NULL);39assert(idx != IDX_LAST_CHUNK);40assert(!IsWPI(kChunks[idx].id));41WebPDataInit(data);4243SWITCH_ID_LIST(IDX_VP8X, mux->vp8x_);44SWITCH_ID_LIST(IDX_ICCP, mux->iccp_);45SWITCH_ID_LIST(IDX_ANIM, mux->anim_);46SWITCH_ID_LIST(IDX_EXIF, mux->exif_);47SWITCH_ID_LIST(IDX_XMP, mux->xmp_);48assert(idx != IDX_UNKNOWN);49return WEBP_MUX_NOT_FOUND;50}51#undef SWITCH_ID_LIST5253// Fill the chunk with the given data (includes chunk header bytes), after some54// verifications.55static WebPMuxError ChunkVerifyAndAssign(WebPChunk* chunk,56const uint8_t* data, size_t data_size,57size_t riff_size, int copy_data) {58uint32_t chunk_size;59WebPData chunk_data;6061// Correctness checks.62if (data_size < CHUNK_HEADER_SIZE) return WEBP_MUX_NOT_ENOUGH_DATA;63chunk_size = GetLE32(data + TAG_SIZE);64if (chunk_size > MAX_CHUNK_PAYLOAD) return WEBP_MUX_BAD_DATA;6566{67const size_t chunk_disk_size = SizeWithPadding(chunk_size);68if (chunk_disk_size > riff_size) return WEBP_MUX_BAD_DATA;69if (chunk_disk_size > data_size) return WEBP_MUX_NOT_ENOUGH_DATA;70}7172// Data assignment.73chunk_data.bytes = data + CHUNK_HEADER_SIZE;74chunk_data.size = chunk_size;75return ChunkAssignData(chunk, &chunk_data, copy_data, GetLE32(data + 0));76}7778int MuxImageFinalize(WebPMuxImage* const wpi) {79const WebPChunk* const img = wpi->img_;80const WebPData* const image = &img->data_;81const int is_lossless = (img->tag_ == kChunks[IDX_VP8L].tag);82int w, h;83int vp8l_has_alpha = 0;84const int ok = is_lossless ?85VP8LGetInfo(image->bytes, image->size, &w, &h, &vp8l_has_alpha) :86VP8GetInfo(image->bytes, image->size, image->size, &w, &h);87assert(img != NULL);88if (ok) {89// Ignore ALPH chunk accompanying VP8L.90if (is_lossless && (wpi->alpha_ != NULL)) {91ChunkDelete(wpi->alpha_);92wpi->alpha_ = NULL;93}94wpi->width_ = w;95wpi->height_ = h;96wpi->has_alpha_ = vp8l_has_alpha || (wpi->alpha_ != NULL);97}98return ok;99}100101static int MuxImageParse(const WebPChunk* const chunk, int copy_data,102WebPMuxImage* const wpi) {103const uint8_t* bytes = chunk->data_.bytes;104size_t size = chunk->data_.size;105const uint8_t* const last = (bytes == NULL) ? NULL : bytes + size;106WebPChunk subchunk;107size_t subchunk_size;108WebPChunk** unknown_chunk_list = &wpi->unknown_;109ChunkInit(&subchunk);110111assert(chunk->tag_ == kChunks[IDX_ANMF].tag);112assert(!wpi->is_partial_);113114// ANMF.115{116const size_t hdr_size = ANMF_CHUNK_SIZE;117const WebPData temp = { bytes, hdr_size };118// Each of ANMF chunk contain a header at the beginning. So, its size should119// be at least 'hdr_size'.120if (size < hdr_size) goto Fail;121if (ChunkAssignData(&subchunk, &temp, copy_data,122chunk->tag_) != WEBP_MUX_OK) {123goto Fail;124}125}126if (ChunkSetHead(&subchunk, &wpi->header_) != WEBP_MUX_OK) goto Fail;127wpi->is_partial_ = 1; // Waiting for ALPH and/or VP8/VP8L chunks.128129// Rest of the chunks.130subchunk_size = ChunkDiskSize(&subchunk) - CHUNK_HEADER_SIZE;131bytes += subchunk_size;132size -= subchunk_size;133134while (bytes != last) {135ChunkInit(&subchunk);136if (ChunkVerifyAndAssign(&subchunk, bytes, size, size,137copy_data) != WEBP_MUX_OK) {138goto Fail;139}140switch (ChunkGetIdFromTag(subchunk.tag_)) {141case WEBP_CHUNK_ALPHA:142if (wpi->alpha_ != NULL) goto Fail; // Consecutive ALPH chunks.143if (ChunkSetHead(&subchunk, &wpi->alpha_) != WEBP_MUX_OK) goto Fail;144wpi->is_partial_ = 1; // Waiting for a VP8 chunk.145break;146case WEBP_CHUNK_IMAGE:147if (wpi->img_ != NULL) goto Fail; // Only 1 image chunk allowed.148if (ChunkSetHead(&subchunk, &wpi->img_) != WEBP_MUX_OK) goto Fail;149if (!MuxImageFinalize(wpi)) goto Fail;150wpi->is_partial_ = 0; // wpi is completely filled.151break;152case WEBP_CHUNK_UNKNOWN:153if (wpi->is_partial_) {154goto Fail; // Encountered an unknown chunk155// before some image chunks.156}157if (ChunkAppend(&subchunk, &unknown_chunk_list) != WEBP_MUX_OK) {158goto Fail;159}160break;161default:162goto Fail;163}164subchunk_size = ChunkDiskSize(&subchunk);165bytes += subchunk_size;166size -= subchunk_size;167}168if (wpi->is_partial_) goto Fail;169return 1;170171Fail:172ChunkRelease(&subchunk);173return 0;174}175176//------------------------------------------------------------------------------177// Create a mux object from WebP-RIFF data.178179WebPMux* WebPMuxCreateInternal(const WebPData* bitstream, int copy_data,180int version) {181size_t riff_size;182uint32_t tag;183const uint8_t* end;184WebPMux* mux = NULL;185WebPMuxImage* wpi = NULL;186const uint8_t* data;187size_t size;188WebPChunk chunk;189// Stores the end of the chunk lists so that it is faster to append data to190// their ends.191WebPChunk** chunk_list_ends[WEBP_CHUNK_NIL + 1] = { NULL };192ChunkInit(&chunk);193194if (WEBP_ABI_IS_INCOMPATIBLE(version, WEBP_MUX_ABI_VERSION)) {195return NULL; // version mismatch196}197if (bitstream == NULL) return NULL;198199data = bitstream->bytes;200size = bitstream->size;201202if (data == NULL) return NULL;203if (size < RIFF_HEADER_SIZE + CHUNK_HEADER_SIZE) return NULL;204if (GetLE32(data + 0) != MKFOURCC('R', 'I', 'F', 'F') ||205GetLE32(data + CHUNK_HEADER_SIZE) != MKFOURCC('W', 'E', 'B', 'P')) {206return NULL;207}208209mux = WebPMuxNew();210if (mux == NULL) return NULL;211212tag = GetLE32(data + RIFF_HEADER_SIZE);213if (tag != kChunks[IDX_VP8].tag &&214tag != kChunks[IDX_VP8L].tag &&215tag != kChunks[IDX_VP8X].tag) {216goto Err; // First chunk should be VP8, VP8L or VP8X.217}218219riff_size = GetLE32(data + TAG_SIZE);220if (riff_size > MAX_CHUNK_PAYLOAD) goto Err;221222// Note this padding is historical and differs from demux.c which does not223// pad the file size.224riff_size = SizeWithPadding(riff_size);225// Make sure the whole RIFF header is available.226if (riff_size < RIFF_HEADER_SIZE) goto Err;227if (riff_size > size) goto Err;228// There's no point in reading past the end of the RIFF chunk. Note riff_size229// includes CHUNK_HEADER_SIZE after SizeWithPadding().230if (size > riff_size) {231size = riff_size;232}233234end = data + size;235data += RIFF_HEADER_SIZE;236size -= RIFF_HEADER_SIZE;237238wpi = (WebPMuxImage*)WebPSafeMalloc(1ULL, sizeof(*wpi));239if (wpi == NULL) goto Err;240MuxImageInit(wpi);241242// Loop over chunks.243while (data != end) {244size_t data_size;245WebPChunkId id;246if (ChunkVerifyAndAssign(&chunk, data, size, riff_size,247copy_data) != WEBP_MUX_OK) {248goto Err;249}250data_size = ChunkDiskSize(&chunk);251id = ChunkGetIdFromTag(chunk.tag_);252switch (id) {253case WEBP_CHUNK_ALPHA:254if (wpi->alpha_ != NULL) goto Err; // Consecutive ALPH chunks.255if (ChunkSetHead(&chunk, &wpi->alpha_) != WEBP_MUX_OK) goto Err;256wpi->is_partial_ = 1; // Waiting for a VP8 chunk.257break;258case WEBP_CHUNK_IMAGE:259if (ChunkSetHead(&chunk, &wpi->img_) != WEBP_MUX_OK) goto Err;260if (!MuxImageFinalize(wpi)) goto Err;261wpi->is_partial_ = 0; // wpi is completely filled.262PushImage:263// Add this to mux->images_ list.264if (MuxImagePush(wpi, &mux->images_) != WEBP_MUX_OK) goto Err;265MuxImageInit(wpi); // Reset for reading next image.266break;267case WEBP_CHUNK_ANMF:268if (wpi->is_partial_) goto Err; // Previous wpi is still incomplete.269if (!MuxImageParse(&chunk, copy_data, wpi)) goto Err;270ChunkRelease(&chunk);271goto PushImage;272default: // A non-image chunk.273if (wpi->is_partial_) goto Err; // Encountered a non-image chunk before274// getting all chunks of an image.275if (chunk_list_ends[id] == NULL) {276chunk_list_ends[id] =277MuxGetChunkListFromId(mux, id); // List to add this chunk.278}279if (ChunkAppend(&chunk, &chunk_list_ends[id]) != WEBP_MUX_OK) goto Err;280if (id == WEBP_CHUNK_VP8X) { // grab global specs281if (data_size < CHUNK_HEADER_SIZE + VP8X_CHUNK_SIZE) goto Err;282mux->canvas_width_ = GetLE24(data + 12) + 1;283mux->canvas_height_ = GetLE24(data + 15) + 1;284}285break;286}287data += data_size;288size -= data_size;289ChunkInit(&chunk);290}291292// Incomplete image.293if (wpi->is_partial_) goto Err;294295// Validate mux if complete.296if (MuxValidate(mux) != WEBP_MUX_OK) goto Err;297298MuxImageDelete(wpi);299return mux; // All OK;300301Err: // Something bad happened.302ChunkRelease(&chunk);303MuxImageDelete(wpi);304WebPMuxDelete(mux);305return NULL;306}307308//------------------------------------------------------------------------------309// Get API(s).310311// Validates that the given mux has a single image.312static WebPMuxError ValidateForSingleImage(const WebPMux* const mux) {313const int num_images = MuxImageCount(mux->images_, WEBP_CHUNK_IMAGE);314const int num_frames = MuxImageCount(mux->images_, WEBP_CHUNK_ANMF);315316if (num_images == 0) {317// No images in mux.318return WEBP_MUX_NOT_FOUND;319} else if (num_images == 1 && num_frames == 0) {320// Valid case (single image).321return WEBP_MUX_OK;322} else {323// Frame case OR an invalid mux.324return WEBP_MUX_INVALID_ARGUMENT;325}326}327328// Get the canvas width, height and flags after validating that VP8X/VP8/VP8L329// chunk and canvas size are valid.330static WebPMuxError MuxGetCanvasInfo(const WebPMux* const mux,331int* width, int* height, uint32_t* flags) {332int w, h;333uint32_t f = 0;334WebPData data;335assert(mux != NULL);336337// Check if VP8X chunk is present.338if (MuxGet(mux, IDX_VP8X, 1, &data) == WEBP_MUX_OK) {339if (data.size < VP8X_CHUNK_SIZE) return WEBP_MUX_BAD_DATA;340f = GetLE32(data.bytes + 0);341w = GetLE24(data.bytes + 4) + 1;342h = GetLE24(data.bytes + 7) + 1;343} else {344const WebPMuxImage* const wpi = mux->images_;345// Grab user-forced canvas size as default.346w = mux->canvas_width_;347h = mux->canvas_height_;348if (w == 0 && h == 0 && ValidateForSingleImage(mux) == WEBP_MUX_OK) {349// single image and not forced canvas size => use dimension of first frame350assert(wpi != NULL);351w = wpi->width_;352h = wpi->height_;353}354if (wpi != NULL) {355if (wpi->has_alpha_) f |= ALPHA_FLAG;356}357}358if (w * (uint64_t)h >= MAX_IMAGE_AREA) return WEBP_MUX_BAD_DATA;359360if (width != NULL) *width = w;361if (height != NULL) *height = h;362if (flags != NULL) *flags = f;363return WEBP_MUX_OK;364}365366WebPMuxError WebPMuxGetCanvasSize(const WebPMux* mux, int* width, int* height) {367if (mux == NULL || width == NULL || height == NULL) {368return WEBP_MUX_INVALID_ARGUMENT;369}370return MuxGetCanvasInfo(mux, width, height, NULL);371}372373WebPMuxError WebPMuxGetFeatures(const WebPMux* mux, uint32_t* flags) {374if (mux == NULL || flags == NULL) return WEBP_MUX_INVALID_ARGUMENT;375return MuxGetCanvasInfo(mux, NULL, NULL, flags);376}377378static uint8_t* EmitVP8XChunk(uint8_t* const dst, int width,379int height, uint32_t flags) {380const size_t vp8x_size = CHUNK_HEADER_SIZE + VP8X_CHUNK_SIZE;381assert(width >= 1 && height >= 1);382assert(width <= MAX_CANVAS_SIZE && height <= MAX_CANVAS_SIZE);383assert(width * (uint64_t)height < MAX_IMAGE_AREA);384PutLE32(dst, MKFOURCC('V', 'P', '8', 'X'));385PutLE32(dst + TAG_SIZE, VP8X_CHUNK_SIZE);386PutLE32(dst + CHUNK_HEADER_SIZE, flags);387PutLE24(dst + CHUNK_HEADER_SIZE + 4, width - 1);388PutLE24(dst + CHUNK_HEADER_SIZE + 7, height - 1);389return dst + vp8x_size;390}391392// Assemble a single image WebP bitstream from 'wpi'.393static WebPMuxError SynthesizeBitstream(const WebPMuxImage* const wpi,394WebPData* const bitstream) {395uint8_t* dst;396397// Allocate data.398const int need_vp8x = (wpi->alpha_ != NULL);399const size_t vp8x_size = need_vp8x ? CHUNK_HEADER_SIZE + VP8X_CHUNK_SIZE : 0;400const size_t alpha_size = need_vp8x ? ChunkDiskSize(wpi->alpha_) : 0;401// Note: No need to output ANMF chunk for a single image.402const size_t size = RIFF_HEADER_SIZE + vp8x_size + alpha_size +403ChunkDiskSize(wpi->img_);404uint8_t* const data = (uint8_t*)WebPSafeMalloc(1ULL, size);405if (data == NULL) return WEBP_MUX_MEMORY_ERROR;406407// There should be at most one alpha_ chunk and exactly one img_ chunk.408assert(wpi->alpha_ == NULL || wpi->alpha_->next_ == NULL);409assert(wpi->img_ != NULL && wpi->img_->next_ == NULL);410411// Main RIFF header.412dst = MuxEmitRiffHeader(data, size);413414if (need_vp8x) {415dst = EmitVP8XChunk(dst, wpi->width_, wpi->height_, ALPHA_FLAG); // VP8X.416dst = ChunkListEmit(wpi->alpha_, dst); // ALPH.417}418419// Bitstream.420dst = ChunkListEmit(wpi->img_, dst);421assert(dst == data + size);422423// Output.424bitstream->bytes = data;425bitstream->size = size;426return WEBP_MUX_OK;427}428429WebPMuxError WebPMuxGetChunk(const WebPMux* mux, const char fourcc[4],430WebPData* chunk_data) {431CHUNK_INDEX idx;432if (mux == NULL || fourcc == NULL || chunk_data == NULL) {433return WEBP_MUX_INVALID_ARGUMENT;434}435idx = ChunkGetIndexFromFourCC(fourcc);436assert(idx != IDX_LAST_CHUNK);437if (IsWPI(kChunks[idx].id)) { // An image chunk.438return WEBP_MUX_INVALID_ARGUMENT;439} else if (idx != IDX_UNKNOWN) { // A known chunk type.440return MuxGet(mux, idx, 1, chunk_data);441} else { // An unknown chunk type.442const WebPChunk* const chunk =443ChunkSearchList(mux->unknown_, 1, ChunkGetTagFromFourCC(fourcc));444if (chunk == NULL) return WEBP_MUX_NOT_FOUND;445*chunk_data = chunk->data_;446return WEBP_MUX_OK;447}448}449450static WebPMuxError MuxGetImageInternal(const WebPMuxImage* const wpi,451WebPMuxFrameInfo* const info) {452// Set some defaults for unrelated fields.453info->x_offset = 0;454info->y_offset = 0;455info->duration = 1;456info->dispose_method = WEBP_MUX_DISPOSE_NONE;457info->blend_method = WEBP_MUX_BLEND;458// Extract data for related fields.459info->id = ChunkGetIdFromTag(wpi->img_->tag_);460return SynthesizeBitstream(wpi, &info->bitstream);461}462463static WebPMuxError MuxGetFrameInternal(const WebPMuxImage* const wpi,464WebPMuxFrameInfo* const frame) {465const int is_frame = (wpi->header_->tag_ == kChunks[IDX_ANMF].tag);466const WebPData* frame_data;467if (!is_frame) return WEBP_MUX_INVALID_ARGUMENT;468assert(wpi->header_ != NULL); // Already checked by WebPMuxGetFrame().469// Get frame chunk.470frame_data = &wpi->header_->data_;471if (frame_data->size < kChunks[IDX_ANMF].size) return WEBP_MUX_BAD_DATA;472// Extract info.473frame->x_offset = 2 * GetLE24(frame_data->bytes + 0);474frame->y_offset = 2 * GetLE24(frame_data->bytes + 3);475{476const uint8_t bits = frame_data->bytes[15];477frame->duration = GetLE24(frame_data->bytes + 12);478frame->dispose_method =479(bits & 1) ? WEBP_MUX_DISPOSE_BACKGROUND : WEBP_MUX_DISPOSE_NONE;480frame->blend_method = (bits & 2) ? WEBP_MUX_NO_BLEND : WEBP_MUX_BLEND;481}482frame->id = ChunkGetIdFromTag(wpi->header_->tag_);483return SynthesizeBitstream(wpi, &frame->bitstream);484}485486WebPMuxError WebPMuxGetFrame(487const WebPMux* mux, uint32_t nth, WebPMuxFrameInfo* frame) {488WebPMuxError err;489WebPMuxImage* wpi;490491if (mux == NULL || frame == NULL) {492return WEBP_MUX_INVALID_ARGUMENT;493}494495// Get the nth WebPMuxImage.496err = MuxImageGetNth((const WebPMuxImage**)&mux->images_, nth, &wpi);497if (err != WEBP_MUX_OK) return err;498499// Get frame info.500if (wpi->header_ == NULL) {501return MuxGetImageInternal(wpi, frame);502} else {503return MuxGetFrameInternal(wpi, frame);504}505}506507WebPMuxError WebPMuxGetAnimationParams(const WebPMux* mux,508WebPMuxAnimParams* params) {509WebPData anim;510WebPMuxError err;511512if (mux == NULL || params == NULL) return WEBP_MUX_INVALID_ARGUMENT;513514err = MuxGet(mux, IDX_ANIM, 1, &anim);515if (err != WEBP_MUX_OK) return err;516if (anim.size < kChunks[WEBP_CHUNK_ANIM].size) return WEBP_MUX_BAD_DATA;517params->bgcolor = GetLE32(anim.bytes);518params->loop_count = GetLE16(anim.bytes + 4);519520return WEBP_MUX_OK;521}522523// Get chunk index from chunk id. Returns IDX_NIL if not found.524static CHUNK_INDEX ChunkGetIndexFromId(WebPChunkId id) {525int i;526for (i = 0; kChunks[i].id != WEBP_CHUNK_NIL; ++i) {527if (id == kChunks[i].id) return (CHUNK_INDEX)i;528}529return IDX_NIL;530}531532// Count number of chunks matching 'tag' in the 'chunk_list'.533// If tag == NIL_TAG, any tag will be matched.534static int CountChunks(const WebPChunk* const chunk_list, uint32_t tag) {535int count = 0;536const WebPChunk* current;537for (current = chunk_list; current != NULL; current = current->next_) {538if (tag == NIL_TAG || current->tag_ == tag) {539count++; // Count chunks whose tags match.540}541}542return count;543}544545WebPMuxError WebPMuxNumChunks(const WebPMux* mux,546WebPChunkId id, int* num_elements) {547if (mux == NULL || num_elements == NULL) {548return WEBP_MUX_INVALID_ARGUMENT;549}550551if (IsWPI(id)) {552*num_elements = MuxImageCount(mux->images_, id);553} else {554WebPChunk* const* chunk_list = MuxGetChunkListFromId(mux, id);555const CHUNK_INDEX idx = ChunkGetIndexFromId(id);556*num_elements = CountChunks(*chunk_list, kChunks[idx].tag);557}558559return WEBP_MUX_OK;560}561562//------------------------------------------------------------------------------563564565