/* * The copyright in this software is being made available under the 2-clauses * BSD License, included below. This software may be subject to other third * party and contributor rights, including patent rights, and no such rights * are granted under this license. * * Copyright (c) 2017, IntoPix SA * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS `AS IS' * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include "opj_includes.h" struct opj_sparse_array_int32 { OPJ_UINT32 width; OPJ_UINT32 height; OPJ_UINT32 block_width; OPJ_UINT32 block_height; OPJ_UINT32 block_count_hor; OPJ_UINT32 block_count_ver; OPJ_INT32** data_blocks; }; opj_sparse_array_int32_t* opj_sparse_array_int32_create(OPJ_UINT32 width, OPJ_UINT32 height, OPJ_UINT32 block_width, OPJ_UINT32 block_height) { opj_sparse_array_int32_t* sa; if (width == 0 || height == 0 || block_width == 0 || block_height == 0) { return NULL; } if (block_width > ((OPJ_UINT32)~0U) / block_height / sizeof(OPJ_INT32)) { return NULL; } sa = (opj_sparse_array_int32_t*) opj_calloc(1, sizeof(opj_sparse_array_int32_t)); sa->width = width; sa->height = height; sa->block_width = block_width; sa->block_height = block_height; sa->block_count_hor = opj_uint_ceildiv(width, block_width); sa->block_count_ver = opj_uint_ceildiv(height, block_height); if (sa->block_count_hor > ((OPJ_UINT32)~0U) / sa->block_count_ver) { opj_free(sa); return NULL; } sa->data_blocks = (OPJ_INT32**) opj_calloc(sizeof(OPJ_INT32*), (size_t) sa->block_count_hor * sa->block_count_ver); if (sa->data_blocks == NULL) { opj_free(sa); return NULL; } return sa; } void opj_sparse_array_int32_free(opj_sparse_array_int32_t* sa) { if (sa) { OPJ_UINT32 i; for (i = 0; i < sa->block_count_hor * sa->block_count_ver; i++) { if (sa->data_blocks[i]) { opj_free(sa->data_blocks[i]); } } opj_free(sa->data_blocks); opj_free(sa); } } OPJ_BOOL opj_sparse_array_is_region_valid(const opj_sparse_array_int32_t* sa, OPJ_UINT32 x0, OPJ_UINT32 y0, OPJ_UINT32 x1, OPJ_UINT32 y1) { return !(x0 >= sa->width || x1 <= x0 || x1 > sa->width || y0 >= sa->height || y1 <= y0 || y1 > sa->height); } static OPJ_BOOL opj_sparse_array_int32_read_or_write( const opj_sparse_array_int32_t* sa, OPJ_UINT32 x0, OPJ_UINT32 y0, OPJ_UINT32 x1, OPJ_UINT32 y1, OPJ_INT32* buf, OPJ_UINT32 buf_col_stride, OPJ_UINT32 buf_line_stride, OPJ_BOOL forgiving, OPJ_BOOL is_read_op) { OPJ_UINT32 y, block_y; OPJ_UINT32 y_incr = 0; const OPJ_UINT32 block_width = sa->block_width; if (!opj_sparse_array_is_region_valid(sa, x0, y0, x1, y1)) { return forgiving; } block_y = y0 / sa->block_height; for (y = y0; y < y1; block_y ++, y += y_incr) { OPJ_UINT32 x, block_x; OPJ_UINT32 x_incr = 0; OPJ_UINT32 block_y_offset; y_incr = (y == y0) ? sa->block_height - (y0 % sa->block_height) : sa->block_height; block_y_offset = sa->block_height - y_incr; y_incr = opj_uint_min(y_incr, y1 - y); block_x = x0 / block_width; for (x = x0; x < x1; block_x ++, x += x_incr) { OPJ_UINT32 j; OPJ_UINT32 block_x_offset; OPJ_INT32* src_block; x_incr = (x == x0) ? block_width - (x0 % block_width) : block_width; block_x_offset = block_width - x_incr; x_incr = opj_uint_min(x_incr, x1 - x); src_block = sa->data_blocks[block_y * sa->block_count_hor + block_x]; if (is_read_op) { if (src_block == NULL) { if (buf_col_stride == 1) { OPJ_INT32* dest_ptr = buf + (y - y0) * (OPJ_SIZE_T)buf_line_stride + (x - x0) * buf_col_stride; for (j = 0; j < y_incr; j++) { memset(dest_ptr, 0, sizeof(OPJ_INT32) * x_incr); dest_ptr += buf_line_stride; } } else { OPJ_INT32* dest_ptr = buf + (y - y0) * (OPJ_SIZE_T)buf_line_stride + (x - x0) * buf_col_stride; for (j = 0; j < y_incr; j++) { OPJ_UINT32 k; for (k = 0; k < x_incr; k++) { dest_ptr[k * buf_col_stride] = 0; } dest_ptr += buf_line_stride; } } } else { const OPJ_INT32* OPJ_RESTRICT src_ptr = src_block + block_y_offset * (OPJ_SIZE_T)block_width + block_x_offset; if (buf_col_stride == 1) { OPJ_INT32* OPJ_RESTRICT dest_ptr = buf + (y - y0) * (OPJ_SIZE_T)buf_line_stride + (x - x0) * buf_col_stride; if (x_incr == 4) { /* Same code as general branch, but the compiler */ /* can have an efficient memcpy() */ (void)(x_incr); /* trick to silent cppcheck duplicateBranch warning */ for (j = 0; j < y_incr; j++) { memcpy(dest_ptr, src_ptr, sizeof(OPJ_INT32) * x_incr); dest_ptr += buf_line_stride; src_ptr += block_width; } } else { for (j = 0; j < y_incr; j++) { memcpy(dest_ptr, src_ptr, sizeof(OPJ_INT32) * x_incr); dest_ptr += buf_line_stride; src_ptr += block_width; } } } else { OPJ_INT32* OPJ_RESTRICT dest_ptr = buf + (y - y0) * (OPJ_SIZE_T)buf_line_stride + (x - x0) * buf_col_stride; if (x_incr == 1) { for (j = 0; j < y_incr; j++) { *dest_ptr = *src_ptr; dest_ptr += buf_line_stride; src_ptr += block_width; } } else if (y_incr == 1 && buf_col_stride == 2) { OPJ_UINT32 k; for (k = 0; k < (x_incr & ~3U); k += 4) { dest_ptr[k * buf_col_stride] = src_ptr[k]; dest_ptr[(k + 1) * buf_col_stride] = src_ptr[k + 1]; dest_ptr[(k + 2) * buf_col_stride] = src_ptr[k + 2]; dest_ptr[(k + 3) * buf_col_stride] = src_ptr[k + 3]; } for (; k < x_incr; k++) { dest_ptr[k * buf_col_stride] = src_ptr[k]; } } else if (x_incr >= 8 && buf_col_stride == 8) { for (j = 0; j < y_incr; j++) { OPJ_UINT32 k; for (k = 0; k < (x_incr & ~3U); k += 4) { dest_ptr[k * buf_col_stride] = src_ptr[k]; dest_ptr[(k + 1) * buf_col_stride] = src_ptr[k + 1]; dest_ptr[(k + 2) * buf_col_stride] = src_ptr[k + 2]; dest_ptr[(k + 3) * buf_col_stride] = src_ptr[k + 3]; } for (; k < x_incr; k++) { dest_ptr[k * buf_col_stride] = src_ptr[k]; } dest_ptr += buf_line_stride; src_ptr += block_width; } } else { /* General case */ for (j = 0; j < y_incr; j++) { OPJ_UINT32 k; for (k = 0; k < x_incr; k++) { dest_ptr[k * buf_col_stride] = src_ptr[k]; } dest_ptr += buf_line_stride; src_ptr += block_width; } } } } } else { if (src_block == NULL) { src_block = (OPJ_INT32*) opj_calloc(1, (size_t) sa->block_width * sa->block_height * sizeof(OPJ_INT32)); if (src_block == NULL) { return OPJ_FALSE; } sa->data_blocks[block_y * sa->block_count_hor + block_x] = src_block; } if (buf_col_stride == 1) { OPJ_INT32* OPJ_RESTRICT dest_ptr = src_block + block_y_offset * (OPJ_SIZE_T)block_width + block_x_offset; const OPJ_INT32* OPJ_RESTRICT src_ptr = buf + (y - y0) * (OPJ_SIZE_T)buf_line_stride + (x - x0) * buf_col_stride; if (x_incr == 4) { /* Same code as general branch, but the compiler */ /* can have an efficient memcpy() */ (void)(x_incr); /* trick to silent cppcheck duplicateBranch warning */ for (j = 0; j < y_incr; j++) { memcpy(dest_ptr, src_ptr, sizeof(OPJ_INT32) * x_incr); dest_ptr += block_width; src_ptr += buf_line_stride; } } else { for (j = 0; j < y_incr; j++) { memcpy(dest_ptr, src_ptr, sizeof(OPJ_INT32) * x_incr); dest_ptr += block_width; src_ptr += buf_line_stride; } } } else { OPJ_INT32* OPJ_RESTRICT dest_ptr = src_block + block_y_offset * (OPJ_SIZE_T)block_width + block_x_offset; const OPJ_INT32* OPJ_RESTRICT src_ptr = buf + (y - y0) * (OPJ_SIZE_T)buf_line_stride + (x - x0) * buf_col_stride; if (x_incr == 1) { for (j = 0; j < y_incr; j++) { *dest_ptr = *src_ptr; src_ptr += buf_line_stride; dest_ptr += block_width; } } else if (x_incr >= 8 && buf_col_stride == 8) { for (j = 0; j < y_incr; j++) { OPJ_UINT32 k; for (k = 0; k < (x_incr & ~3U); k += 4) { dest_ptr[k] = src_ptr[k * buf_col_stride]; dest_ptr[k + 1] = src_ptr[(k + 1) * buf_col_stride]; dest_ptr[k + 2] = src_ptr[(k + 2) * buf_col_stride]; dest_ptr[k + 3] = src_ptr[(k + 3) * buf_col_stride]; } for (; k < x_incr; k++) { dest_ptr[k] = src_ptr[k * buf_col_stride]; } src_ptr += buf_line_stride; dest_ptr += block_width; } } else { /* General case */ for (j = 0; j < y_incr; j++) { OPJ_UINT32 k; for (k = 0; k < x_incr; k++) { dest_ptr[k] = src_ptr[k * buf_col_stride]; } src_ptr += buf_line_stride; dest_ptr += block_width; } } } } } } return OPJ_TRUE; } OPJ_BOOL opj_sparse_array_int32_read(const opj_sparse_array_int32_t* sa, OPJ_UINT32 x0, OPJ_UINT32 y0, OPJ_UINT32 x1, OPJ_UINT32 y1, OPJ_INT32* dest, OPJ_UINT32 dest_col_stride, OPJ_UINT32 dest_line_stride, OPJ_BOOL forgiving) { return opj_sparse_array_int32_read_or_write( (opj_sparse_array_int32_t*)sa, x0, y0, x1, y1, dest, dest_col_stride, dest_line_stride, forgiving, OPJ_TRUE); } OPJ_BOOL opj_sparse_array_int32_write(opj_sparse_array_int32_t* sa, OPJ_UINT32 x0, OPJ_UINT32 y0, OPJ_UINT32 x1, OPJ_UINT32 y1, const OPJ_INT32* src, OPJ_UINT32 src_col_stride, OPJ_UINT32 src_line_stride, OPJ_BOOL forgiving) { return opj_sparse_array_int32_read_or_write(sa, x0, y0, x1, y1, (OPJ_INT32*)src, src_col_stride, src_line_stride, forgiving, OPJ_FALSE); }