fix potential overflow when width * height * 4 >= (1<<32)

Mostly: avoid doing calculation like: ptr + j * stride
when stride is 'int'. Rather use size_t, or pointer increments (ptr += stride)
when possible.

BUG=webp:314

Change-Id: I81c684b515dd1ec4f601f32d50a6e821c4e46e20
(cherry picked from commit e2affacc35)
This commit is contained in:
Pascal Massimino 2016-10-10 11:48:39 +02:00 committed by James Zern
parent 883d41fb40
commit bb2336170b

View File

@ -20,6 +20,7 @@
#include "webp/encode.h" #include "webp/encode.h"
#include "webp/mux_types.h" #include "webp/mux_types.h"
#include "webp/format_constants.h"
#define GIF_TRANSPARENT_COLOR 0x00ffffff #define GIF_TRANSPARENT_COLOR 0x00ffffff
#define GIF_WHITE_COLOR 0xffffffff #define GIF_WHITE_COLOR 0xffffffff
@ -103,12 +104,19 @@ int GIFReadFrame(GifFileType* const gif, int transparent_index,
const GifImageDesc* const image_desc = &gif->Image; const GifImageDesc* const image_desc = &gif->Image;
uint32_t* dst = NULL; uint32_t* dst = NULL;
uint8_t* tmp = NULL; uint8_t* tmp = NULL;
int ok = 0; const GIFFrameRect rect = {
GIFFrameRect rect = {
image_desc->Left, image_desc->Top, image_desc->Width, image_desc->Height image_desc->Left, image_desc->Top, image_desc->Width, image_desc->Height
}; };
const uint64_t memory_needed = 4 * rect.width * (uint64_t)rect.height;
int ok = 0;
*gif_rect = rect; *gif_rect = rect;
if (memory_needed != (size_t)memory_needed ||
memory_needed > 4 * MAX_IMAGE_AREA) {
fprintf(stderr, "Image is too large (%d x %d).", rect.width, rect.height);
return 0;
}
// Use a view for the sub-picture: // Use a view for the sub-picture:
if (!WebPPictureView(picture, rect.x_offset, rect.y_offset, if (!WebPPictureView(picture, rect.x_offset, rect.y_offset,
rect.width, rect.height, &sub_image)) { rect.width, rect.height, &sub_image)) {
@ -132,15 +140,15 @@ int GIFReadFrame(GifFileType* const gif, int transparent_index,
y += interlace_jumps[pass]) { y += interlace_jumps[pass]) {
if (DGifGetLine(gif, tmp, rect.width) == GIF_ERROR) goto End; if (DGifGetLine(gif, tmp, rect.width) == GIF_ERROR) goto End;
Remap(gif, tmp, rect.width, transparent_index, Remap(gif, tmp, rect.width, transparent_index,
dst + y * sub_image.argb_stride); dst + y * (size_t)sub_image.argb_stride);
} }
} }
} else { // Non-interlaced image. } else { // Non-interlaced image.
int y; int y;
for (y = 0; y < rect.height; ++y) { uint32_t* ptr = dst;
for (y = 0; y < rect.height; ++y, ptr += sub_image.argb_stride) {
if (DGifGetLine(gif, tmp, rect.width) == GIF_ERROR) goto End; if (DGifGetLine(gif, tmp, rect.width) == GIF_ERROR) goto End;
Remap(gif, tmp, rect.width, transparent_index, Remap(gif, tmp, rect.width, transparent_index, ptr);
dst + y * sub_image.argb_stride);
} }
} }
ok = 1; ok = 1;
@ -216,13 +224,11 @@ int GIFReadMetadata(GifFileType* const gif, GifByteType** const buf,
static void ClearRectangle(WebPPicture* const picture, static void ClearRectangle(WebPPicture* const picture,
int left, int top, int width, int height) { int left, int top, int width, int height) {
int j; int i, j;
for (j = top; j < top + height; ++j) { const size_t stride = picture->argb_stride;
uint32_t* const dst = picture->argb + j * picture->argb_stride; uint32_t* dst = picture->argb + top * stride + left;
int i; for (j = 0; j < height; ++j, dst += stride) {
for (i = left; i < left + width; ++i) { for (i = 0; i < width; ++i) dst[i] = GIF_TRANSPARENT_COLOR;
dst[i] = GIF_TRANSPARENT_COLOR;
}
} }
} }
@ -246,29 +252,31 @@ void GIFDisposeFrame(GIFDisposeMethod dispose, const GIFFrameRect* const rect,
if (dispose == GIF_DISPOSE_BACKGROUND) { if (dispose == GIF_DISPOSE_BACKGROUND) {
GIFClearPic(curr_canvas, rect); GIFClearPic(curr_canvas, rect);
} else if (dispose == GIF_DISPOSE_RESTORE_PREVIOUS) { } else if (dispose == GIF_DISPOSE_RESTORE_PREVIOUS) {
const int src_stride = prev_canvas->argb_stride; const size_t src_stride = prev_canvas->argb_stride;
const uint32_t* const src = const uint32_t* const src = prev_canvas->argb + rect->x_offset
prev_canvas->argb + rect->x_offset + rect->y_offset * src_stride; + rect->y_offset * src_stride;
const int dst_stride = curr_canvas->argb_stride; const size_t dst_stride = curr_canvas->argb_stride;
uint32_t* const dst = uint32_t* const dst = curr_canvas->argb + rect->x_offset
curr_canvas->argb + rect->x_offset + rect->y_offset * dst_stride; + rect->y_offset * dst_stride;
assert(prev_canvas != NULL); assert(prev_canvas != NULL);
WebPCopyPlane((uint8_t*)src, 4 * src_stride, (uint8_t*)dst, 4 * dst_stride, WebPCopyPlane((uint8_t*)src, (int)(4 * src_stride),
(uint8_t*)dst, (int)(4 * dst_stride),
4 * rect->width, rect->height); 4 * rect->width, rect->height);
} }
} }
void GIFBlendFrames(const WebPPicture* const src, void GIFBlendFrames(const WebPPicture* const src,
const GIFFrameRect* const rect, WebPPicture* const dst) { const GIFFrameRect* const rect, WebPPicture* const dst) {
int j; int i, j;
const size_t src_stride = src->argb_stride;
const size_t dst_stride = dst->argb_stride;
assert(src->width == dst->width && src->height == dst->height); assert(src->width == dst->width && src->height == dst->height);
for (j = rect->y_offset; j < rect->y_offset + rect->height; ++j) { for (j = rect->y_offset; j < rect->y_offset + rect->height; ++j) {
int i;
for (i = rect->x_offset; i < rect->x_offset + rect->width; ++i) { for (i = rect->x_offset; i < rect->x_offset + rect->width; ++i) {
const uint32_t src_pixel = src->argb[j * src->argb_stride + i]; const uint32_t src_pixel = src->argb[j * src_stride + i];
const int src_alpha = src_pixel >> 24; const int src_alpha = src_pixel >> 24;
if (src_alpha != 0) { if (src_alpha != 0) {
dst->argb[j * dst->argb_stride + i] = src_pixel; dst->argb[j * dst_stride + i] = src_pixel;
} }
} }
} }